From ef4c5eb65e6e04fac4f0e1fa8bbeff56b75c1f98 Mon Sep 17 00:00:00 2001 From: twitter-team <> Date: Fri, 31 Mar 2023 17:36:31 -0500 Subject: [PATCH] Twitter Recommendation Algorithm Please note we have force-pushed a new initial commit in order to remove some publicly-available Twitter user information. Note that this process may be required in the future. --- .gitignore | 2 + COPYING | 661 +++ README.md | 39 + ann/src/main/java/com/twitter/ann/faiss/BUILD | 15 + .../com/twitter/ann/faiss/NativeUtils.java | 151 + .../ann/faiss/swig/AlignedTableFloat32.java | 98 + .../ann/faiss/swig/AlignedTableUint16.java | 98 + .../ann/faiss/swig/AlignedTableUint8.java | 98 + .../ann/faiss/swig/ArrayInvertedLists.java | 86 + .../ann/faiss/swig/AutoTuneCriterion.java | 89 + .../java/com/twitter/ann/faiss/swig/BUILD | 26 + .../ann/faiss/swig/BitstringReader.java | 72 + .../ann/faiss/swig/BitstringWriter.java | 72 + .../twitter/ann/faiss/swig/BufferList.java | 80 + .../twitter/ann/faiss/swig/ByteVector.java | 76 + .../ann/faiss/swig/ByteVectorVector.java | 76 + .../ann/faiss/swig/CenteringTransform.java | 68 + .../twitter/ann/faiss/swig/CharVector.java | 75 + .../twitter/ann/faiss/swig/Clustering.java | 101 + .../twitter/ann/faiss/swig/Clustering1D.java | 51 + .../faiss/swig/ClusteringIterationStats.java | 83 + .../ann/faiss/swig/ClusteringParameters.java | 131 + .../ann/faiss/swig/DistanceComputer.java | 47 + .../twitter/ann/faiss/swig/DoubleVector.java | 76 + .../twitter/ann/faiss/swig/FloatVector.java | 76 + .../ann/faiss/swig/FloatVectorVector.java | 76 + .../ann/faiss/swig/GenHammingComputer16.java | 63 + .../ann/faiss/swig/GenHammingComputer32.java | 79 + .../ann/faiss/swig/GenHammingComputer8.java | 55 + .../ann/faiss/swig/GenHammingComputerM8.java | 64 + .../java/com/twitter/ann/faiss/swig/HNSW.java | 437 ++ .../com/twitter/ann/faiss/swig/HNSWStats.java | 111 + .../ann/faiss/swig/HStackInvertedLists.java | 86 + .../ann/faiss/swig/HammingComputer16.java | 71 + .../ann/faiss/swig/HammingComputer20.java | 79 + .../ann/faiss/swig/HammingComputer32.java | 87 + .../ann/faiss/swig/HammingComputer4.java | 63 + .../ann/faiss/swig/HammingComputer64.java | 119 + .../ann/faiss/swig/HammingComputer8.java | 63 + .../faiss/swig/HammingComputerDefault.java | 80 + .../ann/faiss/swig/HammingComputerM4.java | 72 + .../ann/faiss/swig/HammingComputerM8.java | 72 + .../twitter/ann/faiss/swig/IDSelector.java | 43 + .../ann/faiss/swig/IDSelectorArray.java | 63 + .../ann/faiss/swig/IDSelectorBatch.java | 63 + .../ann/faiss/swig/IDSelectorRange.java | 63 + .../com/twitter/ann/faiss/swig/ITQMatrix.java | 76 + .../twitter/ann/faiss/swig/ITQTransform.java | 106 + .../ann/faiss/swig/IVFPQSearchParameters.java | 59 + .../ann/faiss/swig/IVFSearchParameters.java | 59 + .../com/twitter/ann/faiss/swig/Index.java | 165 + .../twitter/ann/faiss/swig/Index2Layer.java | 114 + .../twitter/ann/faiss/swig/IndexBinary.java | 139 + .../ann/faiss/swig/IndexBinaryFlat.java | 96 + .../ann/faiss/swig/IndexBinaryFromFloat.java | 80 + .../ann/faiss/swig/IndexBinaryHNSW.java | 110 + .../ann/faiss/swig/IndexBinaryIVF.java | 237 ++ .../com/twitter/ann/faiss/swig/IndexFlat.java | 85 + .../twitter/ann/faiss/swig/IndexFlat1D.java | 80 + .../ann/faiss/swig/IndexFlatCodes.java | 80 + .../twitter/ann/faiss/swig/IndexFlatIP.java | 47 + .../twitter/ann/faiss/swig/IndexFlatL2.java | 47 + .../com/twitter/ann/faiss/swig/IndexHNSW.java | 150 + .../ann/faiss/swig/IndexHNSW2Level.java | 55 + .../twitter/ann/faiss/swig/IndexHNSWFlat.java | 51 + .../twitter/ann/faiss/swig/IndexHNSWPQ.java | 51 + .../twitter/ann/faiss/swig/IndexHNSWSQ.java | 51 + .../twitter/ann/faiss/swig/IndexIDMap.java | 101 + .../com/twitter/ann/faiss/swig/IndexIVF.java | 250 ++ .../twitter/ann/faiss/swig/IndexIVFFlat.java | 76 + .../ann/faiss/swig/IndexIVFFlatDedup.java | 95 + .../twitter/ann/faiss/swig/IndexIVFPQ.java | 182 + .../ann/faiss/swig/IndexIVFPQStats.java | 79 + .../faiss/swig/IndexIVFScalarQuantizer.java | 100 + .../twitter/ann/faiss/swig/IndexIVFStats.java | 99 + .../com/twitter/ann/faiss/swig/IndexLSH.java | 122 + .../com/twitter/ann/faiss/swig/IndexPQ.java | 182 + .../twitter/ann/faiss/swig/IndexPQStats.java | 71 + .../twitter/ann/faiss/swig/IndexRefine.java | 121 + .../ann/faiss/swig/IndexRefineFlat.java | 55 + .../ann/faiss/swig/IndexScalarQuantizer.java | 80 + .../twitter/ann/faiss/swig/IndexShards.java | 99 + .../ann/faiss/swig/IndexSplitVectors.java | 96 + .../com/twitter/ann/faiss/swig/IntVector.java | 76 + .../ann/faiss/swig/InterruptCallback.java | 59 + .../ann/faiss/swig/IntersectionCriterion.java | 55 + .../twitter/ann/faiss/swig/InvertedLists.java | 262 ++ .../faiss/swig/InvertedListsPtrVector.java | 77 + .../ann/faiss/swig/Level1Quantizer.java | 114 + .../ann/faiss/swig/LinearTransform.java | 117 + .../twitter/ann/faiss/swig/LongVector.java | 76 + .../ann/faiss/swig/LongVectorVector.java | 76 + .../twitter/ann/faiss/swig/MapLong2Long.java | 63 + .../ann/faiss/swig/MaskedInvertedLists.java | 95 + .../twitter/ann/faiss/swig/MetricType.java | 60 + .../ann/faiss/swig/MultiIndexQuantizer.java | 76 + .../ann/faiss/swig/MultiIndexQuantizer2.java | 72 + .../faiss/swig/NormalizationTransform.java | 67 + .../com/twitter/ann/faiss/swig/OPQMatrix.java | 116 + .../ann/faiss/swig/OnDiskInvertedLists.java | 251 ++ .../faiss/swig/OnDiskInvertedListsIOHook.java | 57 + .../twitter/ann/faiss/swig/OnDiskOneList.java | 67 + .../ann/faiss/swig/OneRecallAtRCriterion.java | 55 + .../ann/faiss/swig/OperatingPoint.java | 75 + .../ann/faiss/swig/OperatingPointVector.java | 76 + .../ann/faiss/swig/OperatingPoints.java | 101 + .../com/twitter/ann/faiss/swig/PCAMatrix.java | 138 + .../twitter/ann/faiss/swig/PQDecoder16.java | 57 + .../twitter/ann/faiss/swig/PQDecoder8.java | 57 + .../ann/faiss/swig/PQDecoderGeneric.java | 80 + .../twitter/ann/faiss/swig/PQEncoder16.java | 56 + .../twitter/ann/faiss/swig/PQEncoder8.java | 56 + .../ann/faiss/swig/PQEncoderGeneric.java | 80 + .../ann/faiss/swig/ParameterRange.java | 60 + .../ann/faiss/swig/ParameterSpace.java | 136 + .../ann/faiss/swig/PartitionStats.java | 63 + .../ann/faiss/swig/PermutationObjective.java | 55 + .../ann/faiss/swig/PolysemousTraining.java | 144 + .../ann/faiss/swig/ProductQuantizer.java | 279 ++ .../faiss/swig/ProgressiveDimClustering.java | 85 + .../ProgressiveDimClusteringParameters.java | 59 + .../swig/ProgressiveDimIndexFactory.java | 43 + .../ann/faiss/swig/RandomRotationMatrix.java | 55 + .../ann/faiss/swig/RangeQueryResult.java | 72 + .../faiss/swig/RangeSearchPartialResult.java | 81 + .../ann/faiss/swig/RangeSearchResult.java | 85 + .../ann/faiss/swig/ReadOnlyInvertedLists.java | 51 + .../faiss/swig/ReconstructFromNeighbors.java | 161 + .../faiss/swig/RemapDimensionsTransform.java | 72 + .../swig/ReproduceDistancesObjective.java | 106 + .../SWIGTYPE_p_AlignedTableT_float_32_t.java | 26 + .../SWIGTYPE_p_AlignedTableT_float_t.java | 26 + .../ann/faiss/swig/SWIGTYPE_p_DirectMap.java | 26 + .../swig/SWIGTYPE_p_DirectMap__Type.java | 26 + .../ann/faiss/swig/SWIGTYPE_p_FILE.java | 26 + .../ann/faiss/swig/SWIGTYPE_p_IOReader.java | 26 + .../ann/faiss/swig/SWIGTYPE_p_IOWriter.java | 26 + .../swig/SWIGTYPE_p_ScalarQuantizer.java | 26 + ...TYPE_p_ScalarQuantizer__QuantizerType.java | 26 + .../ann/faiss/swig/SWIGTYPE_p_double.java | 26 + ...s__AlignedTableTightAllocT_float_32_t.java | 26 + ...AlignedTableTightAllocT_uint16_t_32_t.java | 26 + ...edTableTightAllocT_unsigned_char_32_t.java | 26 + ...PE_p_faiss__BinaryInvertedListScanner.java | 26 + ...ArrayT_faiss__CMaxT_float_int64_t_t_t.java | 26 + ...apArrayT_faiss__CMaxT_int_int64_t_t_t.java | 26 + ...ArrayT_faiss__CMinT_float_int64_t_t_t.java | 26 + .../swig/SWIGTYPE_p_faiss__IOReader.java | 26 + .../swig/SWIGTYPE_p_faiss__IOWriter.java | 26 + ...SWIGTYPE_p_faiss__InvertedListScanner.java | 26 + .../swig/SWIGTYPE_p_faiss__LockLevels.java | 26 + ..._OnDiskInvertedLists__OngoingPrefetch.java | 26 + .../SWIGTYPE_p_faiss__RandomGenerator.java | 26 + .../ann/faiss/swig/SWIGTYPE_p_float.java | 26 + .../ann/faiss/swig/SWIGTYPE_p_int.java | 26 + .../ann/faiss/swig/SWIGTYPE_p_long.java | 26 + .../ann/faiss/swig/SWIGTYPE_p_long_long.java | 26 + .../ann/faiss/swig/SWIGTYPE_p_omp_lock_t.java | 26 + .../faiss/swig/SWIGTYPE_p_p_faiss__Index.java | 26 + .../SWIGTYPE_p_p_faiss__InvertedLists.java | 26 + .../SWIGTYPE_p_p_faiss__VectorTransform.java | 26 + ...tT_faiss__OnDiskInvertedLists__Slot_t.java | 26 + .../SWIGTYPE_p_std__pairT_float_int_t.java | 26 + ...queueT_faiss__HNSW__NodeDistFarther_t.java | 26 + ...ority_queueT_std__pairT_float_int_t_t.java | 26 + ...YPE_p_std__unordered_mapT_long_long_t.java | 26 + ...unordered_multimapT_int64_t_int64_t_t.java | 26 + ...__vectorT_faiss__BufferList__Buffer_t.java | 26 + ...orT_faiss__ClusteringIterationStats_t.java | 26 + ...ectorT_faiss__HNSW__NodeDistFarther_t.java | 26 + ...GTYPE_p_std__vectorT_faiss__Index_p_t.java | 26 + ...ectorT_faiss__InvertedLists_const_p_t.java | 26 + ...p_std__vectorT_faiss__OnDiskOneList_t.java | 26 + ..._std__vectorT_faiss__ParameterRange_t.java | 26 + ...td__vectorT_faiss__RangeQueryResult_t.java | 26 + ...T_faiss__RangeSearchPartialResult_p_t.java | 26 + .../SWIGTYPE_p_std__vectorT_int64_t_t.java | 26 + .../swig/SWIGTYPE_p_std__vectorT_long_t.java | 26 + .../SWIGTYPE_p_std__vectorT_omp_lock_t_t.java | 26 + ...std__vectorT_std__vectorT_int64_t_t_t.java | 26 + ...ectorT_std__vectorT_unsigned_long_t_t.java | 26 + .../ann/faiss/swig/SWIGTYPE_p_uint16_t.java | 26 + .../ann/faiss/swig/SWIGTYPE_p_uint32_t.java | 26 + .../faiss/swig/SWIGTYPE_p_unsigned_char.java | 26 + .../faiss/swig/SWIGTYPE_p_unsigned_long.java | 26 + .../ann/faiss/swig/SWIGTYPE_p_void.java | 26 + .../swig/SimulatedAnnealingOptimizer.java | 94 + .../swig/SimulatedAnnealingParameters.java | 107 + .../ann/faiss/swig/SliceInvertedLists.java | 102 + .../ann/faiss/swig/SlidingIndexWindow.java | 90 + .../faiss/swig/StopWordsInvertedLists.java | 94 + .../twitter/ann/faiss/swig/Uint64Vector.java | 76 + .../ann/faiss/swig/VStackInvertedLists.java | 95 + .../ann/faiss/swig/VectorTransform.java | 80 + .../ann/faiss/swig/VectorTransformVector.java | 77 + .../twitter/ann/faiss/swig/VisitedTable.java | 72 + .../twitter/ann/faiss/swig/doubleArray.java | 61 + .../twitter/ann/faiss/swig/floatArray.java | 61 + .../ann/faiss/swig/float_maxheap_array_t.java | 133 + .../ann/faiss/swig/float_minheap_array_t.java | 133 + .../com/twitter/ann/faiss/swig/intArray.java | 61 + .../ann/faiss/swig/int_maxheap_array_t.java | 133 + .../ann/faiss/swig/int_minheap_array_t.java | 133 + .../com/twitter/ann/faiss/swig/longArray.java | 61 + .../ann/faiss/swig/resources/.gitignore | 7 + .../twitter/ann/faiss/swig/resources/.gitkeep | 0 .../twitter/ann/faiss/swig/resources/BUILD | 17 + .../com/twitter/ann/faiss/swig/swigfaiss.java | 575 +++ .../ann/faiss/swig/swigfaissConstants.java | 15 + .../twitter/ann/faiss/swig/swigfaissJNI.java | 2147 ++++++++++ ann/src/main/java/com/twitter/ann/hnsw/BUILD | 18 + .../twitter/ann/hnsw/DistanceFunction.java | 8 + .../com/twitter/ann/hnsw/DistancedItem.java | 23 + .../twitter/ann/hnsw/DistancedItemQueue.java | 196 + .../java/com/twitter/ann/hnsw/HnswIndex.java | 711 ++++ .../com/twitter/ann/hnsw/HnswIndexIOUtil.java | 133 + .../java/com/twitter/ann/hnsw/HnswMeta.java | 45 + .../java/com/twitter/ann/hnsw/HnswNode.java | 45 + .../hnsw/IllegalDuplicateInsertException.java | 7 + ann/src/main/python/dataflow/BUILD.bazel | 38 + ann/src/main/python/dataflow/bq.sql | 6 + .../python/dataflow/faiss_index_bq_dataset.py | 232 + .../python/dataflow/worker_harness/Dockerfile | 34 + .../dataflow/worker_harness/cloudbuild.yml | 6 + .../com/twitter/ann/annoy/AnnoyCommon.scala | 44 + .../main/scala/com/twitter/ann/annoy/BUILD | 23 + .../ann/annoy/RawAnnoyIndexBuilder.scala | 123 + .../ann/annoy/RawAnnoyQueryIndex.scala | 142 + .../twitter/ann/annoy/TypedAnnoyIndex.scala | 55 + .../TypedAnnoyIndexBuilderWithFile.scala | 55 + .../annoy/TypedAnnoyQueryIndexWithFile.scala | 42 + .../scala/com/twitter/ann/brute_force/BUILD | 12 + .../BruteForceDeserialization.scala | 64 + .../ann/brute_force/BruteForceIndex.scala | 162 + .../twitter/ann/common/AnnInjections.scala | 28 + .../scala/com/twitter/ann/common/Api.scala | 150 + .../main/scala/com/twitter/ann/common/BUILD | 21 + .../ann/common/EmbeddingProducer.scala | 13 + .../twitter/ann/common/IndexOutputFile.scala | 226 + .../twitter/ann/common/IndexTransformer.scala | 118 + .../twitter/ann/common/MemoizedInEpochs.scala | 37 + .../scala/com/twitter/ann/common/Metric.scala | 290 ++ .../twitter/ann/common/QueryableById.scala | 41 + .../common/QueryableByIdImplementation.scala | 91 + .../ann/common/QueryableOperations.scala | 26 + .../ann/common/ReadWriteFuturePool.scala | 29 + .../twitter/ann/common/Serialization.scala | 28 + .../ann/common/ServiceClientQueryable.scala | 64 + .../com/twitter/ann/common/ShardApi.scala | 87 + .../ann/common/ShardedSerialization.scala | 89 + .../scala/com/twitter/ann/common/Task.scala | 121 + .../offline/ANNIndexBuilderBeamJob.scala | 461 ++ .../com/twitter/ann/dataflow/offline/BUILD | 27 + .../dataflow/offline/BaseEmbeddingData.scala | 6 + .../dataflow/offline/FlatEmbeddingData.scala | 8 + .../offline/GroupedEmbeddingData.scala | 9 + .../com/twitter/ann/experimental/BUILD.bazel | 29 + .../com/twitter/ann/experimental/Runner.scala | 171 + .../main/scala/com/twitter/ann/faiss/BUILD | 23 + .../com/twitter/ann/faiss/FaissCommon.scala | 44 + .../com/twitter/ann/faiss/FaissIndex.scala | 43 + .../com/twitter/ann/faiss/FaissIndexer.scala | 154 + ...ourlyDirectoryWithSuccessFileListing.scala | 64 + .../ann/faiss/HourlyShardedIndex.scala | 94 + .../ann/faiss/QueryableIndexAdapter.scala | 196 + .../scala/com/twitter/ann/featurestore/BUILD | 10 + .../FeatureStoreEmbeddingProducer.scala | 66 + .../scala/com/twitter/ann/file_store/BUILD | 12 + .../file_store/ReadableIndexIdFileStore.scala | 35 + .../file_store/WritableIndexIdFileStore.scala | 71 + ann/src/main/scala/com/twitter/ann/hnsw/BUILD | 23 + .../ann/hnsw/DistanceFunctionGenerator.scala | 37 + .../scala/com/twitter/ann/hnsw/Hnsw.scala | 183 + .../com/twitter/ann/hnsw/HnswCommon.scala | 62 + .../com/twitter/ann/hnsw/HnswIOUtil.scala | 106 + .../com/twitter/ann/hnsw/IdEmbeddingMap.scala | 13 + .../ann/hnsw/JMapBasedIdEmbeddingMap.scala | 87 + .../ann/hnsw/MapDbBasedIdEmbeddingMap.scala | 81 + .../twitter/ann/hnsw/SerializableHnsw.scala | 196 + .../com/twitter/ann/hnsw/TypedHnswIndex.scala | 173 + .../scala/com/twitter/ann/manhattan/BUILD | 12 + .../ManhattanEmbeddingProducer.scala | 63 + .../scala/com/twitter/ann/manhattan/README | 9 + .../com/twitter/ann/scalding/benchmark/BUILD | 56 + .../twitter/ann/scalding/benchmark/Knn.scala | 128 + .../twitter/ann/scalding/offline/BUILD.bazel | 44 + .../scalding/offline/IndexingStrategy.scala | 116 + .../ann/scalding/offline/KnnDebug.scala | 117 + .../offline/KnnEntityRecoDebugJob.scala | 91 + .../ann/scalding/offline/KnnHelper.scala | 438 ++ .../ann/scalding/offline/KnnOfflineJob.scala | 108 + .../offline/KnnTruthSetGenerator.scala | 84 + .../offline/ParameterlessQueryable.scala | 24 + .../com/twitter/ann/scalding/offline/README | 4 + .../offline/faissindexbuilder/BUILD.bazel | 37 + .../faissindexbuilder/IndexBuilder.scala | 42 + .../faissindexbuilder/IndexBuilderApp.scala | 75 + .../scalding/offline/indexbuilder/BUILD.bazel | 37 + .../offline/indexbuilder/IndexBuilder.scala | 53 + .../indexbuilder/IndexBuilderApp.scala | 91 + .../scalding/offline/indexbuilder/README.rst | 132 + .../offline/indexbuilderfrombq/BUILD.bazel | 37 + .../IndexBuilderFromBQ.scala | 53 + .../IndexBuilderFromBQApp.scala | 194 + .../scala/com/twitter/ann/serialization/BUILD | 14 + .../DummyANNIndexInjection.scala | 12 + .../PersistedEmbeddingInjection.scala | 28 + .../ann/serialization/ThriftIteratorIO.scala | 57 + .../ann/service/loadtest/AnnLoadTest.scala | 66 + .../service/loadtest/AnnLoadTestMain.scala | 379 ++ .../service/loadtest/AnnLoadTestWorker.scala | 116 + .../com/twitter/ann/service/loadtest/BUILD | 31 + .../service/loadtest/EmbeddingIndexer.scala | 28 + .../service/loadtest/LoadTestRecorder.scala | 231 + .../ann/service/loadtest/LoadTestUtils.scala | 200 + .../twitter/ann/service/loadtest/README.md | 218 + .../ann/service/query_server/common/BUILD | 63 + .../common/BaseQueryIndexServer.scala | 53 + .../query_server/common/Exceptions.scala | 15 + .../common/FaissIndexPathProvider.scala | 20 + .../common/IndexPathProvider.scala | 179 + .../common/QueryIndexThriftController.scala | 92 + .../query_server/common/QueryServerUtil.scala | 34 + .../common/QueryableProvider.scala | 8 + .../common/RefreshableQueryable.scala | 212 + .../common/UnsafeQueryIndexServer.scala | 109 + .../throttling/AuroraCPUStatsReader.scala | 28 + .../query_server/common/throttling/BUILD | 12 + .../ThrottlingBasedQualityTask.scala | 73 + .../common/throttling/WindowedStats.scala | 22 + .../WindowedThrottlingInstrument.scala | 50 + .../service/query_server/common/warmup/BUILD | 10 + .../query_server/common/warmup/Warmup.scala | 50 + .../ann/service/query_server/faiss/BUILD | 33 + .../faiss/FaissQueryIndexServer.scala | 149 + .../ann/service/query_server/hnsw/BUILD | 32 + .../hnsw/HnswQueryIndexServer.scala | 98 + ann/src/main/scala/com/twitter/ann/util/BUILD | 9 + .../twitter/ann/util/IndexBuilderUtils.scala | 31 + .../main/thrift/com/twitter/ann/common/BUILD | 18 + .../com/twitter/ann/common/ann_common.thrift | 169 + ann/src/main/thrift/com/twitter/ann/knn/BUILD | 11 + .../thrift/com/twitter/ann/knn/knn.thrift | 15 + .../com/twitter/ann/serialization/BUILD | 13 + .../ann/serialization/serialization.thrift | 10 + ci/ci.sh | 3 + cr-mixer/BUILD.bazel | 24 + cr-mixer/README.md | 7 + .../server/src/main/resources/BUILD.bazel | 8 + .../src/main/resources/config/decider.yml | 146 + .../server/src/main/resources/logback.xml | 168 + .../scala/com/twitter/cr_mixer/BUILD.bazel | 48 + .../CrMixerHttpServerWarmupHandler.scala | 18 + .../com/twitter/cr_mixer/CrMixerServer.scala | 229 + .../CrMixerThriftServerWarmupHandler.scala | 75 + .../twitter/cr_mixer/blender/AdsBlender.scala | 77 + .../scala/com/twitter/cr_mixer/blender/BUILD | 20 + .../blender/BlendedCandidatesBuilder.scala | 48 + .../blender/ContentSignalBlender.scala | 121 + .../CountWeightedInterleaveBlender.scala | 90 + .../cr_mixer/blender/InterleaveBlender.scala | 33 + .../blender/SourceTypeBackFillBlender.scala | 64 + .../cr_mixer/blender/SwitchBlender.scala | 81 + .../AdsCandidateGenerator.scala | 140 + .../AdsCandidateSourcesRouter.scala | 516 +++ .../cr_mixer/candidate_generation/BUILD | 51 + .../CandidateSourcesRouter.scala | 536 +++ .../CrCandidateGenerator.scala | 350 ++ ...stomizedRetrievalCandidateGeneration.scala | 345 ++ .../FrsTweetCandidateGenerator.scala | 220 + .../RelatedTweetCandidateGenerator.scala | 156 + .../RelatedVideoTweetCandidateGenerator.scala | 139 + ...stersInterestedInCandidateGeneration.scala | 640 +++ .../TopicTweetCandidateGenerator.scala | 232 + .../UtegTweetCandidateGenerator.scala | 179 + .../scala/com/twitter/cr_mixer/config/BUILD | 13 + .../config/SimClustersANNConfig.scala | 473 +++ .../cr_mixer/config/TimeoutConfig.scala | 24 + .../twitter/cr_mixer/controller/BUILD.bazel | 48 + .../controller/CrMixerThriftController.scala | 757 ++++ .../com/twitter/cr_mixer/exception/BUILD | 7 + .../InvalidSANNConfigException.scala | 4 + .../com/twitter/cr_mixer/featureswitch/BUILD | 35 + .../CrMixerLoggingABDecider.scala | 79 + .../featureswitch/ParamsBuilder.scala | 151 + ...etImpressedBucketsLocalContextFilter.scala | 22 + .../scala/com/twitter/cr_mixer/filter/BUILD | 22 + .../twitter/cr_mixer/filter/FilterBase.scala | 22 + .../filter/ImpressedTweetlistFilter.scala | 63 + .../cr_mixer/filter/InNetworkFilter.scala | 80 + .../filter/PostRankFilterRunner.scala | 58 + .../cr_mixer/filter/PreRankFilterRunner.scala | 99 + .../twitter/cr_mixer/filter/ReplyFilter.scala | 40 + .../cr_mixer/filter/RetweetFilter.scala | 41 + .../cr_mixer/filter/TweetAgeFilter.scala | 39 + .../filter/TweetInfoHealthFilterBase.scala | 39 + .../cr_mixer/filter/UtegFilterRunner.scala | 96 + .../cr_mixer/filter/UtegHealthFilter.scala | 51 + .../cr_mixer/filter/VideoTweetFilter.scala | 81 + .../AdsRecommendationsScribeLogger.scala | 139 + .../scala/com/twitter/cr_mixer/logging/BUILD | 34 + .../logging/CrMixerScribeLogger.scala | 489 +++ .../logging/RelatedTweetScribeLogger.scala | 193 + .../cr_mixer/logging/ScribeLoggerUtils.scala | 43 + .../cr_mixer/logging/ScribeMetadata.scala | 45 + .../logging/TopLevelDdgMetricsMetadata.scala | 22 + .../logging/UtegTweetScribeLogger.scala | 147 + .../scala/com/twitter/cr_mixer/model/BUILD | 16 + .../twitter/cr_mixer/model/Candidate.scala | 200 + .../model/CandidateGenerationInfo.scala | 67 + .../model/CandidateGeneratorQuery.scala | 96 + .../model/EarlybirdSimilarityEngineType.scala | 6 + .../cr_mixer/model/HealthThreshold.scala | 11 + .../twitter/cr_mixer/model/ModelConfig.scala | 77 + .../twitter/cr_mixer/model/ModuleNames.scala | 122 + .../cr_mixer/model/TopicTweetWithScore.scala | 13 + .../cr_mixer/model/TweetWithAuthor.scala | 6 + .../cr_mixer/model/TweetWithScore.scala | 8 + .../model/TweetWithScoreAndSocialProof.scala | 12 + .../ActivePromotedTweetStoreModule.scala | 135 + .../com/twitter/cr_mixer/module/BUILD.bazel | 130 + .../BlueVerifiedAnnotationStoreModule.scala | 52 + .../module/CertoStratoStoreModule.scala | 57 + ...ConsumersBasedUserAdGraphStoreModule.scala | 30 + ...sumersBasedUserTweetGraphStoreModule.scala | 30 + ...sumersBasedUserVideoGraphStoreModule.scala | 30 + .../module/CrMixerParamConfigModule.scala | 16 + .../module/DiffusionStoreModule.scala | 54 + ...birdRecencyBasedCandidateStoreModule.scala | 189 + .../module/EmbeddingStoreModule.scala | 195 + .../cr_mixer/module/FrsStoreModule.scala | 29 + .../cr_mixer/module/MHMtlsParamsModule.scala | 17 + .../module/OfflineCandidateStoreModule.scala | 150 + .../module/RealGraphOonStoreModule.scala | 39 + .../module/RealGraphStoreMhModule.scala | 67 + .../module/RepresentationManagerModule.scala | 107 + .../module/RepresentationScorerModule.scala | 56 + .../module/SampleSimilarityEngineModule.scala | 90 + ...ClustersANNServiceNameToClientMapper.scala | 33 + .../module/SkitStratoStoreModule.scala | 65 + .../StrongTiePredictionStoreModule.scala | 39 + .../module/TripCandidateStoreModule.scala | 34 + .../module/TweetInfoStoreModule.scala | 205 + .../TweetRecentEngagedUserStoreModule.scala | 42 + ...weetRecommendationResultsStoreModule.scala | 32 + .../TwhinCollabFilterStratoStoreModule.scala | 67 + .../TwiceClustersMembersStoreModule.scala | 42 + .../cr_mixer/module/UnifiedCacheClient.scala | 83 + .../UserSignalServiceColumnModule.scala | 30 + .../module/UserSignalServiceStoreModule.scala | 37 + .../module/UserStateStoreModule.scala | 113 + .../module/core/ABDeciderModule.scala | 33 + .../module/core/CrMixerFlagModule.scala | 20 + .../core/CrMixerLoggingABDeciderModule.scala | 20 + .../core/FeatureContextBuilderModule.scala | 16 + .../module/core/FeatureSwitchesModule.scala | 74 + .../module/core/KafkaProducerModule.scala | 70 + .../module/core/LoggerFactoryModule.scala | 155 + .../core/MemoizingStatsReceiverModule.scala | 12 + .../module/core/TimeoutConfigModule.scala | 104 + .../grpc_client/NaviGRPCClientModule.scala | 90 + ...ertoTopicTweetSimilarityEngineModule.scala | 57 + ...sumerBasedWalsSimilarityEngineModule.scala | 54 + ...ddingBasedTripSimilarityEngineModule.scala | 60 + ...dingBasedTwHINSimilarityEngineModule.scala | 58 + ...gBasedTwoTowerSimilarityEngineModule.scala | 51 + ...sedUserAdGraphSimilarityEngineModule.scala | 61 + ...UserVideoGraphSimilarityEngineModule.scala | 62 + ...DiffusionBasedSimilarityEngineModule.scala | 52 + .../EarlybirdSimilarityEngineModule.scala | 120 + ...erBasedUnifiedSimilarityEngineModule.scala | 68 + ...sedUserAdGraphSimilarityEngineModule.scala | 67 + ...UserTweetGraphSimilarityEngineModule.scala | 67 + ...SimClustersANNSimilarityEngineModule.scala | 117 + ...SkitTopicTweetSimilarityEngineModule.scala | 88 + .../TweetBasedQigSimilarityEngineModule.scala | 66 + ...TweetBasedTwHINSimlarityEngineModule.scala | 70 + ...etBasedUnifiedSimilarityEngineModule.scala | 83 + ...sedUserAdGraphSimilarityEngineModule.scala | 91 + ...UserTweetGraphSimilarityEngineModule.scala | 92 + ...UserVideoGraphSimilarityEngineModule.scala | 92 + ...abFilterLookupSimilarityEngineModule.scala | 71 + ...eetEntityGraphSimilarityEngineModule.scala | 55 + .../AnnQueryServiceClientModule.scala | 107 + .../EarlybirdSearchClientModule.scala | 39 + .../thrift_client/FrsClientModule.scala | 41 + .../HydraPartitionClientModule.scala | 25 + .../thrift_client/HydraRootClientModule.scala | 25 + .../QigServiceClientModule.scala | 40 + .../SimClustersAnnServiceClientModule.scala | 147 + .../thrift_client/TweetyPieClientModule.scala | 60 + .../UserAdGraphClientModule.scala | 47 + .../UserTweetEntityGraphClientModule.scala | 44 + .../UserTweetGraphClientModule.scala | 43 + .../UserTweetGraphPlusClientModule.scala | 46 + .../UserVideoGraphClientModule.scala | 46 + .../twitter/cr_mixer/param/AdsParams.scala | 64 + .../scala/com/twitter/cr_mixer/param/BUILD | 27 + .../cr_mixer/param/BlenderParams.scala | 152 + .../param/BypassInterleaveAndRankParams.scala | 98 + .../param/ConsumerBasedWalsParams.scala | 96 + ...eddingBasedCandidateGenerationParams.scala | 55 + .../ConsumerEmbeddingBasedTripParams.scala | 46 + .../ConsumerEmbeddingBasedTwHINParams.scala | 33 + ...ConsumerEmbeddingBasedTwoTowerParams.scala | 32 + .../ConsumersBasedUserAdGraphParams.scala | 54 + .../ConsumersBasedUserTweetGraphParams.scala | 44 + .../ConsumersBasedUserVideoGraphParams.scala | 65 + .../cr_mixer/param/CrMixerParamConfig.scala | 122 + ...rievalBasedCandidateGenerationParams.scala | 81 + ...valBasedFTROfflineInterestedInParams.scala | 31 + ...rievalBasedOfflineInterestedInParams.scala | 33 + .../CustomizedRetrievalBasedTwhinParams.scala | 60 + ...irdFrsBasedCandidateGenerationParams.scala | 117 + .../twitter/cr_mixer/param/FrsParams.scala | 131 + .../twitter/cr_mixer/param/GlobalParams.scala | 106 + .../param/GoodProfileClickParams.scala | 60 + .../cr_mixer/param/GoodTweetClickParams.scala | 75 + .../cr_mixer/param/InterestedInParams.scala | 213 + ...oducerBasedCandidateGenerationParams.scala | 143 + .../ProducerBasedUserAdGraphParams.scala | 53 + .../ProducerBasedUserTweetGraphParams.scala | 53 + .../twitter/cr_mixer/param/RankerParams.scala | 59 + .../cr_mixer/param/RealGraphInParams.scala | 25 + .../cr_mixer/param/RealGraphOonParams.scala | 51 + .../cr_mixer/param/RecentFollowsParams.scala | 27 + .../param/RecentNegativeSignalParams.scala | 39 + .../param/RecentNotificationsParams.scala | 28 + .../param/RecentOriginalTweetsParams.scala | 28 + .../param/RecentReplyTweetsParams.scala | 27 + .../cr_mixer/param/RecentRetweetsParams.scala | 30 + .../param/RecentTweetFavoritesParams.scala | 29 + .../param/RelatedTweetGlobalParams.scala | 32 + .../RelatedTweetProducerBasedParams.scala | 111 + .../param/RelatedTweetTweetBasedParams.scala | 141 + .../param/RelatedVideoTweetGlobalParams.scala | 32 + .../RelatedVideoTweetTweetBasedParams.scala | 134 + .../param/RepeatedProfileVisitsParams.scala | 72 + .../cr_mixer/param/SimClustersANNParams.scala | 76 + .../cr_mixer/param/TopicTweetParams.scala | 115 + .../TweetBasedCandidateGenerationParams.scala | 189 + .../param/TweetBasedTwHINParams.scala | 30 + .../param/TweetBasedUserAdGraphParams.scala | 58 + .../TweetBasedUserTweetGraphParams.scala | 89 + .../TweetBasedUserVideoGraphParams.scala | 81 + .../cr_mixer/param/TweetSharesParams.scala | 29 + .../UnifiedSETweetCombinationMethod.scala | 15 + .../param/UnifiedUSSSignalParams.scala | 121 + .../param/UtegTweetGlobalParams.scala | 94 + .../param/VideoTweetFilterParams.scala | 31 + .../param/VideoViewTweetsParams.scala | 64 + .../com/twitter/cr_mixer/param/decider/BUILD | 16 + .../param/decider/CrMixerDecider.scala | 39 + .../cr_mixer/param/decider/DeciderKey.scala | 67 + .../param/decider/EndpointLoadShedder.scala | 57 + .../scala/com/twitter/cr_mixer/ranker/BUILD | 30 + .../cr_mixer/ranker/DefaultRanker.scala | 23 + .../cr_mixer/ranker/SwitchRanker.scala | 46 + .../scala/com/twitter/cr_mixer/scribe/BUILD | 22 + .../cr_mixer/scribe/ScribeCategory.scala | 64 + .../com/twitter/cr_mixer/service/BUILD.bazel | 15 + .../CrMixerAlertNotificationConfig.scala | 26 + .../twitter/cr_mixer/similarity_engine/BUILD | 74 + .../CertoTopicTweetSimilarityEngine.scala | 94 + .../ConsumerBasedWalsSimilarityEngine.scala | 246 ++ ...erEmbeddingBasedTripSimilarityEngine.scala | 118 + ...rEmbeddingBasedTwHINSimilarityEngine.scala | 18 + ...beddingBasedTwoTowerSimilarityEngine.scala | 18 + ...mersBasedUserAdGraphSimilarityEngine.scala | 90 + ...sBasedUserVideoGraphSimilarityEngine.scala | 91 + .../DiffusionBasedSimilarityEngine.scala | 73 + .../EarlybirdModelBasedSimilarityEngine.scala | 92 + ...arlybirdRecencyBasedSimilarityEngine.scala | 86 + .../EarlybirdSimilarityEngine.scala | 32 + .../EarlybirdSimilarityEngineBase.scala | 56 + .../EarlybirdSimilarityEngineRouter.scala | 136 + ...ybirdTensorflowBasedSimilarityEngine.scala | 171 + .../similarity_engine/FilterUtil.scala | 42 + .../HnswANNSimilarityEngine.scala | 187 + .../LookupSimilarityEngine.scala | 78 + .../ModelBasedANNStore.scala | 136 + ...ProducerBasedUnifiedSimilarityEngine.scala | 641 +++ ...ucerBasedUserAdGraphSimilarityEngine.scala | 96 + ...rBasedUserTweetGraphSimilarityEngine.scala | 96 + .../SimClustersANNSimilarityEngine.scala | 113 + .../similarity_engine/SimilarityEngine.scala | 169 + .../SimilaritySourceOrderingUtil.scala | 32 + ...hPrecisionTopicTweetSimilarityEngine.scala | 123 + .../SkitTopicTweetSimilarityEngine.scala | 143 + .../StandardSimilarityEngine.scala | 65 + .../TweetBasedQigSimilarityEngine.scala | 114 + .../TweetBasedUnifiedSimilarityEngine.scala | 962 +++++ ...weetBasedUserAdGraphSimilarityEngine.scala | 129 + ...tBasedUserTweetGraphSimilarityEngine.scala | 184 + ...tBasedUserVideoGraphSimilarityEngine.scala | 184 + .../TwhinCollabFilterSimilarityEngine.scala | 72 + ...UserTweetEntityGraphSimilarityEngine.scala | 110 + .../com/twitter/cr_mixer/source_signal/BUILD | 32 + .../source_signal/FrsSourceGraphFetcher.scala | 54 + .../FrsSourceSignalFetcher.scala | 65 + .../cr_mixer/source_signal/FrsStore.scala | 81 + .../RealGraphInSourceGraphFetcher.scala | 55 + .../RealGraphOonSourceGraphFetcher.scala | 55 + .../source_signal/SourceFetcher.scala | 101 + .../source_signal/SourceGraphFetcher.scala | 70 + .../source_signal/SourceInfoRouter.scala | 68 + .../source_signal/SourceSignalFetcher.scala | 45 + .../UssSourceSignalFetcher.scala | 160 + .../cr_mixer/source_signal/UssStore.scala | 209 + .../scala/com/twitter/cr_mixer/util/BUILD | 29 + .../util/CandidateGenerationKeyUtil.scala | 39 + .../util/CountWeightedInterleaveUtil.scala | 180 + .../cr_mixer/util/EarlybirdSearchUtil.scala | 130 + .../cr_mixer/util/InterleaveUtil.scala | 160 + .../twitter/cr_mixer/util/MetricTagUtil.scala | 135 + .../util/SignalTimestampStatsUtil.scala | 66 + cr-mixer/thrift/src/main/thrift/BUILD | 48 + cr-mixer/thrift/src/main/thrift/ads.thrift | 33 + .../thrift/candidate_generation_key.thrift | 21 + .../thrift/src/main/thrift/cr_mixer.thrift | 104 + .../src/main/thrift/frs_based_tweet.thrift | 35 + .../thrift/src/main/thrift/metric_tags.thrift | 44 + .../thrift/src/main/thrift/product.thrift | 19 + .../src/main/thrift/product_context.thrift | 21 + .../src/main/thrift/related_tweet.thrift | 24 + .../main/thrift/related_video_tweet.thrift | 23 + cr-mixer/thrift/src/main/thrift/scribe.thrift | 168 + .../thrift/src/main/thrift/source_type.thrift | 123 + .../thrift/src/main/thrift/topic_tweet.thrift | 28 + cr-mixer/thrift/src/main/thrift/uteg.thrift | 31 + .../thrift/src/main/thrift/validation.thrift | 19 + docs/system-diagram.png | Bin 0 -> 95955 bytes follow-recommendations-service/BUILD | 48 + follow-recommendations-service/CONFIG.ini | 24 + .../FRS_architecture.png | Bin 0 -> 181765 bytes follow-recommendations-service/README.md | 40 + .../follow_recommendations/common/base/BUILD | 18 + .../common/base/CandidateSourceRegistry.scala | 36 + .../common/base/EnrichedCandidateSource.scala | 164 + .../common/base/ParamPredicate.scala | 17 + .../common/base/Predicate.scala | 282 ++ .../common/base/PredicateResult.scala | 18 + .../common/base/Ranker.scala | 90 + .../common/base/RecommendationFlow.scala | 250 ++ .../common/base/SideEffectsUtil.scala | 24 + .../common/base/StatsUtil.scala | 272 ++ .../common/base/Transform.scala | 85 + .../addressbook/AddressBookParams.scala | 9 + .../candidate_sources/addressbook/BUILD | 27 + .../addressbook/ForwardEmailBookSource.scala | 74 + .../addressbook/ForwardPhoneBookSource.scala | 72 + .../candidate_sources/addressbook/README.md | 4 + .../addressbook/ReverseEmailBookSource.scala | 78 + .../addressbook/ReversePhoneBookSource.scala | 77 + .../common/candidate_sources/base/BUILD | 23 + .../base/CachedCandidateSource.scala | 26 + .../base/ExperimentalCandidateSource.scala | 66 + .../base/RealGraphExpansionRepository.scala | 208 + .../base/SimilarUserExpanderParams.scala | 31 + .../base/SimilarUserExpanderRepository.scala | 313 ++ .../SocialProofEnforcedCandidateSource.scala | 86 + ...ProofEnforcedCandidateSourceFSConfig.scala | 30 + ...alProofEnforcedCandidateSourceParams.scala | 56 + .../base/StratoFetcherSource.scala | 27 + .../StratoFetcherWithUnitViewSource.scala | 9 + .../base/TweetAuthorsCandidateSource.scala | 71 + .../base/TwoHopExpansionCandidateSource.scala | 46 + .../crowd_search_accounts/BUILD | 22 + .../CrowdSearchAccountsFSConfig.scala | 18 + .../CrowdSearchAccountsParams.scala | 32 + .../CrowdSearchAccountsSource.scala | 111 + .../crowd_search_accounts/README.md | 4 + .../common/candidate_sources/geo/BUILD | 23 + .../geo/BasePopGeoHashSource.scala | 74 + .../geo/PopCountryBackFillSource.scala | 33 + .../geo/PopCountrySource.scala | 63 + .../geo/PopGeoQualityFollowSource.scala | 99 + .../PopGeoQualityFollowSourceFSConfig.scala | 24 + .../geo/PopGeoQualityFollowSourceParams.scala | 42 + .../candidate_sources/geo/PopGeoSource.scala | 69 + .../geo/PopGeoSourceFSConfig.scala | 20 + .../geo/PopGeoSourceParams.scala | 30 + .../geo/PopGeohashSource.scala | 36 + .../common/candidate_sources/geo/README.md | 4 + .../ppmi_locale_follow/BUILD | 23 + .../PPMILocaleFollowSource.scala | 84 + .../PPMILocaleFollowSourceFSConfig.scala | 24 + .../PPMILocaleFollowSourceParams.scala | 22 + .../ppmi_locale_follow/README.md | 6 + .../candidate_sources/promoted_accounts/BUILD | 11 + .../PromotedAccountsCandidateSource.scala | 111 + .../promoted_accounts/README.md | 2 + .../common/candidate_sources/real_graph/BUILD | 24 + .../candidate_sources/real_graph/README.md | 6 + .../real_graph/RealGraphOonFSConfig.scala | 27 + .../real_graph/RealGraphOonParams.scala | 47 + .../real_graph/RealGraphOonV2Source.scala | 58 + .../real_graph/RealGraphSource.scala | 40 + .../candidate_sources/recent_engagement/BUILD | 29 + .../recent_engagement/README.md | 4 + .../RecentEngagementDirectFollowSource.scala | 38 + ...ecentEngagementNonDirectFollowSource.scala | 38 + .../RepeatedProfileVisitsFSConfig.scala | 22 + .../RepeatedProfileVisitsParams.scala | 37 + .../RepeatedProfileVisitsSource.scala | 157 + .../common/candidate_sources/salsa/BUILD | 21 + .../common/candidate_sources/salsa/README.md | 10 + ...mentDirectFollowSalsaExpansionSource.scala | 40 + .../salsa/SalsaExpander.scala | 117 + .../SalsaExpansionBasedCandidateSource.scala | 32 + .../common/candidate_sources/sims/BUILD | 24 + .../sims/CacheBasedSimsStore.scala | 50 + .../sims/DBV2SimsRefreshStore.scala | 35 + .../sims/DBV2SimsStore.scala | 38 + .../Follow2vecNearestNeighborsStore.scala | 69 + .../common/candidate_sources/sims/README.md | 32 + .../sims/SimsExperimentalStore.scala | 36 + .../sims/SimsSourceFSConfig.scala | 14 + .../sims/SimsSourceParams.scala | 16 + .../candidate_sources/sims/SimsStore.scala | 36 + .../sims/StratoBasedSimsCandidateSource.scala | 40 + ...BasedSimsCandidateSourceWithUnitView.scala | 10 + .../sims/SwitchingSimsSource.scala | 55 + .../candidate_sources/sims_expansion/BUILD | 23 + .../DBV2SimsExpansionParams.scala | 22 + .../sims_expansion/README.md | 6 + ...RecentEngagementSimilarUsersFSConfig.scala | 14 + .../RecentEngagementSimilarUsersParams.scala | 17 + .../RecentEngagementSimilarUsersSource.scala | 113 + .../RecentFollowingSimilarUsersParams.scala | 29 + .../RecentFollowingSimilarUsersSource.scala | 99 + ...gementDirectFollowSimilarUsersSource.scala | 53 + .../SimsExpansionBasedCandidateSource.scala | 114 + .../SimsExpansionFSConfig.scala | 26 + .../SimsExpansionSourceParams.scala | 17 + .../candidate_sources/socialgraph/BUILD | 18 + .../candidate_sources/socialgraph/README.md | 6 + ...lowingRecentFollowingExpansionSource.scala | 102 + ...centFollowingExpansionSourceFSConfig.scala | 16 + ...RecentFollowingExpansionSourceParams.scala | 10 + .../common/candidate_sources/stp/BUILD | 28 + .../stp/BaseOnlineSTPSource.scala | 55 + .../candidate_sources/stp/Dbv2StpScorer.scala | 30 + .../candidate_sources/stp/EpStpScorer.scala | 65 + ...utualFollowStrongTiePredictionSource.scala | 61 + .../OfflineMutualFollowExpansionSource.scala | 23 + .../stp/OfflineStpSourceFsConfig.scala | 14 + .../stp/OfflineStpSourceParams.scala | 9 + .../OfflineStpSourceWithDensePmiMatrix.scala | 22 + .../OfflineStpSourceWithLegacyPmiMatrix.scala | 23 + ...OfflineStrongTiePredictionBaseSource.scala | 57 + .../OfflineStrongTiePredictionSource.scala | 44 + .../stp/OnlineSTPSourceFSConfig.scala | 15 + .../stp/OnlineSTPSourceParams.scala | 19 + .../stp/OnlineSTPSourceScorer.scala | 29 + .../OnlineSTPSourceWithDeepbirdV2Scorer.scala | 76 + .../stp/OnlineSTPSourceWithEPScorer.scala | 58 + .../common/candidate_sources/stp/README.md | 47 + .../stp/STPFirstDegreeFetcher.scala | 155 + .../stp/STPGraphBuilder.scala | 32 + .../stp/STPSecondDegreeFetcher.scala | 94 + ...rcedOfflineStrongTiePredictionSource.scala | 28 + .../common/candidate_sources/stp/img.png | Bin 0 -> 54417 bytes .../top_organic_follows_accounts/BUILD | 22 + .../top_organic_follows_accounts/README.md | 2 + .../TopOrganicFollowsAccountsFSConfig.scala | 18 + .../TopOrganicFollowsAccountsParams.scala | 31 + .../TopOrganicFollowsAccountsSource.scala | 110 + .../candidate_sources/triangular_loops/BUILD | 21 + .../triangular_loops/README.md | 5 + .../TriangularLoopsFSConfig.scala | 12 + .../TriangularLoopsParams.scala | 11 + .../TriangularLoopsSource.scala | 91 + .../two_hop_random_walk/BUILD | 20 + .../two_hop_random_walk/README.md | 7 + .../TwoHopRandomWalkSource.scala | 40 + .../candidate_sources/user_user_graph/BUILD | 18 + .../user_user_graph/README.md | 4 + .../UserUserGraphCandidateSource.scala | 125 + .../UserUserGraphFSConfig.scala | 15 + .../user_user_graph/UserUserGraphParams.scala | 19 + .../addressbook/AddressbookClient.scala | 221 + .../addressbook/AddressbookModule.scala | 10 + .../common/clients/addressbook/BUILD | 21 + .../common/clients/addressbook/models/BUILD | 10 + .../clients/addressbook/models/Contact.scala | 29 + .../clients/addressbook/models/EdgeType.scala | 16 + .../addressbook/models/QueryOption.scala | 24 + .../addressbook/models/RecordIdentifier.scala | 10 + .../common/clients/adserver/AdRequest.scala | 45 + .../clients/adserver/AdserverClient.scala | 16 + .../clients/adserver/AdserverModule.scala | 15 + .../common/clients/adserver/BUILD | 14 + .../common/clients/cache/BUILD | 15 + .../common/clients/cache/MemcacheClient.scala | 121 + .../common/clients/cache/MemcacheModule.scala | 30 + .../clients/cache/ThriftBijection.scala | 81 + .../common/clients/common/BUILD | 11 + .../clients/common/BaseClientModule.scala | 20 + .../common/clients/deepbirdv2/BUILD | 20 + ...pBirdV2PredictionServiceClientModule.scala | 67 + .../common/clients/dismiss_store/BUILD | 19 + .../clients/dismiss_store/DismissStore.scala | 60 + .../clients/email_storage_service/BUILD | 14 + .../EmailStorageServiceClient.scala | 28 + .../EmailStorageServiceModule.scala | 12 + .../common/clients/geoduck/BUILD | 22 + .../geoduck/LocationServiceClient.scala | 62 + .../geoduck/LocationServiceModule.scala | 12 + .../geoduck/ReverseGeocodeClient.scala | 57 + .../clients/geoduck/UserLocationFetcher.scala | 59 + .../common/clients/gizmoduck/BUILD | 21 + .../clients/gizmoduck/GizmoduckClient.scala | 81 + .../clients/gizmoduck/GizmoduckModule.scala | 24 + .../clients/graph_feature_service/BUILD | 14 + .../GraphFeatureServiceClient.scala | 50 + .../GraphFeatureStoreModule.scala | 12 + .../common/clients/impression_store/BUILD | 18 + .../ImpressionStoreModule.scala | 31 + .../impression_store/WtfImpressionStore.scala | 42 + .../common/clients/interests_service/BUILD | 14 + .../InterestServiceClient.scala | 115 + .../clients/phone_storage_service/BUILD | 14 + .../PhoneStorageServiceClient.scala | 34 + .../PhoneStorageServiceModule.scala | 12 + .../common/clients/real_time_real_graph/BUILD | 20 + .../real_time_real_graph/Engagement.scala | 14 + .../EngagementScorer.scala | 58 + .../RealTimeRealGraphClient.scala | 128 + .../common/clients/socialgraph/BUILD | 26 + .../socialgraph/SocialGraphClient.scala | 421 ++ .../socialgraph/SocialGraphModule.scala | 25 + .../common/clients/strato/BUILD | 30 + .../clients/strato/StratoClientModule.scala | 249 ++ .../common/clients/user_state/BUILD | 17 + .../clients/user_state/UserStateClient.scala | 83 + .../common/constants/BUILD | 9 + .../CandidateAlgorithmTypeConstants.scala | 91 + .../constants/GuiceNamedConstants.scala | 43 + .../common/constants/ServiceConstants.scala | 15 + .../common/feature_hydration/adapters/BUILD | 82 + .../adapters/CandidateAlgorithmAdapter.scala | 72 + .../adapters/ClientContextAdapter.scala | 79 + .../adapters/PostNuxAlgorithmAdapter.scala | 151 + .../adapters/PreFetchedFeatureAdapter.scala | 91 + .../common/feature_hydration/common/BUILD | 18 + .../common/FeatureSource.scala | 23 + .../common/FeatureSourceId.scala | 19 + .../common/HasPreFetchedFeature.scala | 25 + .../common/feature_hydration/sources/BUILD | 59 + .../sources/CandidateAlgorithmSource.scala | 73 + .../sources/ClientContextSource.scala | 43 + .../FeatureHydrationSourcesFSConfig.scala | 42 + ...ureHydrationSourcesFeatureSwitchKeys.scala | 42 + .../sources/FeatureStoreFeatures.scala | 342 ++ .../sources/FeatureStoreGizmoduckSource.scala | 188 + .../sources/FeatureStoreParameters.scala | 79 + .../FeatureStorePostNuxAlgorithmSource.scala | 232 + .../sources/FeatureStoreSource.scala | 368 ++ .../sources/FeatureStoreSourceParams.scala | 148 + .../FeatureStoreTimelinesAuthorSource.scala | 191 + .../FeatureStoreUserMetricCountsSource.scala | 187 + .../sources/HydrationSourcesModule.scala | 152 + .../sources/PreFetchedFeatureSource.scala | 36 + .../sources/UserScoringFeatureSource.scala | 86 + .../feature_hydration/sources/Utils.scala | 30 + .../common/features/BUILD | 9 + .../common/features/LocationFeature.scala | 10 + .../features/TrackingTokenFeature.scala | 8 + .../common/features/UserStateFeature.scala | 7 + .../common/models/AddressBookMetadata.scala | 29 + .../common/models/AlgorithmType.scala | 20 + .../common/models/BUILD | 29 + .../common/models/CandidateUser.scala | 192 + .../models/ClientContextConverter.scala | 53 + .../common/models/DisplayLocation.scala | 420 ++ .../common/models/EngagementType.scala | 62 + .../common/models/FilterReason.scala | 133 + .../common/models/FlowContext.scala | 20 + .../common/models/FlowRecommendation.scala | 23 + .../common/models/GeohashAndCountryCode.scala | 3 + .../common/models/HasAdMetadata.scala | 23 + .../common/models/HasByfSeedUserIds.scala | 5 + .../common/models/HasDataRecord.scala | 86 + .../common/models/HasDebugOptions.scala | 30 + .../common/models/HasDismissedUserIds.scala | 6 + .../common/models/HasDisplayLocation.scala | 5 + .../common/models/HasEngagements.scala | 7 + .../common/models/HasExcludedUserIds.scala | 6 + .../models/HasGeohashAndCountryCode.scala | 5 + .../models/HasInfoPerRankingStage.scala | 5 + .../common/models/HasInterestIds.scala | 11 + .../HasInvalidRelationshipUserIds.scala | 6 + .../common/models/HasIsSoftUser.scala | 5 + .../models/HasMutualFollowedUserIds.scala | 10 + .../HasPreviousRecommendationsContext.scala | 12 + .../common/models/HasProfileId.scala | 5 + .../common/models/HasQualityFactor.scala | 5 + .../models/HasRecentFollowedByUserIds.scala | 8 + .../models/HasRecentFollowedUserIds.scala | 14 + .../HasRecentFollowedUserIdsWithTime.scala | 9 + .../models/HasRecentlyEngagedUserIds.scala | 5 + .../HasRecommendationFlowIdentifier.scala | 5 + .../common/models/HasScores.scala | 5 + .../common/models/HasSimilarToContext.scala | 7 + .../common/models/HasTopicId.scala | 5 + .../HasUserCandidateSourceDetails.scala | 162 + .../common/models/HasUserState.scala | 7 + .../common/models/HasWtfImpressions.scala | 30 + .../common/models/OptimusRequest.scala | 15 + .../common/models/Product.scala | 15 + .../common/models/RankingInfo.scala | 28 + .../common/models/Reason.scala | 206 + .../common/models/RecentlyEngagedUserId.scala | 31 + .../common/models/RecommendationStep.scala | 30 + .../common/models/STPGraph.scala | 22 + .../common/models/SafetyLevel.scala | 17 + .../common/models/Score.scala | 144 + .../common/models/Session.scala | 16 + .../common/models/SignalData.scala | 42 + .../common/models/TrackingToken.scala | 62 + .../common/models/TweetCandidate.scala | 6 + .../models/UserCandidateSourceDetails.scala | 97 + .../common/models/UserIdAndTimestamp.scala | 3 + .../common/models/WtfImpression.scala | 12 + .../common/predicates/BUILD | 21 + .../predicates/CandidateParamPredicate.scala | 21 + .../CandidateSourceParamPredicate.scala | 31 + .../CuratedCompetitorListPredicate.scala | 66 + .../predicates/ExcludedUserIdPredicate.scala | 24 + .../common/predicates/InactivePredicate.scala | 121 + .../predicates/InactivePredicateParams.scala | 21 + ...reviouslyRecommendedUserIdsPredicate.scala | 34 + .../common/predicates/dismiss/BUILD | 17 + .../dismiss/DismissedCandidatePredicate.scala | 32 + .../DismissedCandidatePredicateParams.scala | 9 + .../common/predicates/gizmoduck/BUILD | 23 + .../gizmoduck/GizmoduckPredicate.scala | 284 ++ .../gizmoduck/GizmoduckPredicateCache.scala | 50 + .../GizmoduckPredicateFSConfig.scala | 17 + .../gizmoduck/GizmoduckPredicateParams.scala | 21 + .../common/predicates/health/BUILD | 21 + .../predicates/health/HssPredicate.scala | 95 + .../health/HssPredicateFSConfig.scala | 22 + .../health/HssPredicateParams.scala | 34 + .../common/predicates/sgs/BUILD | 19 + .../sgs/InvalidRelationshipPredicate.scala | 36 + .../sgs/RecentFollowingPredicate.scala | 33 + .../predicates/sgs/SgsPredicateFSConfig.scala | 16 + .../predicates/sgs/SgsPredicateParams.scala | 19 + .../SgsRelationshipsByUserIdPredicate.scala | 113 + .../sgs/SgsRelationshipsPredicate.scala | 146 + .../common/predicates/user_activity/BUILD | 20 + .../user_activity/UserActivityPredicate.scala | 161 + .../UserActivityPredicateParams.scala | 10 + .../common/AdhocScoreModificationType.scala | 20 + .../common/rankers/common/BUILD | 10 + .../rankers/common/DedupCandidates.scala | 11 + .../common/rankers/common/RankerId.scala | 27 + .../common/rankers/fatigue_ranker/BUILD | 13 + .../ImpressionBasedFatigueRanker.scala | 141 + ...ImpressionBasedFatigueRankerFSConfig.scala | 12 + .../ImpressionBasedFatigueRankerParams.scala | 14 + .../common/rankers/first_n_ranker/BUILD | 20 + .../rankers/first_n_ranker/FirstNRanker.scala | 115 + .../first_n_ranker/FirstNRankerFSConfig.scala | 21 + .../FirstNRankerFeatureSwitchKeys.scala | 8 + .../first_n_ranker/FirstNRankerParams.scala | 26 + .../common/rankers/interleave_ranker/BUILD | 21 + .../interleave_ranker/InterleaveRanker.scala | 204 + .../InterleaveRankerFSConfig.scala | 12 + .../InterleaveRankerParams.scala | 8 + .../common/rankers/ml_ranker/ranking/BUILD | 37 + .../ranking/HydrateFeaturesTransform.scala | 57 + .../rankers/ml_ranker/ranking/MlRanker.scala | 219 + .../ml_ranker/ranking/MlRankerFSConfig.scala | 12 + .../ml_ranker/ranking/MlRankerParams.scala | 30 + .../ml_ranker/scoring/AdhocScorer.scala | 28 + .../common/rankers/ml_ranker/scoring/BUILD | 23 + .../ml_ranker/scoring/DeepbirdScorer.scala | 151 + .../scoring/PostnuxDeepbirdProdScorer.scala | 34 + .../ml_ranker/scoring/RandomScorer.scala | 42 + .../rankers/ml_ranker/scoring/Scorer.scala | 34 + .../ml_ranker/scoring/ScorerFactory.scala | 38 + .../common/rankers/utils/BUILD | 8 + .../common/rankers/utils/Utils.scala | 28 + .../weighted_candidate_source_ranker/BUILD | 20 + .../CandidateShuffle.scala | 36 + .../WeightMethod.scala | 6 + .../WeightedCandidateSourceBaseRanker.scala | 118 + .../WeightedCandidateSourceRanker.scala | 100 + ...eightedCandidateSourceRankerFSConfig.scala | 13 + .../WeightedCandidateSourceRankerParams.scala | 8 + .../common/stores/BUILD | 19 + .../stores/LowTweepCredFollowStore.scala | 39 + .../common/transforms/dedup/BUILD | 8 + .../transforms/dedup/DedupTransform.scala | 14 + .../transforms/modify_social_proof/BUILD | 22 + .../ModifySocialProofTransform.scala | 202 + .../RemoveAccountProofTransform.scala | 27 + .../common/transforms/ranker_id/BUILD | 19 + .../ranker_id/RandomRankerIdTransform.scala | 24 + ...ecommendationFlowIdentifierTransform.scala | 20 + .../recommendation_flow_identifier/BUILD | 9 + .../common/transforms/tracking_token/BUILD | 18 + .../TrackingTokenTransform.scala | 76 + .../common/transforms/weighted_sampling/BUILD | 10 + .../weighted_sampling/SamplingTransform.scala | 138 + .../SamplingTransformFSConfig.scala | 19 + .../SamplingTransformParams.scala | 25 + .../follow_recommendations/common/utils/BUILD | 13 + .../common/utils/CollectionUtil.scala | 22 + .../DisplayLocationProductConverterUtil.scala | 27 + .../common/utils/MergeUtil.scala | 51 + .../common/utils/RandomUtil.scala | 88 + .../common/utils/RescueWithStatsUtils.scala | 50 + .../common/utils/UserSignupUtil.scala | 14 + .../common/utils/Weighted.scala | 21 + .../server/src/main/resources/BUILD | 20 + .../src/main/resources/config/decider.yml | 129 + .../server/src/main/resources/logback.xml | 133 + .../quality/stp_models/20141223/epModel | 8 + .../stp_models/20141223/trainingConfig | 1 + .../com/twitter/follow_recommendations/BUILD | 48 + ...owRecommendationsServiceThriftServer.scala | 118 + .../assembler/models/Action.scala | 9 + .../assembler/models/BUILD | 12 + .../assembler/models/Config.scala | 8 + .../assembler/models/FeedbackAction.scala | 13 + .../assembler/models/Footer.scala | 9 + .../assembler/models/Header.scala | 9 + .../assembler/models/Layout.scala | 16 + .../models/RecommendationOptions.scala | 11 + .../assembler/models/SocialProof.scala | 16 + .../assembler/models/Title.scala | 9 + .../assembler/models/WTFPresentation.scala | 47 + .../follow_recommendations/blenders/BUILD | 16 + .../blenders/PromotedAccountsBlender.scala | 138 + .../follow_recommendations/configapi/BUILD | 28 + .../configapi/ConfigBuilder.scala | 16 + .../configapi/DeciderConfigs.scala | 52 + .../configapi/FeatureSwitchConfigs.scala | 138 + .../configapi/GlobalFeatureSwitchConfig.scala | 49 + .../configapi/ParamsFactory.scala | 29 + .../configapi/RequestContext.scala | 19 + .../configapi/RequestContextFactory.scala | 74 + .../configapi/candidates/BUILD | 18 + .../candidates/CandidateUserContext.scala | 19 + .../CandidateUserContextFactory.scala | 55 + .../CandidateUserParamsFactory.scala | 35 + .../HydrateCandidateParamsTransform.scala | 21 + .../configapi/common/BUILD | 8 + .../common/FeatureSwitchConfig.scala | 60 + .../configapi/deciders/BUILD | 10 + .../configapi/deciders/DeciderKey.scala | 51 + .../configapi/deciders/DeciderParams.scala | 8 + .../configapi/params/BUILD | 13 + .../configapi/params/GlobalParams.scala | 35 + .../follow_recommendations/controllers/BUILD | 29 + .../CandidateUserDebugParamsBuilder.scala | 25 + .../RecommendationRequestBuilder.scala | 41 + .../RequestBuilderUserFetcher.scala | 48 + .../ScoringUserRequestBuilder.scala | 53 + .../controllers/ThriftController.scala | 41 + .../follow_recommendations/flows/ads/BUILD | 19 + .../flows/ads/PromotedAccountsFlow.scala | 112 + .../ads/PromotedAccountsFlowParams.scala | 19 + .../ads/PromotedAccountsFlowRequest.scala | 33 + .../flows/ads/PromotedAccountsUtil.scala | 28 + .../flows/content_recommender_flow/BUILD | 32 + .../ContentRecommenderFlow.scala | 202 + ...commenderFlowCandidateSourceRegistry.scala | 78 + ...ecommenderFlowCandidateSourceWeights.scala | 71 + ...nderFlowCandidateSourceWeightsParams.scala | 117 + .../ContentRecommenderFlowFSConfig.scala | 60 + ...tentRecommenderFlowFeatureSwitchKeys.scala | 70 + .../ContentRecommenderParams.scala | 85 + .../ContentRecommenderRequest.scala | 45 + .../ContentRecommenderRequestBuilder.scala | 121 + .../flows/post_nux_ml/BUILD | 58 + .../PostNuxMlCandidateSourceRegistry.scala | 103 + ...PostNuxMlCandidateSourceWeightParams.scala | 177 + .../PostNuxMlCombinedRankerBuilder.scala | 193 + .../flows/post_nux_ml/PostNuxMlFlow.scala | 304 ++ .../PostNuxMlFlowCandidateSourceWeights.scala | 68 + ...didateSourceWeightsFeatureSwitchKeys.scala | 46 + .../post_nux_ml/PostNuxMlFlowFSConfig.scala | 80 + .../PostNuxMlFlowFeatureSwitchKeys.scala | 27 + .../flows/post_nux_ml/PostNuxMlParams.scala | 133 + .../flows/post_nux_ml/PostNuxMlRequest.scala | 54 + .../post_nux_ml/PostNuxMlRequestBuilder.scala | 173 + .../PostNuxMlRequestBuilderParams.scala | 45 + .../follow_recommendations/logging/BUILD | 18 + .../logging/FrsLogger.scala | 164 + .../follow_recommendations/models/BUILD | 13 + .../models/CandidateSourceType.scala | 9 + .../models/CandidateUserDebugParams.scala | 5 + .../models/DebugParams.scala | 28 + .../models/DisplayContext.scala | 113 + .../models/FeatureValue.scala | 24 + .../models/RecommendationFlowData.scala | 104 + .../models/RecommendationRequest.scala | 29 + .../models/RecommendationResponse.scala | 14 + .../models/Request.scala | 22 + .../models/ScoringUserRequest.scala | 45 + .../models/ScoringUserResponse.scala | 15 + .../models/failures/BUILD | 8 + .../failures/TimeoutPipelineFailure.scala | 12 + .../modules/ABDeciderModule.scala | 31 + .../follow_recommendations/modules/BUILD | 24 + .../modules/ConfigApiModule.scala | 20 + .../modules/DiffyModule.scala | 71 + .../modules/FeatureSwitchesModule.scala | 85 + .../modules/FlagsModule.scala | 18 + .../modules/ProductRegistryModule.scala | 12 + .../modules/ScorerModule.scala | 40 + .../modules/ScribeModule.scala | 95 + .../modules/TimerModule.scala | 13 + .../follow_recommendations/products/BUILD | 16 + .../products/ProdProductRegistry.scala | 44 + .../products/common/BUILD | 12 + .../products/common/Exceptions.scala | 7 + .../products/common/Product.scala | 56 + .../products/common/ProductRegistry.scala | 9 + .../products/explore_tab/BUILD | 14 + .../explore_tab/ExploreTabProduct.scala | 50 + .../products/explore_tab/configapi/BUILD | 9 + .../configapi/ExploreTabFSConfig.scala | 14 + .../configapi/ExploreTabParams.scala | 10 + .../products/home_timeline/BUILD | 14 + .../home_timeline/HTLProductMixer.scala | 9 + .../home_timeline/HomeTimelineProduct.scala | 114 + .../home_timeline/HomeTimelineStrings.scala | 26 + .../products/home_timeline/configapi/BUILD | 9 + .../configapi/HomeTimelineFSConfig.scala | 22 + .../configapi/HomeTimelineParams.scala | 38 + .../products/home_timeline_tweet_recs/BUILD | 13 + .../HomeTimelineTweetRecsProduct.scala | 50 + .../home_timeline_tweet_recs/configapi/BUILD | 10 + .../HomeTimelineTweetRecsParams.scala | 7 + .../products/sidebar/BUILD | 14 + .../products/sidebar/SidebarProduct.scala | 73 + .../products/sidebar/configapi/BUILD | 8 + .../sidebar/configapi/SidebarParams.scala | 9 + .../follow_recommendations/services/BUILD | 34 + ...wRecommendationsServiceWarmupHandler.scala | 101 + .../ProductMixerRecommendationService.scala | 72 + .../services/ProductPipelineSelector.scala | 188 + .../ProductPipelineSelectorConfig.scala | 19 + .../services/ProductRecommenderService.scala | 72 + .../services/RecommendationsService.scala | 28 + .../services/UserScoringService.scala | 84 + .../services/exceptions/BUILD | 14 + .../exceptions/UnknownExceptionMapper.scala | 18 + .../follow_recommendations/utils/BUILD | 29 + .../utils/CandidateSourceHoldbackUtil.scala | 82 + ...ecommendationFlowBaseSideEffectsUtil.scala | 121 + .../thrift/src/main/thrift/BUILD | 21 + .../thrift/src/main/thrift/assembler.thrift | 42 + .../src/main/thrift/client_context.thrift | 19 + .../thrift/src/main/thrift/debug.thrift | 73 + .../src/main/thrift/display_context.thrift | 62 + .../src/main/thrift/display_location.thrift | 55 + .../src/main/thrift/engagementType.thrift | 11 + .../thrift/src/main/thrift/flows.thrift | 20 + .../follow-recommendations-service.thrift | 100 + ...low_recommendations_serving_history.thrift | 9 + .../thrift/src/main/thrift/logging/BUILD | 18 + .../main/thrift/logging/client_context.thrift | 14 + .../src/main/thrift/logging/debug.thrift | 8 + .../thrift/logging/display_context.thrift | 66 + .../thrift/logging/display_location.thrift | 55 + .../main/thrift/logging/engagementType.thrift | 11 + .../src/main/thrift/logging/flows.thrift | 16 + .../src/main/thrift/logging/logs.thrift | 72 + .../src/main/thrift/logging/reasons.thrift | 62 + .../logging/recently_engaged_user_id.thrift | 10 + .../thrift/logging/recommendations.thrift | 26 + .../src/main/thrift/logging/scoring.thrift | 38 + .../src/main/thrift/logging/tracking.thrift | 16 + .../thrift/src/main/thrift/reasons.thrift | 61 + .../thrift/recently_engaged_user_id.thrift | 10 + .../src/main/thrift/recommendations.thrift | 40 + .../thrift/src/main/thrift/scoring.thrift | 49 + .../thrift/src/main/thrift/tracking.thrift | 17 + graph-feature-service/BUILD.bazel | 67 + graph-feature-service/README.md | 3 + graph-feature-service/doc/common.md | 62 + graph-feature-service/doc/getintersection.md | 43 + .../graph_feature_service/common/BUILD.bazel | 8 + .../common/Configs.scala | 73 + .../graph_feature_service/server/BUILD.bazel | 35 + .../graph_feature_service/server/Main.scala | 56 + .../server/controllers/ServerController.scala | 46 + .../ServerGetIntersectionHandler.scala | 198 + .../server/handlers/ServerWarmupHandler.scala | 45 + .../modules/GetIntersectionStoreModule.scala | 91 + ...aphFeatureServiceWorkerClientsModule.scala | 51 + .../server/modules/LZ4Injection.scala | 17 + .../server/modules/ServerFlagModule.scala | 31 + .../server/stores/FeatureTypesEncoder.scala | 16 + .../server/stores/GetIntersectionStore.scala | 181 + .../twitter/graph_feature_service/util/BUILD | 7 + .../util/FeatureTypesCalculator.scala | 58 + .../util/IntersectionValueCalculator.scala | 242 ++ .../graph_feature_service/worker/BUILD.bazel | 30 + .../graph_feature_service/worker/Main.scala | 58 + .../worker/controllers/WorkerController.scala | 38 + .../WorkerGetIntersectionHandler.scala | 105 + .../worker/handlers/WorkerWarmupHandler.scala | 14 + .../GraphContainerProviderModule.scala | 62 + .../worker/modules/WorkerFlagModule.scala | 33 + .../worker/util/AutoUpdatingGraph.scala | 69 + .../worker/util/GfsQuery.scala | 14 + .../worker/util/GraphContainer.scala | 19 + .../worker/util/GraphKey.scala | 32 + .../worker/util/GraphType.scala | 16 + .../scalding/BUILD.bazel | 66 + .../scalding/EdgeFeature.scala | 9 + .../scalding/GraphFeatureServiceAppBase.scala | 85 + .../scalding/GraphFeatureServiceApps.scala | 52 + .../scalding/GraphFeatureServiceMainJob.scala | 297 ++ .../scalding/adhoc/BUILD.bazel | 27 + .../adhoc/RandomRequestGenerationApp.scala | 77 + .../com/twitter/graph_feature_service/BUILD | 15 + .../graph_feature_service.thrift | 123 + home-mixer/BUILD.bazel | 30 + home-mixer/README.md | 102 + .../scala/com/twitter/home_mixer/BUILD.bazel | 46 + .../HomeMixerHttpServerWarmupHandler.scala | 18 + .../twitter/home_mixer/HomeMixerServer.scala | 118 + .../HomeMixerThriftServerWarmupHandler.scala | 73 + .../home_mixer/candidate_pipeline/BUILD.bazel | 36 + ...sationServiceCandidatePipelineConfig.scala | 107 + ...erviceCandidatePipelineConfigBuilder.scala | 34 + ...ionServiceResponseFeatureTransformer.scala | 39 + .../EditedTweetsCandidatePipelineConfig.scala | 84 + ...NewTweetsPillCandidatePipelineConfig.scala | 123 + ...ineServiceResponseFeatureTransformer.scala | 34 + .../twitter/home_mixer/controller/BUILD.bazel | 20 + .../controller/HomeThriftController.scala | 50 + .../candidate_source/BUILD.bazel | 22 + .../EarlybirdCandidateSource.scala | 44 + .../SimilarityBasedUsersCandidateSource.scala | 34 + .../StaleTweetsCacheCandidateSource.scala | 30 + .../AuthorChildFeedbackActionBuilder.scala | 34 + .../decorator/BUILD.bazel | 32 + .../BlockUserChildFeedbackActionBuilder.scala | 54 + .../DontLikeFeedbackActionBuilder.scala | 88 + .../EngagerSocialContextBuilder.scala | 119 + .../ExtendedReplySocialContextBuilder.scala | 78 + .../decorator/FeedbackUtil.scala | 61 + .../FollowedBySocialContextBuilder.scala | 53 + .../HomeAdsClientEventDetailsBuilder.scala | 46 + .../HomeClientEventDetailsBuilder.scala | 92 + ...onversationServiceCandidateDecorator.scala | 49 + .../HomeFeedbackActionInfoBuilder.scala | 54 + .../decorator/HomeQueryTypePredicates.scala | 18 + .../HomeTimelinesScoreInfoBuilder.scala | 26 + .../HomeTweetSocialContextBuilder.scala | 44 + .../decorator/HomeTweetTypePredicates.scala | 250 ++ .../LikedBySocialContextBuilder.scala | 54 + ...onversationServiceCandidateDecorator.scala | 47 + .../MuteUserChildFeedbackActionBuilder.scala | 55 + ...InterestedTopicFeedbackActionBuilder.scala | 71 + ...otRelevantChildFeedbackActionBuilder.scala | 55 + .../ReceivedReplySocialContextBuilder.scala | 76 + ...eportTweetChildFeedbackActionBuilder.scala | 38 + .../RetweeterChildFeedbackActionBuilder.scala | 39 + .../decorator/TopicSocialContextBuilder.scala | 42 + ...followUserChildFeedbackActionBuilder.scala | 57 + .../YouMightLikeSocialContextBuilder.scala | 50 + .../decorator/builder/BUILD.bazel | 23 + .../builder/HomeClientEventInfoBuilder.scala | 44 + ...omeConversationModuleMetadataBuilder.scala | 30 + .../ListClientEventDetailsBuilder.scala | 33 + ...wAlertAndShowCoverInstructionBuilder.scala | 30 + .../decorator/urt/builder/BUILD.bazel | 12 + ...WhoToFollowFeedbackActionInfoBuilder.scala | 51 + .../AncestorFeatureHydrator.scala | 56 + .../AuthorFeatureHydrator.scala | 95 + .../feature_hydrator/BUILD.bazel | 103 + .../DismissInfoQueryFeatureHydrator.scala | 45 + .../EarlybirdFeatureHydrator.scala | 129 + .../FeedbackHistoryQueryFeatureHydrator.scala | 38 + .../FocalTweetFeatureHydrator.scala | 84 + .../FollowedTopicsQueryFeatureHydrator.scala | 41 + ...GizmoduckAuthorSafetyFeatureHydrator.scala | 58 + .../GizmoduckUserQueryFeatureHydrator.scala | 50 + .../GraphTwoHopFeatureHydrator.scala | 105 + ...ssionBloomFilterQueryFeatureHydrator.scala | 57 + ...stNonPollingTimeQueryFeatureHydrator.scala | 68 + .../ListMembersQueryFeatureHydrator.scala | 42 + ...ricCenterUserCountingFeatureHydrator.scala | 81 + .../NamesFeatureHydrator.scala | 97 + ...PersistenceStoreQueryFeatureHydrator.scala | 95 + ...FilteredSocialContextFeatureHydrator.scala | 71 + ...hInNetworkScoresQueryFeatureHydrator.scala | 42 + .../RealGraphQueryFeatureHydrator.scala | 48 + ...RealGraphViewerAuthorFeatureHydrator.scala | 123 + ...aphViewerRelatedUsersFeatureHydrator.scala | 74 + ...eInteractionGraphEdgeFeatureHydrator.scala | 64 + ...nGraphUserVertexQueryFeatureHydrator.scala | 49 + .../ReplyFeatureHydrator.scala | 196 + .../RequestQueryFeatureHydrator.scala | 128 + .../RetweetSourceTweetFeatureHydrator.scala | 76 + ...SGSFollowedUsersQueryFeatureHydrator.scala | 46 + ...SGSValidSocialContextFeatureHydrator.scala | 105 + ...sEngagementSimilarityFeatureHydrator.scala | 83 + .../SocialGraphServiceFeatureHydrator.scala | 67 + .../TSPInferredTopicFeatureHydrator.scala | 162 + .../TimeFeaturesHydrator.scala | 251 ++ ...ineServiceTweetsQueryFeatureHydrator.scala | 63 + ...TweetImpressionsQueryFeatureHydrator.scala | 87 + .../TweetMetaDataFeatureHydrator.scala | 66 + .../TweetypieContentFeatureHydrator.scala | 149 + .../TweetypieFeatureHydrator.scala | 156 + ...eetypieStaticEntitiesFeatureHydrator.scala | 161 + ...nAuthorFollow20220101FeatureHydrator.scala | 96 + ...inUserEngagementQueryFeatureHydrator.scala | 80 + .../TwhinUserFollowQueryFeatureHydrator.scala | 80 + .../UserFollowedTopicIdsFeatureHydrator.scala | 84 + .../UserLanguagesFeatureHydrator.scala | 35 + .../UserStateQueryFeatureHydrator.scala | 54 + .../UtegFeatureHydrator.scala | 88 + .../AuthorFeaturesAdapter.scala | 59 + .../adapters/author_features/BUILD.bazel | 17 + .../adapters/content/BUILD.bazel | 17 + .../content/ContentFeatureAdapter.scala | 260 ++ .../adapters/earlybird/BUILD.bazel | 19 + .../adapters/earlybird/EarlybirdAdapter.scala | 453 ++ .../adapters/inferred_topic/BUILD.bazel | 13 + .../inferred_topic/InferredTopicAdapter.scala | 25 + .../adapters/non_ml_features/BUILD.bazel | 15 + .../NonMLCandidateFeaturesAdapter.scala | 44 + .../NonMLCommonFeaturesAdapter.scala | 48 + .../adapters/offline_aggregates/BUILD.bazel | 13 + .../PassThroughAdapter.scala | 12 + .../SparseAggregatesToDenseAdapter.scala | 17 + .../adapters/twhin_embeddings/BUILD.bazel | 16 + .../TwhinEmbeddingsAdapter.scala | 81 + .../AggregateFeatureInfo.scala | 37 + ...ggregateFeaturesToDecodeWithMetadata.scala | 68 + .../offline_aggregates/BUILD.bazel | 38 + .../BaseAggregateQueryFeatureHydrator.scala | 76 + .../BaseEdgeAggregateFeatureHydrator.scala | 93 + .../EdgeAggregateFeatures.scala | 107 + .../PartAAggregateQueryFeatureHydrator.scala | 35 + .../PartBAggregateQueryFeatureHydrator.scala | 144 + .../Phase1EdgeAggregateFeatureHydrator.scala | 20 + .../Phase2EdgeAggregateFeatureHydrator.scala | 22 + .../offline_aggregates/Utils.scala | 36 + .../real_time_aggregates/BUILD.bazel | 25 + ...ggregateBulkCandidateFeatureHydrator.scala | 41 + ...ealTimeAggregateQueryFeatureHydrator.scala | 35 + .../BaseRealtimeAggregateHydrator.scala | 154 + ...thorRealTimeAggregateFeatureHydrator.scala | 52 + .../RealTimeAggregateTimeDecay.scala | 50 + ...mentRealTimeAggregateFeatureHydrator.scala | 64 + ...mentRealTimeAggregateFeatureHydrator.scala | 53 + ...mentRealTimeAggregateFeatureHydrator.scala | 55 + ...mentRealTimeAggregateFeatureHydrator.scala | 49 + ...mentRealTimeAggregateFeatureHydrator.scala | 57 + ...mentRealTimeAggregateFeatureHydrator.scala | 61 + ...entRealTimeAggregatesFeatureHydrator.scala | 56 + .../functional_component/filter/BUILD.bazel | 26 + .../filter/DropMaxCandidatesFilter.scala | 27 + .../filter/FeedbackFatigueFilter.scala | 89 + .../InvalidConversationModuleFilter.scala | 50 + ...BestOutOfNetworkTweetPerAuthorFilter.scala | 36 + .../filter/OutOfNetworkCompetitorFilter.scala | 38 + .../OutOfNetworkCompetitorURLFilter.scala | 37 + .../filter/PredicateFeatureFilter.scala | 59 + .../filter/PredicateGatedFilter.scala | 47 + .../filter/PreviouslySeenTweetsFilter.scala | 37 + .../PreviouslyServedAncestorsFilter.scala | 44 + .../filter/PreviouslyServedTweetsFilter.scala | 42 + .../filter/RejectTweetFromViewerFilter.scala | 24 + .../filter/RetweetDeduplicationFilter.scala | 45 + .../RetweetSourceTweetRemovingFilter.scala | 40 + .../filter/SocialContextFilter.scala | 57 + .../functional_component/gate/BUILD.bazel | 21 + .../gate/DismissFatigueGate.scala | 48 + .../gate/ExcludeSoftUserGate.scala | 23 + .../gate/MinCachedTweetsGate.scala | 34 + .../gate/NonEmptySeqFeatureGate.scala | 18 + .../gate/RequestContextGate.scala | 22 + .../gate/RequestContextNotGate.scala | 24 + .../gate/SupportedLanguagesGate.scala | 68 + ...nesPersistenceStoreLastInjectionGate.scala | 51 + .../gate/ViewerIsListOwnerGate.scala | 29 + .../query_transformer/BUILD.bazel | 13 + ...etsCandidatePipelineQueryTransformer.scala | 85 + .../functional_component/scorer/BUILD.bazel | 17 + .../scorer/FeedbackFatigueScorer.scala | 130 + .../scorer/OONTweetScalingScorer.scala | 48 + .../scorer/VerifiedAuthorScalingScorer.scala | 61 + .../functional_component/selector/BUILD.bazel | 24 + .../selector/DebunchCandidates.scala | 83 + .../selector/UpdateConversationModuleId.scala | 40 + .../UpdateHomeClientEventDetails.scala | 137 + .../UpdateNewTweetsPillDecoration.scala | 80 + .../side_effect/BUILD.bazel | 49 + .../side_effect/ClientEventsBuilder.scala | 177 + .../HomeScribeClientEventSideEffect.scala | 58 + .../HomeScribeServedEntriesSideEffect.scala | 212 + ...entSentImpressionsEventBusSideEffect.scala | 92 + ...ntSentImpressionsManhattanSideEffect.scala | 65 + ...dCandidateFeatureKeysKafkaSideEffect.scala | 112 + ...ateFeatureKeysKafkaSideEffectBuilder.scala | 20 + .../ServedCandidateKafkaSideEffect.scala | 50 + .../ServedCandidateKeysKafkaSideEffect.scala | 111 + ...dCandidateKeysKafkaSideEffectBuilder.scala | 20 + .../side_effect/ServedStatsSideEffect.scala | 80 + ...eTimelinesPersistenceStoreSideEffect.scala | 68 + ...pdateImpressionBloomFilterSideEffect.scala | 60 + .../UpdateLastNonPollingTimeSideEffect.scala | 78 + ...eTimelinesPersistenceStoreSideEffect.scala | 243 ++ .../home_mixer/marshaller/request/BUILD.bazel | 20 + .../request/DeviceContextUnmarshaller.scala | 19 + .../HomeMixerDebugParamsUnmarshaller.scala | 27 + .../HomeMixerProductContextUnmarshaller.scala | 54 + .../HomeMixerProductUnmarshaller.scala | 28 + .../HomeMixerRequestUnmarshaller.scala | 30 + .../marshaller/timeline_logging/BUILD.bazel | 13 + .../ConversationEntryMarshaller.scala | 16 + .../PromotedTweetEntryMarshaller.scala | 17 + .../TweetEntryMarshaller.scala | 29 + .../WhoToFollowEntryMarshaller.scala | 19 + .../marshaller/timelines/BUILD.bazel | 16 + .../ChronologicalCursorMarshaller.scala | 20 + .../ChronologicalCursorUnmarshaller.scala | 26 + .../timelines/DeviceContextMarshaller.scala | 34 + .../RecommendedUsersCursorUnmarshaller.scala | 20 + .../TimelineServiceCursorMarshaller.scala | 21 + ...ContextFunctionalityTypeUnmarshaller.scala | 22 + .../com/twitter/home_mixer/model/BUILD.bazel | 49 + .../model/ClearCacheIncludeInstruction.scala | 43 + .../home_mixer/model/ContentFeatures.scala | 96 + .../model/GapIncludeInstruction.scala | 63 + .../home_mixer/model/HomeAdsQuery.scala | 42 + .../home_mixer/model/HomeFeatures.scala | 266 ++ .../home_mixer/model/request/BUILD.bazel | 16 + .../model/request/DeviceContext.scala | 74 + .../home_mixer/model/request/HasListId.scala | 8 + .../model/request/HasSeenTweetIds.scala | 9 + .../model/request/HomeMixerDebugOptions.scala | 8 + .../model/request/HomeMixerProduct.scala | 35 + .../request/HomeMixerProductContext.scala | 34 + .../model/request/HomeMixerRequest.scala | 19 + ...rtiserBrandSafetySettingsStoreModule.scala | 56 + .../com/twitter/home_mixer/module/BUILD.bazel | 87 + ...ClientSentImpressionsPublisherModule.scala | 48 + .../module/ConversationServiceModule.scala | 37 + .../module/FeedbackHistoryClientModule.scala | 39 + .../module/HomeAdsCandidateSourceModule.scala | 32 + .../module/HomeMixerFlagsModule.scala | 53 + .../module/HomeMixerResourcesModule.scala | 18 + .../module/HomeNaviModelClientModule.scala | 52 + .../module/ImpressionBloomFilterModule.scala | 54 + .../module/InjectionHistoryClientModule.scala | 88 + .../module/ManhattanClientsModule.scala | 56 + .../ManhattanFeatureRepositoryModule.scala | 453 ++ .../ManhattanTweetImpressionStoreModule.scala | 52 + .../MemcachedFeatureRepositoryModule.scala | 113 + .../module/OptimizedStratoClientModule.scala | 46 + .../module/PeopleDiscoveryServiceModule.scala | 35 + .../PipelineFailureExceptionMapper.scala | 29 + .../RealGraphInNetworkScoresModule.scala | 26 + ...timeAggregateFeatureRepositoryModule.scala | 253 ++ .../module/ScoredTweetsMemcacheModule.scala | 65 + .../module/ScribeEventPublisherModule.scala | 125 + ...lustersRecentEngagementsClientModule.scala | 23 + .../module/StaleTweetsCacheModule.scala | 37 + .../ThriftFeatureRepositoryModule.scala | 375 ++ ...imelinesPersistenceStoreClientModule.scala | 43 + .../module/TweetyPieClientModule.scala | 51 + ...typieStaticEntitiesCacheClientModule.scala | 69 + .../module/UserMetadataStoreModule.scala | 26 + .../com/twitter/home_mixer/param/BUILD.bazel | 9 + .../param/GlobalParamConfigModule.scala | 10 + .../param/HomeGlobalParamConfig.scala | 40 + .../home_mixer/param/HomeGlobalParams.scala | 110 + .../home_mixer/param/HomeMixerFlagName.scala | 12 + .../param/HomeMixerInjectionNames.scala | 50 + .../home_mixer/param/decider/BUILD.bazel | 10 + .../home_mixer/param/decider/DeciderKey.scala | 40 + .../twitter/home_mixer/product/BUILD.bazel | 26 + .../product/HomeMixerProductModule.scala | 11 + .../HomeProductPipelineRegistryConfig.scala | 54 + .../home_mixer/product/following/BUILD.bazel | 95 + ...FollowingAdsCandidatePipelineBuilder.scala | 102 + ...wingEarlybirdCandidatePipelineConfig.scala | 54 + .../FollowingEarlybirdQueryTransformer.scala | 84 + ...gEarlybirdResponseFeatureTransformer.scala | 38 + .../FollowingMixerPipelineConfig.scala | 278 ++ .../FollowingProductPipelineConfig.scala | 131 + ...lowArmCandidatePipelineConfigBuilder.scala | 65 + .../product/following/model/BUILD.bazel | 22 + .../following/model/FollowingQuery.scala | 50 + .../model/HomeMixerExternalStrings.scala | 66 + .../product/following/param/BUILD.bazel | 14 + .../following/param/FollowingParam.scala | 85 + .../param/FollowingParamConfig.scala | 34 + .../home_mixer/product/for_you/BUILD.bazel | 87 + .../ForYouAdsCandidatePipelineBuilder.scala | 94 + ...sationServiceCandidatePipelineConfig.scala | 135 + .../for_you/ForYouProductPipelineConfig.scala | 135 + ...uScoredTweetsCandidatePipelineConfig.scala | 166 + ...orYouScoredTweetsMixerPipelineConfig.scala | 317 ++ ...oredTweetsResponseFeatureTransformer.scala | 78 + ...imelineScorerCandidatePipelineConfig.scala | 237 ++ ...YouTimelineScorerMixerPipelineConfig.scala | 336 ++ ...lineScorerResponseFeatureTransformer.scala | 189 + ...FollowCandidatePipelineConfigBuilder.scala | 59 + .../for_you/candidate_source/BUILD.bazel | 25 + .../ScoredTweetsProductCandidateSource.scala | 154 + .../product/for_you/model/BUILD.bazel | 21 + .../product/for_you/model/ForYouQuery.scala | 50 + .../for_you/model/ForYouTweetsResponse.scala | 5 + .../product/for_you/param/BUILD.bazel | 14 + .../product/for_you/param/ForYouParam.scala | 117 + .../for_you/param/ForYouParamConfig.scala | 43 + .../list_recommended_users/BUILD.bazel | 49 + ...berBasedUsersCandidatePipelineConfig.scala | 89 + ...BasedUsersResponseFeatureTransfromer.scala | 21 + ...tRecommendedUsersMixerPipelineConfig.scala | 100 + ...ecommendedUsersProductPipelineConfig.scala | 79 + .../feature_hydrator/BUILD.bazel | 16 + .../GizmoduckUserFeatureHydrator.scala | 59 + .../IsListMemberFeatureHydrator.scala | 53 + .../list_recommended_users/filter/BUILD.bazel | 13 + .../DropMaxCandidatesByScoreFilter.scala | 29 + .../filter/PreviouslyServedUsersFilter.scala | 35 + .../list_recommended_users/model/BUILD.bazel | 16 + .../model/ListFeatures.scala | 12 + .../model/ListRecommendedUsersQuery.scala | 30 + .../list_recommended_users/param/BUILD.bazel | 12 + .../param/ListRecommendedUsersParam.scala | 23 + .../ListRecommendedUsersParamConfig.scala | 22 + .../product/list_tweets/BUILD.bazel | 59 + ...istTweetsAdsCandidatePipelineBuilder.scala | 93 + .../ListTweetsMixerPipelineConfig.scala | 155 + .../ListTweetsProductPipelineConfig.scala | 94 + ...melineServiceCandidatePipelineConfig.scala | 54 + .../product/list_tweets/model/BUILD.bazel | 19 + .../list_tweets/model/ListTweetsQuery.scala | 38 + .../product/list_tweets/param/BUILD.bazel | 12 + .../list_tweets/param/ListTweetsParam.scala | 22 + .../param/ListTweetsParamConfig.scala | 23 + .../product/scored_tweets/BUILD.bazel | 57 + .../ScoredTweetsProductPipelineConfig.scala | 72 + ...edTweetsRecommendationPipelineConfig.scala | 254 ++ .../candidate_pipeline/BUILD.bazel | 41 + ...dScoredTweetsCandidatePipelineConfig.scala | 53 + ...TweetsCrMixerCandidatePipelineConfig.scala | 98 + ...oredTweetsFrsCandidatePipelineConfig.scala | 67 + ...eetsInNetworkCandidatePipelineConfig.scala | 82 + ...redTweetsUtegCandidatePipelineConfig.scala | 63 + .../candidate_source/BUILD.bazel | 12 + .../CachedScoredTweetsCandidateSource.scala | 24 + .../feature_hydrator/BUILD.bazel | 29 + ...chedScoredTweetsQueryFeatureHydrator.scala | 51 + .../scored_tweets/marshaller/BUILD.bazel | 19 + ...ScoredTweetsResponseDomainMarshaller.scala | 64 + ...redTweetsResponseTransportMarshaller.scala | 43 + .../product/scored_tweets/model/BUILD.bazel | 23 + .../model/ScoredTweetsQuery.scala | 39 + .../model/ScoredTweetsResponse.scala | 29 + .../product/scored_tweets/param/BUILD.bazel | 15 + .../param/ScoredTweetsParam.scala | 176 + .../param/ScoredTweetsParamConfig.scala | 59 + .../query_feature_hydrator/BUILD.bazel | 18 + .../FrsSeedUsersQueryFeatureHydrator.scala | 64 + .../query_transformer/BUILD.bazel | 24 + .../TimelineRankerFrsQueryTransformer.scala | 43 + ...elineRankerInNetworkQueryTransformer.scala | 42 + .../TimelineRankerQueryTransformer.scala | 109 + .../TimelineRankerUtegQueryTransformer.scala | 59 + .../response_transformer/BUILD.bazel | 18 + ...oredTweetsResponseFeatureTransformer.scala | 94 + ...etsCrMixerResponseFeatureTransformer.scala | 65 + ...dTweetsFrsResponseFeatureTransformer.scala | 31 + ...sInNetworkResponseFeatureTransformer.scala | 34 + ...TweetsUtegResponseFeatureTransformer.scala | 31 + .../TimelineRankerResponseTransformer.scala | 91 + .../product/scored_tweets/scorer/BUILD.bazel | 22 + .../scorer/DiversityDiscountProvider.scala | 30 + .../scorer/DiversityScorer.scala | 57 + .../HomeNaviModelDataRecordScorer.scala | 233 + .../scorer/WeightedScoresSumScorer.scala | 91 + .../scoring_pipeline/BUILD.bazel | 33 + ...TweetsDiversityScoringPipelineConfig.scala | 25 + ...weetsRescoreOONScoringPipelineConfig.scala | 23 + ...eVerifiedAuthorScoringPipelineConfig.scala | 24 + .../ScoredTweetsScoringPipelineConfig.scala | 174 + ...ightedScoresSumScoringPipelineConfig.scala | 34 + .../scored_tweets/side_effect/BUILD.bazel | 35 + .../CachedScoredTweetsSideEffect.scala | 123 + ...aturesAndCandidateFeaturesSideEffect.scala | 213 + .../twitter/home_mixer/service/BUILD.bazel | 15 + .../service/HomeMixerAccessPolicy.scala | 13 + .../service/HomeMixerAlertConfig.scala | 65 + .../service/ScoredTweetsService.scala | 24 + .../com/twitter/home_mixer/store/BUILD.bazel | 17 + .../store/RealGraphInNetworkScoresStore.scala | 34 + .../home_mixer/store/UserLanguagesStore.scala | 47 + .../com/twitter/home_mixer/util/BUILD.bazel | 22 + .../util/CachedScoredTweetsHelper.scala | 49 + .../home_mixer/util/CandidatesUtil.scala | 105 + .../util/InjectionTransformer.scala | 43 + .../home_mixer/util/LanguageUtil.scala | 93 + .../home_mixer/util/MissingKeyException.scala | 5 + .../util/ObservedKeyValueResultHandler.scala | 43 + .../home_mixer/util/ReplyRetweetUtil.scala | 120 + .../home_mixer/util/TensorFlowUtil.scala | 32 + .../util/TweetImpressionsHelper.scala | 15 + .../home_mixer/util/earlybird/BUILD.bazel | 18 + .../util/earlybird/EarlybirdRequestUtil.scala | 58 + .../earlybird/EarlybirdResponseUtil.scala | 369 ++ .../util/earlybird/RelevanceSearchUtil.scala | 71 + .../home_mixer/util/tweetypie/BUILD.bazel | 10 + .../util/tweetypie/RequestFields.scala | 57 + .../util/tweetypie/content/BUILD.bazel | 19 + .../content/FeatureExtractionHelper.scala | 29 + .../content/TweetMediaFeaturesExtractor.scala | 285 ++ .../content/TweetTextFeaturesExtractor.scala | 44 + navi/dr_transform/Cargo.toml | 29 + navi/dr_transform/src/all_config.rs | 49 + navi/dr_transform/src/converter.rs | 621 +++ navi/dr_transform/src/lib.rs | 5 + navi/dr_transform/src/util.rs | 30 + navi/navi/Cargo.toml | 76 + navi/navi/README.md | 34 + navi/navi/build.rs | 13 + .../proto/kfserving/grpc_predict_v2.proto | 326 ++ .../tensorflow/core/example/example.proto | 306 ++ .../tensorflow/core/example/feature.proto | 110 + .../framework/allocation_description.proto | 29 + .../tensorflow/core/framework/api_def.proto | 138 + .../core/framework/attr_value.proto | 64 + .../core/framework/cost_graph.proto | 89 + .../core/framework/dataset_metadata.proto | 10 + .../core/framework/dataset_options.proto | 196 + .../core/framework/device_attributes.proto | 58 + .../tensorflow/core/framework/full_type.proto | 276 ++ .../tensorflow/core/framework/function.proto | 136 + .../tensorflow/core/framework/graph.proto | 56 + .../core/framework/graph_transfer_info.proto | 71 + .../core/framework/kernel_def.proto | 48 + .../core/framework/log_memory.proto | 95 + .../tensorflow/core/framework/model.proto | 134 + .../tensorflow/core/framework/node_def.proto | 95 + .../tensorflow/core/framework/op_def.proto | 193 + .../core/framework/reader_base.proto | 18 + .../core/framework/resource_handle.proto | 45 + .../core/framework/step_stats.proto | 88 + .../tensorflow/core/framework/summary.proto | 149 + .../tensorflow/core/framework/tensor.proto | 96 + .../core/framework/tensor_description.proto | 24 + .../core/framework/tensor_shape.proto | 46 + .../core/framework/tensor_slice.proto | 39 + .../tensorflow/core/framework/types.proto | 77 + .../tensorflow/core/framework/variable.proto | 84 + .../tensorflow/core/framework/versions.proto | 33 + .../tensorflow/core/protobuf/autotuning.proto | 106 + .../core/protobuf/bfc_memory_map.proto | 47 + .../tensorflow/core/protobuf/cluster.proto | 84 + .../protobuf/composite_tensor_variant.proto | 16 + .../tensorflow/core/protobuf/config.proto | 902 ++++ .../core/protobuf/control_flow.proto | 91 + .../core/protobuf/conv_autotuning.proto | 33 + .../core/protobuf/coordination_config.proto | 32 + .../core/protobuf/coordination_service.proto | 256 ++ .../core/protobuf/critical_section.proto | 24 + .../core/protobuf/data_service.proto | 88 + .../tensorflow/core/protobuf/debug.proto | 94 + .../core/protobuf/debug_event.proto | 300 ++ .../core/protobuf/device_filters.proto | 73 + .../core/protobuf/device_properties.proto | 58 + .../distributed_runtime_payloads.proto | 24 + .../core/protobuf/eager_service.proto | 344 ++ .../core/protobuf/error_codes.proto | 152 + .../core/protobuf/graph_debug_info.proto | 52 + .../tensorflow/core/protobuf/master.proto | 353 ++ .../core/protobuf/master_service.proto | 121 + .../tensorflow/core/protobuf/meta_graph.proto | 342 ++ .../core/protobuf/named_tensor.proto | 25 + .../core/protobuf/queue_runner.proto | 30 + .../core/protobuf/remote_tensor_handle.proto | 34 + .../tensorflow/core/protobuf/replay_log.proto | 47 + .../core/protobuf/rewriter_config.proto | 223 + .../core/protobuf/saved_model.proto | 23 + .../core/protobuf/saved_object_graph.proto | 251 ++ .../tensorflow/core/protobuf/saver.proto | 48 + .../core/protobuf/service_config.proto | 82 + .../tensorflow/core/protobuf/snapshot.proto | 47 + .../tensorflow/core/protobuf/status.proto | 10 + .../tensorflow/core/protobuf/struct.proto | 160 + .../core/protobuf/tensor_bundle.proto | 66 + .../core/protobuf/tensorflow_server.proto | 61 + .../protobuf/trackable_object_graph.proto | 80 + .../core/protobuf/transport_options.proto | 10 + .../core/protobuf/verifier_config.proto | 27 + .../tensorflow/core/protobuf/worker.proto | 611 +++ .../core/protobuf/worker_service.proto | 90 + .../apis/classification.proto | 48 + .../apis/get_model_metadata.proto | 30 + .../apis/get_model_status.proto | 68 + .../tensorflow_serving/apis/inference.proto | 59 + .../proto/tensorflow_serving/apis/input.proto | 82 + .../tensorflow_serving/apis/logging.proto | 17 + .../proto/tensorflow_serving/apis/model.proto | 33 + .../apis/model_management.proto | 16 + .../apis/model_service.proto | 24 + .../tensorflow_serving/apis/predict.proto | 40 + .../apis/prediction_log.proto | 49 + .../apis/prediction_service.proto | 31 + .../tensorflow_serving/apis/regression.proto | 37 + .../apis/session_service.proto | 56 + .../tensorflow_serving/apis/status.proto | 17 + .../file_system_storage_path_source.proto | 88 + .../config/log_collector_config.proto | 12 + .../config/logging_config.proto | 18 + .../config/model_server_config.proto | 85 + navi/navi/scripts/run_onnx.sh | 10 + navi/navi/scripts/run_tf2.sh | 6 + navi/navi/src/batch.rs | 203 + navi/navi/src/bin/navi.rs | 47 + navi/navi/src/bin/navi_onnx.rs | 11 + navi/navi/src/bin/navi_torch.rs | 19 + navi/navi/src/bootstrap.rs | 299 ++ navi/navi/src/cli_args.rs | 236 + navi/navi/src/cores/validator.rs | 22 + navi/navi/src/lib.rs | 214 + navi/navi/src/metrics.rs | 290 ++ navi/navi/src/onnx_model.rs | 267 ++ navi/navi/src/predict_service.rs | 313 ++ navi/navi/src/tf_model.rs | 492 +++ navi/navi/src/torch_model.rs | 183 + navi/segdense/Cargo.toml | 11 + navi/segdense/src/error.rs | 43 + navi/segdense/src/lib.rs | 4 + navi/segdense/src/main.rs | 23 + navi/segdense/src/mapper.rs | 45 + ...segdense_transform_spec_home_recap_2022.rs | 183 + navi/segdense/src/util.rs | 159 + navi/thrift_bpr_adapter/thrift/Cargo.toml | 8 + navi/thrift_bpr_adapter/thrift/src/data.rs | 1213 ++++++ navi/thrift_bpr_adapter/thrift/src/decoder.rs | 78 + navi/thrift_bpr_adapter/thrift/src/lib.rs | 4 + navi/thrift_bpr_adapter/thrift/src/main.rs | 81 + .../thrift/src/prediction_service.rs | 1067 +++++ navi/thrift_bpr_adapter/thrift/src/tensor.rs | 1146 +++++ product-mixer/README.md | 41 + ...tRecommendationsMixerCandidateSource.scala | 57 + .../account_recommendations_mixer/BUILD | 22 + .../ads/AdsProdStratoCandidateSource.scala | 29 + .../ads/AdsProdThriftCandidateSource.scala | 22 + .../ads/AdsStagingCandidateSource.scala | 28 + .../candidate_source/ads/BUILD | 18 + .../ann/AnnCandidateSource.scala | 43 + .../candidate_source/ann/AnnIdQuery.scala | 18 + .../candidate_source/ann/BUILD.bazel | 17 + .../candidate_source/audiospace/BUILD.bazel | 14 + .../CreatedSpacesCandidateSource.scala | 49 + .../candidate_source/business_profiles/BUILD | 13 + .../TeamMembersCandidateSource.scala | 53 + .../candidate_source/cr_mixer/BUILD.bazel | 12 + ...dTweetRecommendationsCandidateSource.scala | 25 + ...rTweetRecommendationsCandidateSource.scala | 21 + .../candidate_source/earlybird/BUILD.bazel | 12 + .../EarlybirdTweetCandidateSource.scala | 26 + .../explore_ranker/BUILD.bazel | 12 + .../ExploreRankerCandidateSource.scala | 31 + .../flexible_injection_pipeline/BUILD | 17 + .../PromptCandidateSource.scala | 50 + .../candidate_source/hermit/BUILD.bazel | 16 + .../UsersSimilarToMeCandidateSource.scala | 32 + .../candidate_source/interest_discovery/BUILD | 18 + .../RelatedTopicsCandidateSource.scala | 34 + .../candidate_source/lists/BUILD.bazel | 14 + .../OrganicPopGeoListsCandidateSource.scala | 39 + .../candidate_source/people_discovery/BUILD | 23 + .../PeopleDiscoveryCandidateSource.scala | 71 + .../recommendations/BUILD.bazel | 21 + ...FollowRecommendationsCandidateSource.scala | 41 + .../candidate_source/social_graph/BUILD.bazel | 22 + .../SocialgraphCandidateSource.scala | 57 + .../SocialgraphCursorConstants.scala | 7 + .../timeline_ranker/BUILD.bazel | 13 + ...melineRankerInNetworkCandidateSource.scala | 51 + .../TimelineRankerRecapCandidateSource.scala | 28 + .../TimelineRankerUtegCandidateSource.scala | 49 + .../timeline_scorer/BUILD.bazel | 16 + .../TimelineScorerCandidateSource.scala | 156 + .../candidate_source/timeline_service/BUILD | 18 + .../TimelineServiceTweetCandidateSource.scala | 48 + .../timelines_impression_store/BUILD | 16 + ...inesImpressionStoreCandidateSourceV2.scala | 30 + .../candidate_source/topics/BUILD | 11 + .../FollowedTopicsCandidateSource.scala | 21 + .../tweetconvosvc/BUILD.bazel | 26 + .../ConversationServiceCandidateSource.scala | 173 + ...ionServiceResponseFeatureTransformer.scala | 49 + ...pMaxConversationModuleItemCandidates.scala | 55 + .../component_library/decorator/slice/BUILD | 23 + .../slice/SliceItemCandidateDecorator.scala | 41 + .../decorator/slice/builder/BUILD | 17 + .../CursorCandidateSliceItemBuilder.scala | 29 + .../component_library/decorator/urt/BUILD | 108 + ...rtConversationItemCandidateDecorator.scala | 44 + .../urt/UrtItemCandidateDecorator.scala | 40 + .../urt/UrtItemInModuleDecorator.scala | 52 + .../urt/UrtMultipleModulesDecorator.scala | 108 + .../ContextualTweetRefBuilder.scala | 12 + .../ConversationModuleMetadataBuilder.scala | 38 + .../FlipPromptCandidateUrtItemBuilder.scala | 205 + .../FlipPromptModuleGrouping.scala | 23 + .../FlipPromptUrtModuleBuilder.scala | 54 + .../OnboardingInjectionConversions.scala | 361 ++ .../RelevancePromptConversions.scala | 76 + .../TilesCarouselConversions.scala | 154 + .../urt/builder/icon/HorizonIconBuilder.scala | 17 + .../item/ad/AdsCandidateUrtItemBuilder.scala | 274 ++ .../item/alert/DurationParamBuilder.scala | 20 + .../ShowAlertCandidateUrtItemBuilder.scala | 61 + ...icShowAlertColorConfigurationBuilder.scala | 18 + ...taticShowAlertDisplayLocationBuilder.scala | 18 + ...taticShowAlertIconDisplayInfoBuilder.scala | 18 + .../ArticleCandidateUrtItemBuilder.scala | 52 + .../AudioSpaceCandidateUrtItemBuilder.scala | 39 + .../card/CardCandidateUtrItemBuilder.scala | 51 + ...mmerceProductCandidateUrtItemBuilder.scala | 41 + ...eProductGroupCandidateUrtItemBuilder.scala | 42 + .../EventCandidateUrtItemBuilder.scala | 51 + .../GenericSummaryActionBuilder.scala | 32 + ...enericSummaryCandidateUrtItemBuilder.scala | 64 + .../GenericSummaryContextBuilder.scala | 22 + .../IconLabelCandidateUrtItemBuilder.scala | 42 + ...tCandidateUrtItemStringCenterBuilder.scala | 59 + ...tCandidateUrtItemStringCenterBuilder.scala | 74 + .../message/MessageTextActionBuilder.scala | 36 + .../item/message/UserFacePileBuilder.scala | 24 + ...entAnnotationCandidateUrtItemBuilder.scala | 46 + ...tCandidateUrtItemStringCenterBuilder.scala | 62 + ...ingSuggestionCandidateUrtItemBuilder.scala | 41 + .../tile/TileCandidateUrtItemBuilder.scala | 48 + .../topic/ParamTopicDisplayTypeBuilder.scala | 41 + .../ParamTopicFunctionalityTypeBuilder.scala | 38 + .../topic/StaticTopicDisplayTypeBuilder.scala | 18 + .../StaticTopicFunctionalityTypeBuilder.scala | 18 + .../topic/TopicCandidateUrtItemBuilder.scala | 47 + ...icalGridTopicCandidateUrtItemBuilder.scala | 46 + .../trend/TrendCandidateUrtItemBuilder.scala | 63 + .../trend/TrendMetaDescriptionBuilder.scala | 38 + .../trend/TrendPromotedMetadataBuilder.scala | 41 + .../tweet/TweetCandidateUrtItemBuilder.scala | 92 + .../TwitterListCandidateUrtItemBuilder.scala | 41 + ...iedTrendEventCandidateUrtItemBuilder.scala | 36 + .../user/UserCandidateUrtItemBuilder.scala | 62 + .../metadata/ClientEventInfoBuilder.scala | 48 + ...sationTweetClientEventDetailsBuilder.scala | 63 + .../builder/metadata/StaticUrlBuilder.scala | 18 + .../TopicClientEventDetailsBuilder.scala | 56 + ...tInterestedFeedbackActionInfoBuilder.scala | 45 + .../TopicTweetClientEventDetailsBuilder.scala | 66 + .../TopicsToFollowModuleMetadataBuilder.scala | 39 + ...WhoToFollowFeedbackActionInfoBuilder.scala | 60 + .../CursorCandidateUrtOperationBuilder.scala | 29 + .../FeaturePromotedMetadataBuilder.scala | 106 + .../builder/richtext/RichTextBuilder.scala | 28 + .../builder/richtext/RichTextMarkupUtil.scala | 135 + .../RichTextReferenceObjectBuilder.scala | 8 + .../richtext/RichTextRtlOptionBuilder.scala | 12 + .../richtext/StaticRichTextBuilder.scala | 17 + .../TwitterTextEntityProcessor.scala | 51 + .../TwitterTextFormatProcessor.scala | 67 + .../twitter_text/TwitterTextRenderer.scala | 390 ++ .../TwitterTextRendererProcessor.scala | 5 + .../TwitterTextRichTextBuilder.scala | 37 + .../FeatureSocialContextBuilder.scala | 101 + .../GeneralModuleSocialContextBuilder.scala | 38 + .../GeneralSocialContextBuilder.scala | 32 + .../WhoToFollowSocialContextBuilder.scala | 48 + .../urt/builder/stringcenter/ModuleStr.scala | 31 + .../urt/builder/stringcenter/Str.scala | 36 + .../FeatureModuleDisplayTypeBuilder.scala | 22 + ...ShowMoreBehaviorRevealByCountBuilder.scala | 22 + .../timeline_module/ModuleFooterBuilder.scala | 27 + .../timeline_module/ModuleHeaderBuilder.scala | 41 + .../ModuleHeaderDisplayTypeBuilder.scala | 22 + .../timeline_module/ModuleIdGeneration.scala | 51 + ...ShowMoreBehaviorRevealByCountBuilder.scala | 25 + .../ParamGatedModuleFooterBuilder.scala | 26 + .../ParamGatedModuleHeaderBuilder.scala | 26 + ...mWhoToFollowModuleDisplayTypeBuilder.scala | 53 + .../StaticModuleDisplayTypeBuilder.scala | 16 + .../TimelineModuleBuilder.scala | 56 + .../experiments/metrics/BUILD | 11 + .../metrics/MetricDefinitions.scala | 116 + .../experiments/metrics/MetricGroup.scala | 54 + .../metrics/MetricTemplateCLIRunner.scala | 101 + .../experiments/metrics/MetricTemplates.scala | 123 + .../metrics/PlaceholderConfig.scala | 37 + .../feature/featurestorev1/BUILD | 14 + .../FeatureStoreV1QueryUserIdFeature.scala | 46 + ...yUserIdTweetCandidateAuthorIdFeature.scala | 68 + ...ryUserIdTweetCandidateTweetIdFeature.scala | 66 + ...StoreV1TweetCandidateAuthorIdFeature.scala | 60 + ...eStoreV1TweetCandidateTweetIdFeature.scala | 58 + ...ureStoreV1UserCandidateUserIdFeature.scala | 40 + .../component_library/feature_hydrator/BUILD | 35 + ...erBrandSafetySettingsFeatureHydrator.scala | 52 + .../feature_hydrator/candidate/ads/BUILD | 21 + .../candidate/decay/BUILD.bazel | 16 + .../decay/DecayCandidateFeatureHydrator.scala | 65 + .../candidate/param_gated/BUILD | 25 + ...ramGatedBulkCandidateFeatureHydrator.scala | 51 + .../ParamGatedCandidateFeatureHydrator.scala | 51 + .../param_gated/featurestorev1/BUILD | 27 + ...atureStoreV1CandidateFeatureHydrator.scala | 58 + .../candidate/qualityfactor_gated/BUILD.bazel | 14 + ...yFactorGatedCandidateFeatureHydrator.scala | 59 + .../candidate/tweet_is_nsfw/BUILD | 37 + .../TweetIsNsfwCandidateFeatureHydrator.scala | 109 + .../candidate/tweet_tlx/BUILD.bazel | 42 + ...weetTLXScoreCandidateFeatureHydrator.scala | 59 + .../candidate/tweet_tweetypie/BUILD | 29 + ...eetTweetypieCandidateFeatureHydrator.scala | 241 ++ .../candidate/tweet_visibility_reason/BUILD | 31 + ...tyReasonBulkCandidateFeatureHydrator.scala | 98 + .../async/AsyncQueryFeatureHydrator.scala | 97 + .../feature_hydrator/query/async/BUILD | 23 + .../query/cr_ml_ranker/BUILD.bazel | 19 + ...CrMlRankerCommonQueryFeatureHydrator.scala | 37 + ...kerCommonQueryFeatureHydratorBuilder.scala | 18 + .../cr_ml_ranker/RankingConfigBuilder.scala | 11 + .../query/impressed_tweets/BUILD | 23 + .../ImpressedTweetsQueryFeatureHydrator.scala | 57 + .../query/logged_in_only/BUILD | 25 + .../LoggedInOnlyQueryFeatureHydrator.scala | 31 + .../AsyncParamGatedQueryFeatureHydrator.scala | 48 + .../feature_hydrator/query/param_gated/BUILD | 25 + .../ParamGatedQueryFeatureHydrator.scala | 39 + ...edFeatureStoreV1QueryFeatureHydrator.scala | 53 + .../query/param_gated/featurestorev1/BUILD | 27 + ...edFeatureStoreV1QueryFeatureHydrator.scala | 47 + .../query/qualityfactor_gated/BUILD.bazel | 14 + ...alityFactorGatedQueryFeatureHydrator.scala | 54 + ...daptiveLongIntBloomFilterDedupFilter.scala | 40 + .../component_library/filter/BUILD | 30 + .../filter/ExcludedIdsFilter.scala | 28 + .../filter/FeatureFilter.scala | 63 + .../FeatureValueConditionalFilter.scala | 64 + .../filter/HasAuthorIdFeatureFilter.scala | 27 + .../filter/ParamGatedFilter.scala | 41 + .../filter/PredicateFilter.scala | 63 + .../filter/SnowflakeIdAgeFilter.scala | 42 + .../filter/TweetAuthorCountryFilter.scala | 47 + .../filter/TweetAuthorIsSelfFilter.scala | 37 + .../filter/TweetIsNotReplyFilter.scala | 36 + .../filter/TweetLanguageFilter.scala | 40 + .../filter/TweetVisibilityFilter.scala | 71 + .../UrtUnorderedExcludeIdsCursorFilter.scala | 32 + .../filter/list_visibility/BUILD.bazel | 17 + .../ListVisibilityFilter.scala | 52 + .../filter/tweet_impression/BUILD.bazel | 14 + .../TweetImpressionFilter.scala | 37 + .../component_library/gate/BUILD | 16 + .../gate/DefinedCountryCodeGate.scala | 13 + .../component_library/gate/FeatureGate.scala | 83 + .../gate/FirstPageGate.scala | 19 + .../gate/NoCandidatesGate.scala | 21 + .../gate/NonEmptyAdsQueryStringGate.scala | 16 + .../gate/NonEmptyCandidatesGate.scala | 22 + .../gate/QualityFactorGate.scala | 25 + .../AnyCandidatesWithoutFeatureGate.scala | 34 + .../BUILD.bazel | 13 + .../model/candidate/ArticleCandidate.scala | 77 + .../model/candidate/AudioSpaceCandidate.scala | 75 + .../component_library/model/candidate/BUILD | 13 + .../model/candidate/CardCandidate.scala | 75 + .../candidate/CommerceItemCandidate.scala | 160 + .../model/candidate/CursorCandidate.scala | 87 + .../model/candidate/DMConvoCandidate.scala | 160 + .../model/candidate/DMEventCandidate.scala | 152 + .../candidate/GenericSummaryCandidate.scala | 75 + .../model/candidate/LabelCandidate.scala | 72 + .../candidate/MomentAnnotationCandidate.scala | 86 + .../model/candidate/PromptCandidate.scala | 433 ++ .../model/candidate/ShowAlertCandidate.scala | 80 + .../model/candidate/TopicCandidate.scala | 159 + .../model/candidate/TweetCandidate.scala | 93 + .../candidate/TwitterListCandidate.scala | 74 + .../model/candidate/UserCandidate.scala | 87 + .../model/candidate/ads/AdsCandidate.scala | 95 + .../model/candidate/ads/BUILD | 22 + .../hubble/AdCreativeCandidate.scala | 91 + .../candidate/hubble/AdGroupCandidate.scala | 86 + .../candidate/hubble/AdUnitCandidate.scala | 86 + .../model/candidate/hubble/BUILD | 15 + .../candidate/hubble/CampaignCandidate.scala | 76 + .../hubble/FundingSourceCandidate.scala | 83 + .../model/candidate/suggestion/BUILD | 17 + .../suggestion/QuerySuggestionCandidate.scala | 296 ++ .../SpellingSuggestionCandidate.scala | 93 + .../model/candidate/trends_events/BUILD | 17 + .../UnifiedTrendEventCandidate.scala | 119 + .../component_library/model/cursor/BUILD | 19 + .../model/cursor/OrderedCursor.scala | 30 + .../model/cursor/PassThroughCursor.scala | 35 + .../cursor/UnorderedBloomFilterCursor.scala | 25 + .../cursor/UnorderedExcludeIdsCursor.scala | 30 + .../model/cursor/UrtPlaceholderCursor.scala | 17 + .../flexible_injection_pipeline/BUILD.bazel | 22 + .../model/presentation/slice/BUILD | 15 + .../slice/SliceItemPresentation.scala | 7 + .../model/presentation/urt/BUILD | 13 + .../urt/ConversationModuleItem.scala | 10 + .../urt/UrtItemPresentation.scala | 10 + .../urt/UrtModulePresentation.scala | 8 + .../urt/UrtOperationPresentation.scala | 8 + .../model/query/ads/AdsQuery.scala | 59 + .../component_library/model/query/ads/BUILD | 17 + .../AccountRecommendationsMixerModule.scala | 60 + .../component_library/module/BUILD | 72 + .../module/ConversationServiceModule.scala | 42 + .../module/CrMixerClientModule.scala | 33 + .../module/DarkTrafficFilterModule.scala | 27 + .../module/EarlybirdModule.scala | 60 + .../module/ExploreRankerClientModule.scala | 38 + .../FollowRecommenderServiceModule.scala | 34 + .../module/GizmoduckClientModule.scala | 47 + .../module/HomeScorerClientModule.scala | 33 + .../InterestsDiscoveryServiceModule.scala | 34 + .../module/OnboardingTaskServiceModule.scala | 30 + .../module/PeopleDiscoveryServiceModule.scala | 42 + .../module/SocialGraphServiceModule.scala | 36 + .../module/TimelineMixerClientModule.scala | 33 + .../module/TimelineRankerClientModule.scala | 47 + .../module/TimelineScorerClientModule.scala | 33 + .../module/TimelineServiceClientModule.scala | 44 + .../module/TweetImpressionStoreModule.scala | 66 + .../module/TweetyPieClientModule.scala | 58 + .../module/UserSessionStoreModule.scala | 74 + .../module/cr_ml_ranker/BUILD.bazel | 13 + .../cr_ml_ranker/CrMlRankerModule.scala | 39 + .../component_library/module/http/BUILD | 30 + .../module/http/FinagleHttpClientModule.scala | 69 + ...eHttpClientWithCredentialProxyModule.scala | 75 + .../FinagleHttpClientWithProxyModule.scala | 81 + .../module/http/FinatraHttpClientModule.scala | 79 + ...aHttpClientWithCredentialProxyModule.scala | 93 + .../FinatraHttpClientWithProxyModule.scala | 89 + .../module/http/ProxyCredentialsModule.scala | 27 + .../ads/AdsCandidatePipelineConfig.scala | 56 + .../AdsCandidatePipelineConfigBuilder.scala | 56 + ...AdsCandidatePipelineQueryTransformer.scala | 79 + ...sCandidatePipelineResultsTransformer.scala | 39 + .../AdsDependentCandidatePipelineConfig.scala | 60 + ...endentCandidatePipelineConfigBuilder.scala | 58 + ...entCandidatePipelineQueryTransformer.scala | 33 + .../ads/AdsDisplayLocationBuilder.scala | 16 + .../pipeline/candidate/ads/BUILD | 22 + .../candidate/ads/CountNumOrganicItems.scala | 54 + .../candidate/ads/GetOrganicItemIds.scala | 29 + .../ads/PromotedTweetsOnlyFilter.scala | 39 + .../ads/ValidAdImpressionIdFilter.scala | 24 + .../flexible_injection_pipeline/BUILD.bazel | 23 + .../FlipPromptCandidatePipelineConfig.scala | 67 + ...PromptCandidatePipelineConfigBuilder.scala | 35 + ...omptDependentCandidatePipelineConfig.scala | 69 + ...endentCandidatePipelineConfigBuilder.scala | 35 + .../transformer/BUILD | 22 + .../FlipCandidateFeatureTransformer.scala | 37 + .../transformer/FlipInjectionParams.scala | 11 + .../transformer/FlipQueryTransformer.scala | 62 + .../PromptResultsTransformer.scala | 54 + .../who_to_follow_module/BUILD.bazel | 20 + .../WhoToFollowArmCandidateDecorator.scala | 82 + ...hoToFollowArmCandidatePipelineConfig.scala | 10 + ...ArmCandidatePipelineQueryTransformer.scala | 72 + ...wArmDependentCandidatePipelineConfig.scala | 76 + ...endentCandidatePipelineConfigBuilder.scala | 66 + ...oFollowArmResponseFeatureTransformer.scala | 38 + .../WhoToFollowCandidateDecorator.scala | 89 + .../WhoToFollowCandidatePipelineConfig.scala | 77 + ...FollowCandidatePipelineConfigBuilder.scala | 69 + ...lowCandidatePipelineQueryTransformer.scala | 39 + ...WhoToFollowClientEventDetailsBuilder.scala | 67 + ...llowDependentCandidatePipelineConfig.scala | 75 + ...endentCandidatePipelineConfigBuilder.scala | 70 + ...hoToFollowResponseFeatureTransformer.scala | 39 + .../premarshaller/cursor/BUILD | 25 + .../cursor/CursorSerializer.scala | 149 + .../cursor/UrtCursorSerializer.scala | 161 + .../premarshaller/slice/BUILD | 25 + .../slice/SliceDomainMarshaller.scala | 96 + .../premarshaller/slice/builder/BUILD | 19 + .../builder/OrderedNextCursorBuilder.scala | 38 + .../builder/OrderedNextCursorUpdater.scala | 28 + .../OrderedPreviousCursorBuilder.scala | 40 + .../OrderedPreviousCursorUpdater.scala | 30 + .../slice/builder/ShouldInclude.scala | 16 + .../slice/builder/SliceBuilder.scala | 62 + .../slice/builder/SliceCursorBuilder.scala | 24 + .../slice/builder/SliceCursorUpdater.scala | 58 + .../component_library/premarshaller/urp/BUILD | 21 + .../urp/UrpDomainMarshaller.scala | 52 + .../premarshaller/urp/builder/BUILD | 16 + .../urp/builder/PageBodyBuilder.scala | 18 + .../urp/builder/PageHeaderBuilder.scala | 18 + .../urp/builder/PageNavBarBuilder.scala | 18 + .../StaticTimelineScribeConfigBuilder.scala | 19 + .../builder/TimelineScribeConfigBuilder.scala | 22 + .../component_library/premarshaller/urt/BUILD | 27 + .../urt/UndecoratedUrtDomainMarshaller.scala | 148 + .../urt/UrtDomainMarshaller.scala | 112 + .../AddEntriesInstructionBuilder.scala | 19 + ...iesWithAddToModuleInstructionBuilder.scala | 33 + ...thPinnedAndReplaceInstructionBuilder.scala | 31 + ...eplaceAndShowAlertInstructionBuilder.scala | 25 + ...EntriesWithReplaceInstructionBuilder.scala | 31 + ...triesWithShowCoverInstructionBuilder.scala | 29 + .../AddToModuleInstructionBuilder.scala | 37 + .../premarshaller/urt/builder/BUILD | 29 + ...orderedExcludeIdsBottomCursorBuilder.scala | 49 + .../ClearCacheInstructionBuilder.scala | 16 + .../FeaturePassThroughCursorBuilder.scala | 33 + .../urt/builder/IncludeInstruction.scala | 31 + .../MarkUnreadInstructionBuilder.scala | 32 + .../builder/OrderedBottomCursorBuilder.scala | 45 + .../urt/builder/OrderedGapCursorBuilder.scala | 54 + .../urt/builder/OrderedTopCursorBuilder.scala | 52 + .../builder/PinEntryInstructionBuilder.scala | 23 + .../builder/PlaceholderTopCursorBuilder.scala | 34 + .../ReplaceEntryInstructionBuilder.scala | 63 + .../builder/ShowAlertInstructionBuilder.scala | 23 + .../builder/ShowCoverInstructionBuilder.scala | 24 + .../StaticTimelineScribeConfigBuilder.scala | 15 + .../builder/TerminateInstructionBuilder.scala | 44 + .../builder/TimelineScribeConfigBuilder.scala | 18 + ...rderedBloomFilterBottomCursorBuilder.scala | 43 + ...orderedExcludeIdsBottomCursorBuilder.scala | 26 + ...eredExcludeIdsSeqBottomCursorBuilder.scala | 30 + .../urt/builder/UrtBuilder.scala | 94 + .../urt/builder/UrtCursorBuilder.scala | 134 + .../urt/builder/UrtCursorUpdater.scala | 44 + .../urt/builder/UrtInstructionBuilder.scala | 15 + .../urt/builder/UrtMetadataBuilder.scala | 43 + .../component_library/scorer/common/BUILD | 30 + .../common/MLModelInferenceClient.scala | 12 + .../scorer/common/ManagedModelClient.scala | 33 + .../scorer/common/ModelSelector.scala | 28 + .../scorer/common/NaviModelClient.scala | 50 + .../component_library/scorer/cortex/BUILD | 35 + ...agedInferenceServiceDataRecordScorer.scala | 137 + ...erenceServiceDataRecordScorerBuilder.scala | 67 + ...xManagedInferenceServiceTensorScorer.scala | 97 + ...dInferenceServiceTensorScorerBuilder.scala | 47 + .../scorer/cortex/ModelFeatureExtractor.scala | 15 + .../scorer/cr_ml_ranker/BUILD.bazel | 18 + .../cr_ml_ranker/CrMlRankerScorer.scala | 52 + .../cr_ml_ranker/CrMlRankerStitchClient.scala | 79 + .../component_library/scorer/deepbird/BUILD | 42 + .../deepbird/BaseDeepbirdV2Scorer.scala | 91 + .../DeepbirdV2PredictionServerScorer.scala | 55 + .../LollyPredictionEngineScorer.scala | 61 + .../TensorflowPredictionEngineScorer.scala | 58 + .../scorer/param_gated/BUILD | 16 + .../scorer/param_gated/ParamGatedScorer.scala | 43 + .../scorer/qualityfactor_gated/BUILD | 16 + .../QualityFactorGatedScorer.scala | 59 + .../scorer/tensorbuilder/BUILD | 21 + .../BooleanInferInputTensorBuilder.scala | 13 + .../BytesInferInputTensorBuilder.scala | 13 + .../CandidateInferInputTensorBuilder.scala | 70 + .../Float32InferInputTensorBuilder.scala | 27 + .../FloatTensorInferInputTensorBuilder.scala | 33 + .../InferInputTensorBuilder.scala | 151 + .../Int64InferInputTensorBuilder.scala | 26 + .../ModelInferRequestBuilder.scala | 40 + .../QueryInferInputTensorBuilder.scala | 61 + .../SparseMapInferInputTensorBuilder.scala | 61 + .../scorer/tweet_tlx/BUILD.bazel | 26 + .../tweet_tlx/TweetTLXStratoScorer.scala | 58 + .../tweet_tlx/TweetTLXThriftScorer.scala | 79 + .../component_library/selector/BUILD | 47 + .../component_library/selector/Bucketer.scala | 20 + .../selector/CandidateMergeStrategy.scala | 82 + .../selector/CandidatePositionInResults.scala | 33 + .../selector/DeduplicationKey.scala | 33 + .../selector/DropAllCandidates.scala | 29 + .../selector/DropDuplicateCandidates.scala | 45 + .../DropDuplicateModuleItemCandidates.scala | 88 + .../selector/DropDuplicateResults.scala | 46 + .../selector/DropFilteredCandidates.scala | 48 + .../DropFilteredModuleItemCandidates.scala | 50 + .../selector/DropMaxCandidates.scala | 84 + .../DropMaxModuleItemCandidates.scala | 54 + .../selector/DropMaxResults.scala | 40 + .../DropModuleTooFewModuleItemResults.scala | 46 + .../selector/DropNonDuplicateCandidates.scala | 75 + .../selector/DropOrthogonalCandidates.scala | 54 + ...DropRequestedMaxModuleItemCandidates.scala | 68 + .../selector/DropRequestedMaxResults.scala | 55 + .../selector/DropSelector.scala | 111 + .../selector/DropTooFewResults.scala | 35 + .../selector/DynamicPositionSelector.scala | 124 + .../InsertAppendIntoModuleCandidates.scala | 56 + .../selector/InsertAppendPatternResults.scala | 105 + .../selector/InsertAppendRatioResults.scala | 171 + .../selector/InsertAppendResults.scala | 53 + .../selector/InsertAppendWeaveResults.scala | 120 + .../InsertAppendWithoutFeatureResults.scala | 35 + .../InsertDynamicPositionResults.scala | 69 + ...ertFixedPositionIntoModuleCandidates.scala | 69 + .../selector/InsertFixedPositionResults.scala | 46 + .../selector/InsertIntoModule.scala | 70 + ...rtPerCandidateDynamicPositionResults.scala | 78 + .../InsertRandomPositionResults.scala | 139 + .../InsertRelativePositionResults.scala | 58 + .../selector/InsertSelector.scala | 39 + .../selector/SelectConditionally.scala | 85 + .../SelectFromSubpoolCandidates.scala | 147 + .../selector/UpdateSortCandidates.scala | 86 + .../UpdateSortModuleItemCandidates.scala | 96 + .../selector/UpdateSortResults.scala | 43 + .../selector/ads/AdsInjector.scala | 73 + .../selector/ads/BUILD.bazel | 30 + .../selector/ads/InsertAdResults.scala | 95 + .../component_library/selector/sorter/BUILD | 20 + .../selector/sorter/FeatureValueSorter.scala | 248 ++ .../selector/sorter/RandomShuffleSorter.scala | 16 + .../selector/sorter/ReverseSorter.scala | 14 + .../selector/sorter/SortOrder.scala | 5 + .../selector/sorter/SorterFromOrdering.scala | 25 + .../selector/sorter/SorterProvider.scala | 40 + .../selector/sorter/featurestorev1/BUILD | 18 + .../FeatureStoreV1FeatureValueSorter.scala | 98 + .../component_library/side_effect/BUILD | 30 + .../KafkaPublishingSideEffect.scala | 233 + .../ParamGatedPipelineResultSideEffect.scala | 76 + .../ScribeClientEventSideEffect.scala | 118 + .../ScribeLogEventAsyncSideEffect.scala | 58 + .../ScribeLogEventSideEffect.scala | 58 + .../side_effect/StratoInsertSideEffect.scala | 70 + .../UserSessionStoreUpdateSideEffect.scala | 35 + .../side_effect/metrics/BUILD.bazel | 18 + .../metrics/CandidateMetricFunction.scala | 59 + .../ScribeClientEventMetricsSideEffect.scala | 74 + ...eClientEventMetricsSideEffectBuilder.scala | 84 + .../core/product/guice/scope/BUILD | 8 + .../product/guice/scope/ProductScoped.java | 14 + .../core/controllers/AlertConfig.scala | 38 + .../product_mixer/core/controllers/BUILD | 39 + .../controllers/DebugTwitterContext.scala | 55 + .../GetComponentRegistryHandler.scala | 114 + .../GetDebugConfigurationHandler.scala | 60 + .../core/controllers/PredicateConfig.scala | 22 + .../controllers/ProductMixerController.scala | 79 + .../QualityFactorMonitoringConfig.scala | 6 + .../twitter/product_mixer/core/feature/BUILD | 10 + .../product_mixer/core/feature/Feature.scala | 74 + .../core/feature/datarecord/BUILD | 21 + .../datarecord/DataRecordCompatible.scala | 316 ++ .../datarecord/DataRecordFeature.scala | 42 + .../core/feature/featuremap/BUILD | 16 + .../core/feature/featuremap/FeatureMap.scala | 195 + .../featuremap/FeatureMapBuilder.scala | 110 + .../featuremap/FeatureMapException.scala | 12 + .../featuremap/FeatureMapSerializer.scala | 63 + .../asyncfeaturemap/AsyncFeatureMap.scala | 134 + .../AsyncFeatureMapSerializer.scala | 45 + .../feature/featuremap/asyncfeaturemap/BUILD | 18 + .../core/feature/featuremap/datarecord/BUILD | 23 + .../datarecord/DataRecordConverter.scala | 51 + .../datarecord/DataRecordExtractor.scala | 60 + .../featuremap/datarecord/FeaturesScope.scala | 157 + .../feature/featuremap/featurestorev1/BUILD | 17 + .../FeatureStoreV1FeatureMap.scala | 191 + .../core/feature/featurestorev1/BUILD | 29 + .../featurestorev1/FeatureStoreV1Entity.scala | 35 + .../FeatureStoreV1Feature.scala | 312 ++ .../feature/featurestorev1/featurevalue/BUILD | 19 + .../featurevalue/FeatureStoreV1Response.scala | 39 + .../functional_component/access_policy/BUILD | 9 + .../candidate_source/BUILD | 21 + .../candidate_source/CandidateSource.scala | 50 + .../CandidatesWithSourceFeatures.scala | 16 + .../PassthroughCandidateSource.scala | 59 + .../StaticCandidateSource.scala | 15 + .../candidate_source/product_pipeline/BUILD | 23 + .../ProductPipelineCandidateSource.scala | 71 + .../candidate_source/strato/BUILD | 21 + .../strato/StratoErrCategorizer.scala | 21 + .../strato/StratoKeyFetcherSeqSource.scala | 27 + .../strato/StratoKeyFetcherSource.scala | 45 + ...toKeyFetcherWithSourceFeaturesSource.scala | 65 + .../strato/StratoKeyView.scala | 4 + .../StratoKeyViewFetcherSeqSource.scala | 30 + .../strato/StratoKeyViewFetcherSource.scala | 51 + ...yViewFetcherWithSourceFeaturesSource.scala | 75 + .../core/functional_component/common/BUILD | 15 + .../common/CandidateScope.scala | 98 + .../common/access_policy/AccessPolicy.scala | 38 + .../access_policy/AccessPolicyEvaluator.scala | 12 + .../common/access_policy/BUILD | 15 + .../WithDebugAccessPolicies.scala | 10 + .../common/alert/Alert.scala | 31 + .../common/alert/AlertType.scala | 25 + .../functional_component/common/alert/BUILD | 21 + .../common/alert/EmptyResponseRateAlert.scala | 27 + .../alert/GenericClientLatencyAlert.scala | 19 + .../alert/GenericClientSuccessRateAlert.scala | 29 + .../alert/GenericClientThroughputAlert.scala | 25 + .../common/alert/IsObservableFromStrato.scala | 10 + .../common/alert/LatencyAlert.scala | 20 + .../common/alert/NotificationGroup.scala | 36 + .../common/alert/Percentile.scala | 17 + .../common/alert/ResponseSizeAlert.scala | 27 + .../common/alert/Source.scala | 47 + .../common/alert/StratoColumnAlert.scala | 34 + .../common/alert/SuccessRateAlert.scala | 27 + .../common/alert/ThroughputAlert.scala | 24 + .../common/alert/predicate/BUILD | 13 + .../alert/predicate/MetricGranularity.scala | 35 + .../common/alert/predicate/Operator.scala | 14 + .../common/alert/predicate/Predicate.scala | 52 + .../alert/predicate/TriggerIfAbove.scala | 15 + .../alert/predicate/TriggerIfBelow.scala | 15 + .../predicate/TriggerIfLatencyAbove.scala | 22 + .../core/functional_component/configapi/BUILD | 25 + .../configapi/ConfigBuilder.scala | 17 + .../configapi/ParamsBuilder.scala | 34 + .../configapi/RequestContext.scala | 19 + .../configapi/RequestContextBuilder.scala | 72 + .../configapi/StaticParam.scala | 6 + .../configapi/registry/BUILD | 23 + .../registry/GlobalParamConfig.scala | 7 + .../registry/GlobalParamRegistry.scala | 21 + .../configapi/registry/ParamConfig.scala | 74 + .../registry/ParamConfigBuilder.scala | 49 + .../core/functional_component/decorator/BUILD | 37 + .../decorator/CandidateDecorator.scala | 61 + .../decorator/Decoration.scala | 11 + .../decorator/slice/builder/BUILD | 21 + .../builder/CandidateSliceItemBuilder.scala | 14 + .../decorator/urt/builder/BUILD | 27 + .../builder/CandidateUrtEntryBuilder.scala | 14 + .../builder/icon/BaseHorizonIconBuilder.scala | 14 + .../item/alert/BaseDurationBuilder.scala | 11 + ...seShowAlertColorConfigurationBuilder.scala | 15 + .../BaseShowAlertDisplayLocationBuilder.scala | 15 + .../BaseShowAlertIconDisplayInfoBuilder.scala | 15 + ...seShowAlertNavigationMetadataBuilder.scala | 15 + .../alert/BaseShowAlertUserIdsBuilder.scala | 10 + .../topic/BaseTopicDisplayTypeBuilder.scala | 15 + .../BaseTopicFunctionalityTypeBuilder.scala | 15 + .../tweet/BaseEntryIdToReplaceBuilder.scala | 14 + .../tweet/BaseTimelinesScoreInfoBuilder.scala | 15 + .../tweet/BaseTweetHighlightsBuilder.scala | 15 + .../BaseUserReactiveTriggersBuilder.scala | 15 + .../BaseClientEventDetailsBuilder.scala | 19 + .../metadata/BaseClientEventInfoBuilder.scala | 20 + .../BaseFeedbackActionInfoBuilder.scala | 15 + .../urt/builder/metadata/BaseModuleStr.scala | 10 + .../urt/builder/metadata/BaseStr.scala | 10 + .../urt/builder/metadata/BaseUrlBuilder.scala | 11 + .../BasePromotedMetadataBuilder.scala | 15 + .../richtext/BaseRichTextBuilder.scala | 11 + .../BaseModuleSocialContextBuilder.scala | 14 + .../BaseSocialContextBuilder.scala | 15 + ...ModuleStringCenterPlaceholderBuilder.scala | 12 + .../BaseStringCenterPlaceholderBuilder.scala | 12 + .../BaseModuleDisplayTypeBuilder.scala | 14 + .../BaseModuleFooterBuilder.scala | 14 + .../BaseModuleHeaderBuilder.scala | 14 + .../BaseModuleHeaderDisplayTypeBuilder.scala | 16 + .../BaseModuleMetadataBuilder.scala | 14 + .../BaseModuleShowMoreBehaviorBuilder.scala | 14 + .../BaseTimelineModuleBuilder.scala | 14 + .../feature_hydrator/BUILD | 21 + .../CandidateFeatureHydrator.scala | 96 + .../feature_hydrator/FeatureHydrator.scala | 9 + .../HydratorCandidateResult.scala | 10 + .../QueryFeatureHydrator.scala | 79 + .../feature_hydrator/featurestorev1/BUILD | 24 + .../FeatureStoreDatasetErrorHandler.scala | 74 + ...atureStoreV1CandidateFeatureHydrator.scala | 97 + .../FeatureStoreV1DynamicClientBuilder.scala | 12 + .../FeatureStoreV1HydrationConfig.scala | 25 + .../FeatureStoreV1QueryFeatureHydrator.scala | 79 + .../core/functional_component/filter/BUILD | 21 + .../functional_component/filter/Filter.scala | 67 + .../filter/FilterResult.scala | 4 + .../core/functional_component/gate/BUILD | 21 + .../core/functional_component/gate/Gate.scala | 127 + .../gate/GateResult.scala | 47 + .../gate/ShouldContinue.scala | 7 + .../functional_component/marshaller/BUILD | 17 + .../marshaller/TransportMarshaller.scala | 42 + .../marshaller/request/BUILD | 17 + .../request/ClientContextMarshaller.scala | 27 + .../request/ClientContextUnmarshaller.scala | 30 + .../request/FeatureValueUnmarshaller.scala | 31 + .../response/graphql/contextual_ref/BUILD | 21 + .../ContextualTweetRefMarshaller.scala | 17 + .../OuterTweetContextMarshaller.scala | 18 + .../TweetHydrationContextMarshaller.scala | 20 + .../response/rtf/safety_level/BUILD | 17 + .../safety_level/SafetyLevelMarshaller.scala | 24 + .../marshaller/response/slice/BUILD | 16 + .../response/slice/CursorTypeMarshaller.scala | 29 + .../response/slice/SliceItemMarshaller.scala | 147 + .../slice/SliceTransportMarshaller.scala | 26 + .../marshaller/response/urp/BUILD | 22 + .../response/urp/PageBodyMarshaller.scala | 21 + .../response/urp/PageHeaderMarshaller.scala | 17 + .../response/urp/PageNavBarMarshaller.scala | 21 + .../urp/SegmentedTimelineMarshaller.scala | 21 + .../urp/SegmentedTimelinesMarshaller.scala | 17 + .../response/urp/TimelineKeyMarshaller.scala | 54 + .../response/urp/TitleNavBarMarshaller.scala | 19 + ...TopicPageHeaderDisplayTypeMarshaller.scala | 19 + .../TopicPageHeaderFacepileMarshaller.scala | 18 + .../urp/TopicPageHeaderMarshaller.scala | 25 + .../urp/TopicPageNavBarMarshaller.scala | 18 + .../response/urp/UrpTransportMarshaller.scala | 29 + .../urp/UrpTransportMarshallerBuilder.scala | 65 + .../urt/AddEntriesInstructionMarshaller.scala | 15 + .../AddToModuleInstructionMarshaller.scala | 17 + .../marshaller/response/urt/BUILD | 40 + .../response/urt/CoverMarshaller.scala | 32 + ...rkEntriesUnreadInstructionMarshaller.scala | 13 + .../response/urt/ModuleItemMarshaller.scala | 23 + .../urt/ModuleItemTreeDisplayMarshaller.scala | 20 + .../urt/PinEntryInstructionMarshaller.scala | 15 + .../urt/ReaderModeConfigMarshaller.scala | 17 + .../ReplaceEntryInstructionMarshaller.scala | 26 + .../urt/ShowAlertInstructionMarshaller.scala | 41 + ...rminateTimelineInstructionMarshaller.scala | 22 + .../urt/TimelineEntryContentMarshaller.scala | 25 + .../urt/TimelineEntryMarshaller.scala | 22 + .../urt/TimelineInstructionMarshaller.scala | 50 + .../urt/TimelineItemContentMarshaller.scala | 130 + .../response/urt/TimelineItemMarshaller.scala | 22 + .../urt/TimelineMetadataMarshaller.scala | 18 + .../urt/TimelineModuleMarshaller.scala | 36 + .../urt/TimelineOperationMarshaller.scala | 24 + .../urt/TimelineScribeConfigMarshaller.scala | 17 + .../response/urt/UrtTransportMarshaller.scala | 101 + .../urt/UrtTransportMarshallerBuilder.scala | 711 ++++ .../marshaller/response/urt/alert/BUILD | 25 + ...howAlertColorConfigurationMarshaller.scala | 19 + .../ShowAlertDisplayLocationMarshaller.scala | 19 + .../ShowAlertIconDisplayInfoMarshaller.scala | 21 + .../urt/alert/ShowAlertIconMarshaller.scala | 17 + ...howAlertNavigationMetadataMarshaller.scala | 14 + .../urt/alert/ShowAlertTypeMarshaller.scala | 17 + .../marshaller/response/urt/button/BUILD | 21 + .../urt/button/ButtonStyleMarshaller.scala | 29 + .../urt/button/CtaButtonMarshaller.scala | 19 + .../urt/button/IconCtaButtonMarshaller.scala | 21 + .../urt/button/TextCtaButtonMarshaller.scala | 18 + .../marshaller/response/urt/color/BUILD | 17 + .../response/urt/color/ColorMarshaller.scala | 16 + .../urt/color/ColorPaletteMarshaller.scala | 16 + .../urt/color/RosettaColorMarshaller.scala | 48 + .../marshaller/response/urt/cover/BUILD | 28 + .../urt/cover/CoverContentMarshaller.scala | 19 + .../cover/CoverCtaBehaviorMarshaller.scala | 25 + .../urt/cover/CoverCtaMarshaller.scala | 28 + .../urt/cover/CoverImageMarshaller.scala | 23 + .../cover/FullCoverContentMarshaller.scala | 37 + .../FullCoverDisplayTypeMarshaller.scala | 16 + .../cover/HalfCoverContentMarshaller.scala | 33 + .../HalfCoverDisplayTypeMarshaller.scala | 18 + .../marshaller/response/urt/icon/BUILD | 17 + .../urt/icon/HorizonIconMarshaller.scala | 52 + .../marshaller/response/urt/item/BUILD | 28 + .../ArticleDisplayTypeMarshaller.scala | 15 + .../item/article/ArticleItemMarshaller.scala | 23 + .../article/ArticleSeedTypeMarshaller.scala | 20 + .../AudioSpaceItemMarshaller.scala | 17 + .../item/card/CardDisplayTypeMarshaller.scala | 16 + .../urt/item/card/CardItemMarshaller.scala | 25 + .../CommerceProductGroupItemMarshaller.scala | 14 + .../CommerceProductItemMarshaller.scala | 13 + .../ConversationAnnotationMarshaller.scala | 22 + ...ConversationAnnotationTypeMarshaller.scala | 20 + .../EventSummaryDisplayTypeMarshaller.scala | 24 + .../event/EventSummaryItemMarshaller.scala | 27 + .../ForwardPivotDisplayTypeMarshaller.scala | 20 + .../ForwardPivotMarshaller.scala | 35 + ...oftInterventionDisplayTypeMarshaller.scala | 24 + .../GenericSummaryActionMarshaller.scala | 20 + .../GenericSummaryContextMarshaller.scala | 20 + .../GenericSummaryDisplayTypeMarshaller.scala | 18 + .../GenericSummaryItemMarshaller.scala | 33 + .../HighlightedSectionMarshaller.scala | 16 + .../icon_label/IconLabelItemMarshaller.scala | 22 + .../label/LabelDisplayTypeMarshaller.scala | 15 + .../urt/item/label/LabelItemMarshaller.scala | 25 + ...ompactPromptMessageContentMarshaller.scala | 29 + ...rImagePromptMessageContentMarshaller.scala | 33 + ...InlinePromptMessageContentMarshaller.scala | 32 + .../message/MessageActionMarshaller.scala | 25 + .../message/MessageActionTypeMarshaller.scala | 15 + .../message/MessageContentMarshaller.scala | 25 + .../item/message/MessageImageMarshaller.scala | 19 + .../message/MessagePromptItemMarshaller.scala | 23 + .../message/MessageTextActionMarshaller.scala | 17 + .../UserFacepileDisplayTypeMarshaller.scala | 18 + .../item/message/UserFacepileMarshaller.scala | 23 + .../MomentAnnotationItemMarshaller.scala | 20 + .../item/prompt/PromptContentMarshaller.scala | 17 + .../item/prompt/PromptItemMarshaller.scala | 26 + .../RelevancePromptContentMarshaller.scala | 29 + ...RelevancePromptDisplayTypeMarshaller.scala | 18 + ...PromptFollowUpFeedbackTypeMarshaller.scala | 19 + ...ncePromptFollowUpTextInputMarshaller.scala | 20 + .../SpellingActionTypeMarshaller.scala | 16 + .../suggestion/SpellingItemMarshaller.scala | 22 + .../suggestion/TextResultMarshaller.scala | 25 + .../ThreadHeaderContentMarshaller.scala | 14 + .../thread/ThreadHeaderItemMarshaller.scala | 18 + .../CallToActionTileContentMarshaller.scala | 21 + .../tile/StandardTileContentMarshaller.scala | 19 + .../urt/item/tile/TileContentMarshaller.scala | 21 + .../urt/item/tile/TileItemMarshaller.scala | 28 + .../TombstoneDisplayTypeMarshaller.scala | 24 + .../tombstone/TombstoneInfoMarshaller.scala | 18 + .../tombstone/TombstoneItemMarshaller.scala | 23 + .../topic/TopicDisplayTypeMarshaller.scala | 21 + ...picFollowPromptDisplayTypeMarshaller.scala | 21 + .../TopicFollowPromptItemMarshaller.scala | 22 + .../TopicFunctionalityTypeMarshaller.scala | 20 + .../urt/item/topic/TopicItemMarshaller.scala | 26 + .../urt/item/trend/TrendItemMarshaller.scala | 37 + .../tweet/TimelinesScoreInfoMarshaller.scala | 13 + .../tweet/TweetDisplayTypeMarshaller.scala | 25 + .../tweet/TweetHighlightsMarshaller.scala | 22 + .../urt/item/tweet/TweetItemMarshaller.scala | 52 + .../TweetComposerDisplayTypeMarshaller.scala | 18 + .../TweetComposerItemMarshaller.scala | 22 + .../TwitterListDisplayTypeMarshaller.scala | 22 + .../TwitterListItemMarshaller.scala | 19 + .../item/user/UserDisplayTypeMarshaller.scala | 20 + .../urt/item/user/UserItemMarshaller.scala | 28 + .../user/UserReactiveTriggersMarshaller.scala | 17 + .../VerticalGridItemContentMarshaller.scala | 17 + .../VerticalGridItemMarshaller.scala | 18 + .../VerticalGridItemTileStyleMarshaller.scala | 20 + ...ItemTopicFunctionalityTypeMarshaller.scala | 21 + .../VerticalGridItemTopicTileMarshaller.scala | 28 + .../urt/media/AspectRatioMarshaller.scala | 15 + .../marshaller/response/urt/media/BUILD | 17 + .../urt/media/BroadcastIdMarshaller.scala | 14 + .../urt/media/MediaEntityMarshaller.scala | 23 + .../urt/media/MediaKeyMarshaller.scala | 15 + .../response/urt/media/MediaMarshaller.scala | 23 + .../response/urt/media/RectMarshaller.scala | 17 + .../urt/media/TweetMediaMarshaller.scala | 15 + .../metadata/ArticleDetailsMarshaller.scala | 15 + .../marshaller/response/urt/metadata/BUILD | 21 + .../urt/metadata/BadgeMarshaller.scala | 18 + .../urt/metadata/CallbackMarshaller.scala | 14 + .../ChildFeedbackActionMarshaller.scala | 33 + .../ClientEventDetailsMarshaller.scala | 26 + .../metadata/ClientEventInfoMarshaller.scala | 21 + .../metadata/CommerceDetailsMarshaller.scala | 18 + .../ConfirmationDisplayTypeMarshaller.scala | 18 + .../ConversationDetailsMarshaller.scala | 15 + .../ConversationSectionMarshaller.scala | 21 + .../urt/metadata/DismissInfoMarshaller.scala | 13 + .../metadata/FeedbackActionMarshaller.scala | 48 + .../FeedbackDisplayContextMarshaller.scala | 15 + .../urt/metadata/FeedbackInfoMarshaller.scala | 22 + .../urt/metadata/FeedbackTypeMarshaller.scala | 27 + .../metadata/GeneralContextMarshaller.scala | 24 + .../GeneralContextTypeMarshaller.scala | 35 + .../ImageAnimationTypeMarshaller.scala | 16 + .../metadata/ImageDisplayTypeMarshaller.scala | 20 + .../urt/metadata/ImageVariantMarshaller.scala | 19 + .../metadata/LiveEventDetailsMarshaller.scala | 14 + .../RichFeedbackBehaviorMarshaller.scala | 55 + .../metadata/SocialContextMarshaller.scala | 22 + .../metadata/TimelinesDetailsMarshaller.scala | 16 + ...icContextFunctionalityTypeMarshaller.scala | 20 + .../urt/metadata/TopicContextMarshaller.scala | 21 + .../response/urt/metadata/UrlMarshaller.scala | 18 + .../urt/metadata/UrlTypeMarshaller.scala | 19 + .../UrtEndpointOptionsMarshaller.scala | 18 + .../marshaller/response/urt/operation/BUILD | 17 + .../CursorDisplayTreatmentMarshaller.scala | 16 + .../urt/operation/CursorItemMarshaller.scala | 21 + .../operation/CursorOperationMarshaller.scala | 21 + .../urt/operation/CursorTypeMarshaller.scala | 38 + .../AdMetadataContainerMarshaller.scala | 26 + .../marshaller/response/urt/promoted/BUILD | 17 + .../urt/promoted/CallToActionMarshaller.scala | 15 + .../ClickTrackingInfoMarshaller.scala | 18 + .../promoted/DisclaimerTypeMarshaller.scala | 17 + .../promoted/DisclosureTypeMarshaller.scala | 21 + .../DynamicPrerollTypeMarshaller.scala | 20 + .../urt/promoted/MediaInfoMarshaller.scala | 24 + .../urt/promoted/PrerollMarshaller.scala | 19 + .../promoted/PrerollMetadataMarshaller.scala | 16 + .../promoted/PromotedMetadataMarshaller.scala | 35 + .../promoted/SkAdNetworkDataMarshaller.scala | 23 + .../promoted/SponsorshipTypeMarshaller.scala | 19 + .../promoted/UrlOverrideTypeMarshaller.scala | 17 + .../promoted/VideoVariantsMarshaller.scala | 17 + .../marshaller/response/urt/reaction/BUILD | 17 + .../reaction/TimelineReactionMarshaller.scala | 28 + .../marshaller/response/urt/richtext/BUILD | 21 + .../richtext/ReferenceObjectMarshaller.scala | 31 + .../RichTextAlignmentMarshaller.scala | 17 + .../richtext/RichTextEntityMarshaller.scala | 19 + .../richtext/RichTextFormatMarshaller.scala | 17 + .../urt/richtext/RichTextMarshaller.scala | 19 + .../AdsMetadataMarshaller.scala | 13 + .../response/urt/timeline_module/BUILD | 19 + .../GridCarouselMetadataMarshaller.scala | 13 + ...ModuleConversationMetadataMarshaller.scala | 20 + .../ModuleDisplayTypeMarshaller.scala | 29 + .../ModuleFooterMarshaller.scala | 16 + .../ModuleHeaderDisplayTypeMarshaller.scala | 21 + .../ModuleHeaderMarshaller.scala | 26 + .../ModuleMetadataMarshaller.scala | 21 + .../ModuleShowMoreBehaviorMarshaller.scala | 19 + ...wMoreBehaviorRevealByCountMarshaller.scala | 20 + .../functional_component/premarshaller/BUILD | 17 + .../premarshaller/DomainMarshaller.scala | 66 + .../core/functional_component/scorer/BUILD | 23 + .../scorer/ScoredCandidateResult.scala | 13 + .../functional_component/scorer/Scorer.scala | 36 + .../core/functional_component/selector/BUILD | 19 + .../selector/Selector.scala | 24 + .../selector/SelectorResult.scala | 11 + .../functional_component/side_effect/BUILD | 19 + .../side_effect/ExecuteSynchronously.scala | 22 + .../PipelineResultSideEffect.scala | 60 + .../side_effect/SideEffect.scala | 28 + .../functional_component/transformer/BUILD | 23 + .../CandidateFeatureTransformer.scala | 8 + .../CandidatePipelineQueryTransformer.scala | 85 + .../CandidatePipelineResultsTransformer.scala | 45 + .../transformer/FeatureTransformer.scala | 27 + .../transformer/Transformer.scala | 15 + .../com/twitter/product_mixer/core/gate/BUILD | 15 + .../core/gate/DenyLoggedOutUsersGate.scala | 31 + .../product_mixer/core/gate/ParamGate.scala | 22 + .../core/gate/ParamNotGate.scala | 14 + .../product_mixer/core/model/common/BUILD | 25 + .../model/common/CandidateWithFeatures.scala | 18 + .../core/model/common/Component.scala | 18 + .../core/model/common/Conditionally.scala | 57 + .../core/model/common/UniversalNoun.scala | 8 + .../core/model/common/identifier/BUILD | 15 + .../CandidatePipelineIdentifier.scala | 70 + .../CandidateSourceIdentifier.scala | 70 + .../identifier/ComponentIdentifier.scala | 111 + .../ComponentIdentifierSerializer.scala | 21 + .../identifier/ComponentIdentifierStack.scala | 64 + .../ComponentIdentifierStackSerializer.scala | 14 + .../identifier/DecoratorIdentifier.scala | 70 + .../DomainMarshallerIdentifier.scala | 70 + .../FeatureHydratorIdentifier.scala | 70 + .../common/identifier/FilterIdentifier.scala | 70 + .../common/identifier/GateIdentifier.scala | 70 + .../identifier/MixerPipelineIdentifier.scala | 70 + .../identifier/PipelineStepIdentifier.scala | 90 + .../identifier/PlatformIdentifier.scala | 70 + .../common/identifier/ProductIdentifier.scala | 70 + .../ProductPipelineIdentifier.scala | 70 + .../RecommendationPipelineIdentifier.scala | 74 + .../common/identifier/RootIdentifier.scala | 66 + .../common/identifier/ScorerIdentifier.scala | 70 + .../ScoringPipelineIdentifier.scala | 70 + .../identifier/SelectorIdentifier.scala | 79 + .../identifier/SideEffectIdentifier.scala | 70 + .../identifier/TransformerIdentifier.scala | 70 + .../TransportMarshallerIdentifier.scala | 70 + .../core/model/common/presentation/BUILD | 19 + .../presentation/CandidateFeatures.scala | 32 + .../presentation/CandidateWithDetails.scala | 140 + .../presentation/ItemPresentation.scala | 6 + .../presentation/ModulePresentation.scala | 3 + .../presentation/UniversalPresentation.scala | 16 + .../model/common/presentation/slice/BUILD | 15 + .../slice/BaseSliceItemPresentation.scala | 8 + .../core/model/common/presentation/urt/BUILD | 15 + .../urt/BaseUrtItemPresentation.scala | 9 + .../urt/BaseUrtModulePresentation.scala | 8 + .../urt/BaseUrtOperationPresentation.scala | 9 + .../presentation/urt/IsDispensable.scala | 14 + .../urt/WithItemTreeDisplay.scala | 11 + .../core/model/marshalling/BUILD | 8 + .../model/marshalling/HasMarshalling.scala | 3 + .../core/model/marshalling/request/BUILD | 24 + .../marshalling/request/ClientContext.scala | 94 + .../marshalling/request/DebugOptions.scala | 15 + .../marshalling/request/DebugParams.scala | 8 + .../marshalling/request/HasExcludedIds.scala | 8 + .../request/HasSerializedRequestCursor.scala | 11 + .../model/marshalling/request/Product.scala | 25 + .../marshalling/request/ProductContext.scala | 7 + .../model/marshalling/request/Request.scala | 10 + .../response/rtf/safety_level/BUILD | 13 + .../rtf/safety_level/SafetyLevel.scala | 13 + .../model/marshalling/response/slice/BUILD | 13 + .../response/slice/SliceItem.scala | 84 + .../core/model/marshalling/response/urp/BUILD | 15 + .../model/marshalling/response/urp/Page.scala | 12 + .../marshalling/response/urp/PageBody.scala | 10 + .../marshalling/response/urp/PageHeader.scala | 15 + .../marshalling/response/urp/PageNavBar.scala | 19 + .../response/urp/SegmentedTimeline.scala | 10 + .../response/urp/TimelineKey.scala | 32 + .../urp/TopicPageHeaderDisplayType.scala | 6 + .../urp/TopicPageHeaderFacepile.scala | 7 + .../core/model/marshalling/response/urt/BUILD | 24 + .../marshalling/response/urt/Cover.scala | 3 + .../response/urt/EntryNamespace.scala | 50 + .../response/urt/HasEntryIdentifier.scala | 8 + .../response/urt/HasExpirationTime.scala | 9 + .../response/urt/HasSortIndex.scala | 7 + .../response/urt/ModuleItemTreeDisplay.scala | 9 + .../response/urt/ReaderModeConfig.scala | 5 + .../marshalling/response/urt/ShowAlert.scala | 46 + .../marshalling/response/urt/Timeline.scala | 10 + .../response/urt/TimelineEntry.scala | 56 + .../response/urt/TimelineInstruction.scala | 65 + .../response/urt/TimelineMetadata.scala | 6 + .../response/urt/TimelineScribeConfig.scala | 6 + .../marshalling/response/urt/alert/BUILD | 19 + .../alert/ShowAlertColorConfiguration.scala | 9 + .../urt/alert/ShowAlertDisplayLocation.scala | 5 + .../response/urt/alert/ShowAlertIcon.scala | 5 + .../urt/alert/ShowAlertIconDisplayInfo.scala | 5 + .../alert/ShowAlertNavigationMetadata.scala | 3 + .../response/urt/alert/ShowAlertType.scala | 5 + .../marshalling/response/urt/button/BUILD | 15 + .../response/urt/button/ButtonStyle.scala | 12 + .../response/urt/button/CtaButton.scala | 11 + .../marshalling/response/urt/color/BUILD | 8 + .../response/urt/color/Color.scala | 7 + .../response/urt/color/ColorPalette.scala | 5 + .../response/urt/color/RosettaColor.scala | 47 + .../response/urt/contextual_ref/BUILD | 15 + .../contextual_ref/ContextualTweetRef.scala | 5 + .../contextual_ref/OuterTweetContext.scala | 6 + .../TweetHydrationContext.scala | 7 + .../marshalling/response/urt/cover/BUILD | 16 + .../response/urt/cover/CoverContent.scala | 34 + .../response/urt/cover/CoverCta.scala | 14 + .../response/urt/cover/CoverCtaBehavior.scala | 9 + .../response/urt/cover/CoverImage.scala | 10 + .../urt/cover/FullCoverDisplayType.scala | 5 + .../urt/cover/HalfCoverDisplayType.scala | 6 + .../response/urt/cover/ShowCover.scala | 49 + .../model/marshalling/response/urt/icon/BUILD | 9 + .../response/urt/icon/HorizonIcon.scala | 43 + .../model/marshalling/response/urt/item/BUILD | 24 + .../urt/item/article/ArticleDisplayType.scala | 5 + .../urt/item/article/ArticleItem.scala | 26 + .../urt/item/article/ArticleSeedType.scala | 18 + .../urt/item/audio_space/AudioSpaceItem.scala | 22 + .../urt/item/card/CardDisplayType.scala | 7 + .../response/urt/item/card/CardItem.scala | 28 + .../commerce/CommerceProductGroupItem.scala | 23 + .../item/commerce/CommerceProductItem.scala | 23 + .../ConversationAnnotation.scala | 8 + .../ConversationAnnotationType.scala | 6 + .../item/event/EventSummaryDisplayType.scala | 7 + .../urt/item/event/EventSummaryItem.scala | 30 + .../urt/item/forward_pivot/ForwardPivot.scala | 18 + .../ForwardPivotDisplayType.scala | 7 + .../SoftInterventionDisplayType.scala | 8 + .../GenericSummaryAction.scala | 8 + .../GenericSummaryContext.scala | 8 + .../GenericSummaryDisplayType.scala | 5 + .../generic_summary/GenericSummaryItem.scala | 34 + .../item/highlight/HighlightedSection.scala | 3 + .../urt/item/icon_label/IconLabelItem.scala | 26 + .../urt/item/label/LabelDisplayType.scala | 6 + .../response/urt/item/label/LabelItem.scala | 28 + .../urt/item/message/MessageAction.scala | 10 + .../urt/item/message/MessageActionType.scala | 5 + .../urt/item/message/MessageContent.scala | 38 + .../urt/item/message/MessageImage.scala | 7 + .../urt/item/message/MessagePromptItem.scala | 27 + .../urt/item/message/MessageTextAction.scala | 5 + .../urt/item/message/UserFacepile.scala | 9 + .../message/UserFacepileDisplayType.scala | 6 + .../item/moment/MomentAnnotationItem.scala | 33 + .../urt/item/prompt/PromptContent.scala | 28 + .../response/urt/item/prompt/PromptItem.scala | 26 + .../prompt/RelevancePromptDisplayType.scala | 13 + .../RelevancePromptFollowUpFeedbackType.scala | 16 + .../item/suggestion/SpellingActionType.scala | 26 + .../urt/item/suggestion/SpellingItem.scala | 32 + .../urt/item/suggestion/TextResult.scala | 14 + .../urt/item/thread/ThreadHeaderContent.scala | 5 + .../urt/item/thread/ThreadHeaderItem.scala | 24 + .../response/urt/item/tile/TileContent.scala | 21 + .../response/urt/item/tile/TileItem.scala | 29 + .../item/tombstone/TombstoneDisplayType.scala | 9 + .../urt/item/tombstone/TombstoneInfo.scala | 8 + .../urt/item/tombstone/TombstoneItem.scala | 26 + .../urt/item/topic/TopicDisplayType.scala | 8 + .../topic/TopicFollowPromptDisplayType.scala | 6 + .../item/topic/TopicFollowPromptItem.scala | 26 + .../item/topic/TopicFunctionalityType.scala | 7 + .../response/urt/item/topic/TopicItem.scala | 24 + .../response/urt/item/trend/TrendItem.scala | 35 + .../urt/item/tweet/TimelinesScoreInfo.scala | 3 + .../urt/item/tweet/TweetDisplayType.scala | 16 + .../urt/item/tweet/TweetHighlights.scala | 8 + .../response/urt/item/tweet/TweetItem.scala | 62 + .../TweetComposerDisplayType.scala | 6 + .../tweet_composer/TweetComposerItem.scala | 26 + .../twitter_list/TwitterListDisplayType.scala | 8 + .../item/twitter_list/TwitterListItem.scala | 23 + .../urt/item/user/UserDisplayType.scala | 7 + .../response/urt/item/user/UserItem.scala | 30 + .../urt/item/user/UserReactiveTriggers.scala | 5 + .../vertical_grid_item/VerticalGridItem.scala | 29 + .../VerticalGridItemTileStyle.scala | 6 + ...rticalGridItemTopicFunctionalityType.scala | 8 + .../response/urt/media/AspectRatio.scala | 5 + .../marshalling/response/urt/media/BUILD | 13 + .../response/urt/media/Media.scala | 7 + .../response/urt/media/MediaEntity.scala | 14 + .../response/urt/media/MediaKey.scala | 5 + .../marshalling/response/urt/media/Rect.scala | 7 + .../urt/metadata/ArticleDetails.scala | 5 + .../marshalling/response/urt/metadata/BUILD | 17 + .../response/urt/metadata/Badge.scala | 8 + .../response/urt/metadata/Callback.scala | 3 + .../urt/metadata/ClientEventInfo.scala | 29 + .../urt/metadata/CommerceDetails.scala | 8 + .../metadata/ConfirmationDisplayType.scala | 6 + .../urt/metadata/ConversationDetails.scala | 3 + .../urt/metadata/ConversationSection.scala | 8 + .../response/urt/metadata/DismissInfo.scala | 3 + .../urt/metadata/FeedbackAction.scala | 29 + .../urt/metadata/FeedbackActionInfo.scala | 15 + .../response/urt/metadata/FeedbackInfo.scala | 13 + .../response/urt/metadata/FeedbackType.scala | 18 + .../urt/metadata/ImageAnimationType.scala | 5 + .../urt/metadata/ImageDisplayType.scala | 7 + .../response/urt/metadata/ImageVariant.scala | 9 + .../urt/metadata/LiveEventDetails.scala | 3 + .../urt/metadata/MarkUnreadableEntry.scala | 6 + .../response/urt/metadata/PinnableEntry.scala | 5 + .../urt/metadata/ReplaceableEntry.scala | 5 + .../response/urt/metadata/ReplyPinState.scala | 7 + .../urt/metadata/RichFeedbackBehavior.scala | 15 + .../response/urt/metadata/SocialContext.scala | 49 + .../urt/metadata/TimelinesDetails.scala | 6 + .../response/urt/metadata/Url.scala | 17 + .../marshalling/response/urt/operation/BUILD | 15 + .../operation/CursorDisplayTreatment.scala | 5 + .../response/urt/operation/CursorItem.scala | 31 + .../urt/operation/CursorOperation.scala | 31 + .../response/urt/operation/CursorType.scala | 37 + .../urt/promoted/AdMetadataContainer.scala | 11 + .../marshalling/response/urt/promoted/BUILD | 11 + .../response/urt/promoted/CallToAction.scala | 5 + .../urt/promoted/ClickTrackingInfo.scala | 8 + .../urt/promoted/DisclaimerType.scala | 6 + .../urt/promoted/DisclosureType.scala | 8 + .../urt/promoted/DynamicPrerollType.scala | 7 + .../response/urt/promoted/MediaInfo.scala | 11 + .../response/urt/promoted/Preroll.scala | 6 + .../urt/promoted/PrerollMetadata.scala | 5 + .../urt/promoted/PromotedMetadata.scala | 23 + .../urt/promoted/SkAdNetworkData.scala | 13 + .../urt/promoted/SponsorshipType.scala | 7 + .../urt/promoted/UrlOverrideType.scala | 6 + .../response/urt/promoted/VideoVariant.scala | 6 + .../marshalling/response/urt/reaction/BUILD | 11 + .../urt/reaction/TimelineReaction.scala | 5 + .../reaction/TimelineReactionExecution.scala | 10 + .../marshalling/response/urt/richtext/BUILD | 11 + .../urt/richtext/ReferenceObject.scala | 9 + .../response/urt/richtext/RichText.scala | 7 + .../urt/richtext/RichTextAlignment.scala | 6 + .../urt/richtext/RichTextEntity.scala | 7 + .../urt/richtext/RichTextFormat.scala | 13 + .../urt/timeline_module/AdsMetadata.scala | 3 + .../response/urt/timeline_module/BUILD | 13 + .../GridCarouselMetadata.scala | 3 + .../ModuleConversationMetadata.scala | 8 + .../timeline_module/ModuleDisplayType.scala | 12 + .../urt/timeline_module/ModuleFooter.scala | 7 + .../urt/timeline_module/ModuleHeader.scala | 13 + .../ModuleHeaderDisplayType.scala | 7 + .../urt/timeline_module/ModuleMetadata.scala | 11 + .../ModuleShowMoreBehavior.scala | 8 + .../core/module/ABDeciderModule.scala | 51 + .../twitter/product_mixer/core/module/BUILD | 49 + .../core/module/ConfigApiModule.scala | 21 + .../core/module/FeatureSwitchesModule.scala | 53 + .../LoggingThrowableExceptionMapper.scala | 24 + .../PipelineExecutionLoggerModule.scala | 12 + .../core/module/ProductMixerModule.scala | 30 + .../core/module/StratoClientModule.scala | 39 + .../core/module/product_mixer_flags/BUILD | 15 + .../ProductMixerFlagModule.scala | 74 + .../core/module/stringcenter/BUILD | 25 + .../ProductScopeStringCenterModule.scala | 135 + .../twitter/product_mixer/core/pipeline/BUILD | 92 + .../pipeline/CandidatePipelineFeatures.scala | 7 + .../core/pipeline/FailOpenPolicy.scala | 42 + .../pipeline/InvalidStepStateException.scala | 8 + .../pipeline/NewPipelineArrowBuilder.scala | 181 + .../core/pipeline/NewPipelineBuilder.scala | 31 + .../core/pipeline/NewPipelineResult.scala | 22 + .../core/pipeline/NewStepData.scala | 13 + .../core/pipeline/Pipeline.scala | 46 + .../core/pipeline/PipelineBuilder.scala | 194 + .../core/pipeline/PipelineConfig.scala | 20 + .../core/pipeline/PipelineCursor.scala | 38 + .../pipeline/PipelineCursorSerializer.scala | 57 + .../core/pipeline/PipelineQuery.scala | 32 + .../core/pipeline/PipelineResult.scala | 59 + .../core/pipeline/candidate/BUILD | 77 + .../candidate/CandidatePipeline.scala | 30 + .../candidate/CandidatePipelineBuilder.scala | 735 ++++ .../CandidatePipelineBuilderFactory.scala | 56 + .../candidate/CandidatePipelineConfig.scala | 264 ++ .../candidate/CandidatePipelineResult.scala | 93 + .../PassthroughCandidatePipelineConfig.scala | 47 + .../StaticCandidatePipelineConfig.scala | 51 + .../product_mixer/core/pipeline/mixer/BUILD | 62 + .../core/pipeline/mixer/MixerPipeline.scala | 24 + .../pipeline/mixer/MixerPipelineBuilder.scala | 582 +++ .../mixer/MixerPipelineBuilderFactory.scala | 49 + .../pipeline/mixer/MixerPipelineConfig.scala | 175 + .../pipeline/mixer/MixerPipelineResult.scala | 70 + .../core/pipeline/pipeline_failure/BUILD | 14 + .../pipeline_failure/PipelineFailure.scala | 49 + .../PipelineFailureCategory.scala | 190 + .../PipelineFailureClassifier.scala | 13 + .../PipelineFailureSerializer.scala | 67 + .../product_mixer/core/pipeline/product/BUILD | 51 + .../pipeline/product/ProductPipeline.scala | 28 + .../product/ProductPipelineBuilder.scala | 385 ++ .../ProductPipelineBuilderFactory.scala | 39 + .../product/ProductPipelineConfig.scala | 107 + .../product/ProductPipelineRequest.scala | 5 + .../product/ProductPipelineResult.scala | 62 + .../core/pipeline/recommendation/BUILD | 62 + .../RecommendationPipeline.scala | 29 + .../RecommendationPipelineBuilder.scala | 1076 +++++ ...RecommendationPipelineBuilderFactory.scala | 67 + .../RecommendationPipelineConfig.scala | 262 ++ .../RecommendationPipelineResult.scala | 84 + .../product_mixer/core/pipeline/scoring/BUILD | 36 + .../scoring/NewScoringPipelineBuilder.scala | 202 + .../pipeline/scoring/ScoringPipeline.scala | 32 + .../scoring/ScoringPipelineBuilder.scala | 367 ++ .../ScoringPipelineBuilderFactory.scala | 30 + .../scoring/ScoringPipelineConfig.scala | 131 + .../scoring/ScoringPipelineResult.scala | 51 + .../product_mixer/core/pipeline/state/BUILD | 33 + .../pipeline/state/HasAsyncFeatureMap.scala | 9 + .../core/pipeline/state/HasCandidates.scala | 8 + .../state/HasCandidatesWithDetails.scala | 11 + .../state/HasCandidatesWithFeatures.scala | 9 + .../pipeline/state/HasExecutorResults.scala | 13 + .../core/pipeline/state/HasParams.scala | 7 + .../core/pipeline/state/HasQuery.scala | 8 + .../core/pipeline/state/HasRequest.scala | 7 + .../core/pipeline/state/HasResult.scala | 10 + .../product_mixer/core/pipeline/step/BUILD | 17 + .../core/pipeline/step/Step.scala | 52 + .../AsyncFeatureMapStep.scala | 70 + .../step/async_feature_map/BUILD.bazel | 36 + .../step/candidate_feature_hydrator/BUILD | 36 + .../CandidateFeatureHydratorStep.scala | 71 + .../step/candidate_source/BUILD.bazel | 36 + .../CandidateSourceStep.scala | 84 + .../core/pipeline/step/decorator/BUILD.bazel | 36 + .../step/decorator/DecoratorStep.scala | 63 + .../step/domain_marshaller/BUILD.bazel | 30 + .../DomainMarshallerStep.scala | 54 + .../core/pipeline/step/filter/BUILD.bazel | 36 + .../pipeline/step/filter/FilterStep.scala | 64 + .../core/pipeline/step/gate/BUILD | 36 + .../core/pipeline/step/gate/GateStep.scala | 43 + .../pipeline/step/group_results/BUILD.bazel | 29 + .../step/group_results/GroupResultsStep.scala | 67 + .../step/pipeline_executor/BUILD.bazel | 32 + .../PipelineExecutorStep.scala | 81 + .../pipeline/step/pipeline_selector/BUILD | 26 + .../PipelineSelectorStep.scala | 43 + .../pipeline/step/quality_factor/BUILD.bazel | 31 + .../quality_factor/QualityFactorStep.scala | 72 + .../step/query_feature_hydrator/BUILD.bazel | 36 + .../QueryFeatureHydratorStep.scala | 59 + .../step/query_transformer/BUILD.bazel | 29 + .../QueryTransformerStep.scala | 52 + .../core/pipeline/step/scorer/BUILD | 36 + .../pipeline/step/scorer/ScorerStep.scala | 69 + .../core/pipeline/step/selector/BUILD | 36 + .../pipeline/step/selector/SelectorStep.scala | 53 + .../pipeline/step/side_effect/BUILD.bazel | 32 + .../step/side_effect/SideEffectStep.scala | 102 + .../step/transport_marshaller/BUILD.bazel | 31 + .../TransportMarshallerStep.scala | 76 + .../twitter/product_mixer/core/product/BUILD | 21 + .../core/product/ProductParamConfig.scala | 36 + .../product/ProductParamConfigBuilder.scala | 21 + .../product_mixer/core/product/guice/BUILD | 18 + .../core/product/guice/ProductScope.scala | 10 + .../product/guice/ProductScopeModule.scala | 27 + .../core/product/guice/SimpleScope.scala | 68 + .../product_mixer/core/product/registry/BUILD | 26 + .../registry/ProductParamRegistry.scala | 26 + .../registry/ProductPipelineRegistry.scala | 189 + .../ProductPipelineRegistryConfig.scala | 9 + .../product_mixer/core/quality_factor/BUILD | 18 + .../core/quality_factor/Bounds.scala | 33 + .../LinearLatencyQualityFactor.scala | 34 + .../LinearLatencyQualityFactorObserver.scala | 18 + .../core/quality_factor/QualityFactor.scala | 33 + .../quality_factor/QualityFactorConfig.scala | 120 + .../QualityFactorObserver.scala | 18 + .../quality_factor/QualityFactorStatus.scala | 60 + .../QueriesPerSecondBasedQualityFactor.scala | 51 + ...sPerSecondBasedQualityFactorObserver.scala | 21 + .../quality_factor/QueryRateCounter.scala | 21 + .../twitter/product_mixer/core/service/BUILD | 32 + .../product_mixer/core/service/Executor.scala | 700 +++ .../core/service/ExecutorObserver.scala | 146 + .../core/service/ExecutorResult.scala | 3 + .../AsyncFeatureMapExecutor.scala | 58 + .../service/async_feature_map_executor/BUILD | 19 + .../candidate_decorator_executor/BUILD | 22 + .../CandidateDecoratorExecutor.scala | 38 + .../CandidateDecoratorExecutorResult.scala | 6 + .../candidate_feature_hydrator_executor/BUILD | 25 + .../CandidateFeatureHydratorExecutor.scala | 277 ++ ...ndidateFeatureHydratorExecutorResult.scala | 20 + .../BUILD | 22 + .../CandidateFeatureTransformerExecutor.scala | 93 + ...dateFeatureTransformerExecutorResult.scala | 8 + .../service/candidate_pipeline_executor/BUILD | 27 + .../CandidatePipelineExecutor.scala | 82 + .../CandidatePipelineExecutorResult.scala | 8 + .../service/candidate_source_executor/BUILD | 35 + .../CandidateSourceExecutor.scala | 173 + .../CandidateSourceExecutorResult.scala | 10 + .../core/service/component_registry/BUILD | 16 + .../ComponentRegistry.scala | 182 + .../RegisteredComponent.scala | 15 + .../debug_query/AuthorizationService.scala | 82 + .../core/service/debug_query/BUILD | 20 + .../DebugQueryNotSupportedService.scala | 43 + .../debug_query/DebugQueryService.scala | 109 + .../debug_query/ParamsSerializerModule.scala | 35 + .../service/domain_marshaller_executor/BUILD | 23 + .../DomainMarshallerExecutor.scala | 44 + .../service/feature_hydrator_observer/BUILD | 21 + .../FeatureHydratorObserver.scala | 136 + .../core/service/filter_executor/BUILD | 24 + .../filter_executor/FilterExecutor.scala | 172 + .../FilterExecutorResult.scala | 18 + .../core/service/gate_executor/BUILD | 25 + .../gate_executor/ExecutedGateResult.scala | 6 + .../service/gate_executor/GateExecutor.scala | 107 + .../gate_executor/GateExecutorResult.scala | 7 + .../gate_executor/StoppedGateException.scala | 29 + .../core/service/group_results_executor/BUILD | 35 + .../GroupResultsExecutor.scala | 122 + .../AllowListedPipelineExecutionLogger.scala | 183 + .../service/pipeline_execution_logger/BUILD | 33 + .../PipelineExecutionLogger.scala | 7 + .../core/service/pipeline_executor/BUILD | 25 + .../pipeline_executor/PipelineExecutor.scala | 66 + .../PipelineExecutorResult.scala | 8 + .../BUILD | 24 + .../PipelineResultSideEffectExecutor.scala | 91 + .../service/pipeline_selector_executor/BUILD | 26 + .../PipelineSelectorExecutor.scala | 48 + .../PipelineSelectorExecutorResult.scala | 5 + .../service/quality_factor_executor/BUILD | 23 + .../QualityFactorExecutorResult.scala | 10 + ...idualFeatureHydratorResultSerializer.scala | 24 + .../query_feature_hydrator_executor/BUILD | 28 + .../QueryFeatureHydratorExecutor.scala | 217 + .../service/scoring_pipeline_executor/BUILD | 24 + .../ScoringPipelineExecutor.scala | 172 + .../ScoringPipelineExecutorResult.scala | 9 + .../core/service/selector_executor/BUILD | 26 + .../selector_executor/SelectorExecutor.scala | 105 + .../SelectorExecutorResult.scala | 12 + .../product_mixer/core/service/slice/BUILD | 22 + .../core/service/slice/SliceService.scala | 29 + .../core/service/transformer_executor/BUILD | 17 + .../PerCandidateTransformerExecutor.scala | 35 + .../TransformerExecutor.scala | 22 + .../transport_marshaller_executor/BUILD | 23 + .../TransportMarshallerExecutor.scala | 40 + .../product_mixer/core/service/urp/BUILD | 18 + .../core/service/urp/UrpService.scala | 26 + .../product_mixer/core/service/urt/BUILD | 20 + .../core/service/urt/UrtService.scala | 60 + .../com/twitter/product_mixer/core/util/BUILD | 24 + .../product_mixer/core/util/FuturePools.scala | 101 + .../core/util/OffloadFuturePools.scala | 59 + .../core/util/SortIndexBuilder.scala | 17 + .../shared_library/http_client/BUILD | 22 + .../FinagleHttpClientBuilder.scala | 57 + .../FinagleHttpClientWithProxyBuilder.scala | 97 + .../http_client/HttpHostPort.scala | 5 + .../shared_library/manhattan_client/BUILD | 24 + .../ManhattanClientBuilder.scala | 116 + .../shared_library/memcached_client/BUILD | 18 + .../MemcachedClientBuilder.scala | 117 + .../shared_library/observer/BUILD | 19 + .../shared_library/observer/Observer.scala | 203 + .../observer/ResultsObserver.scala | 281 ++ .../observer/ResultsStatsObserver.scala | 243 ++ .../shared_library/thrift_client/BUILD | 20 + .../FinagleThriftClientBuilder.scala | 198 + recos-injector/BUILD.bazel | 1 + recos-injector/CONFIG.ini | 10 + recos-injector/README.md | 44 + recos-injector/server/BUILD | 43 + recos-injector/server/config/BUILD | 20 + .../server/config/change_log_config.ini | 7 + recos-injector/server/config/decider.yml | 11 + .../scala/com/twitter/recosinjector/BUILD | 40 + .../com/twitter/recosinjector/Main.scala | 213 + .../com/twitter/recosinjector/clients/BUILD | 20 + .../recosinjector/clients/Gizmoduck.scala | 26 + .../clients/RecosHoseEntitiesCache.scala | 137 + .../recosinjector/clients/SocialGraph.scala | 80 + .../recosinjector/clients/Tweetypie.scala | 30 + .../recosinjector/clients/UrlResolver.scala | 105 + .../com/twitter/recosinjector/config/BUILD | 36 + .../recosinjector/config/CacheConfig.scala | 23 + .../twitter/recosinjector/config/Config.scala | 41 + .../recosinjector/config/DeployConfig.scala | 215 + .../recosinjector/config/ProdConfig.scala | 29 + .../recosinjector/config/StagingConfig.scala | 33 + .../com/twitter/recosinjector/decider/BUILD | 7 + .../decider/RecosInjectorDecider.scala | 33 + .../com/twitter/recosinjector/edges/BUILD | 23 + .../twitter/recosinjector/edges/Edges.scala | 87 + .../edges/EventToMessageBuilder.scala | 82 + ...cialWriteEventToUserUserGraphBuilder.scala | 73 + ...neEventToUserTweetEntityGraphBuilder.scala | 60 + ...TimelineEventToUserTweetGraphBuilder.scala | 54 + ...etEventToUserTweetEntityGraphBuilder.scala | 343 ++ .../TweetEventToUserTweetGraphBuilder.scala | 88 + .../TweetEventToUserUserGraphBuilder.scala | 65 + ...nifiedUserActionToUserAdGraphBuilder.scala | 44 + ...serActionToUserTweetGraphPlusBuilder.scala | 51 + ...iedUserActionToUserVideoGraphBuilder.scala | 56 + .../edges/UserTweetEntityEdgeBuilder.scala | 80 + .../recosinjector/event_processors/BUILD | 20 + .../event_processors/EventBusProcessor.scala | 60 + .../SocialWriteEventProcessor.scala | 33 + .../TimelineEventProcessor.scala | 150 + .../TweetEventProcessor.scala | 256 ++ .../com/twitter/recosinjector/filters/BUILD | 10 + .../filters/NullCastTweetFilter.scala | 34 + .../recosinjector/filters/TweetFilter.scala | 31 + .../recosinjector/filters/UserFilter.scala | 69 + .../twitter/recosinjector/publishers/BUILD | 12 + .../publishers/KafkaEventPublisher.scala | 54 + .../com/twitter/recosinjector/util/BUILD | 12 + .../recosinjector/util/EventDetails.scala | 126 + .../recosinjector/uua_processors/BUILD | 24 + .../UnifiedUserActionProcessor.scala | 181 + .../UnifiedUserActionsConsumer.scala | 71 + science/search/ingester/config/README.md | 2 + .../config/pipeline-indexer.userupdates.xml | 30 + .../config/pipeline-ingester.protected.xml | 202 + .../config/pipeline-ingester.realtime.xml | 240 ++ .../config/pipeline-ingester.realtime_cg.xml | 199 + simclusters-ann/BUILD.bazel | 1 + simclusters-ann/README.md | 99 + simclusters-ann/server/BUILD | 23 + .../server/src/main/resources/BUILD | 7 + .../src/main/resources/config/decider.yml | 95 + .../server/src/main/resources/logback.xml | 167 + .../scala/com/twitter/simclustersann/BUILD | 31 + .../simclustersann/SimclustersAnnServer.scala | 70 + .../SimclustersAnnWarmupHandler.scala | 73 + .../ApproximateCosineSimilarity.scala | 129 + .../simclustersann/candidate_source/BUILD | 14 + ...erimentalApproximateCosineSimilarity.scala | 131 + ...OptimizedApproximateCosineSimilarity.scala | 112 + .../SimClustersANNCandidateSource.scala | 102 + .../com/twitter/simclustersann/common/BUILD | 5 + .../simclustersann/common/FlagNames.scala | 31 + .../twitter/simclustersann/controllers/BUILD | 29 + .../SimClustersANNController.scala | 80 + .../twitter/simclustersann/exceptions/BUILD | 12 + ...estForSimClustersAnnVariantException.scala | 16 + ...SimClustersAnnVariantExceptionMapper.scala | 27 + ...figForSimClustersAnnVariantException.scala | 6 + .../com/twitter/simclustersann/filters/BUILD | 13 + ...etTweetCandidatesResponseStatsFilter.scala | 43 + .../filters/SimClustersAnnVariantFilter.scala | 53 + .../com/twitter/simclustersann/modules/BUILD | 24 + .../simclustersann/modules/CacheModule.scala | 34 + .../modules/ClusterConfigMapperModule.scala | 15 + .../modules/ClusterConfigModule.scala | 25 + .../ClusterTweetIndexProviderModule.scala | 95 + .../CustomMtlsThriftWebFormsModule.scala | 99 + .../modules/EmbeddingStoreModule.scala | 110 + .../simclustersann/modules/FlagsModule.scala | 44 + .../modules/FuturePoolProvider.scala | 27 + .../modules/RateLimiterModule.scala | 23 + .../modules/ServiceNameMapperModule.scala | 15 + .../SimClustersANNCandidateSourceModule.scala | 47 + .../modules/StratoClientProviderModule.scala | 20 + simclusters-ann/thrift/src/main/thrift/BUILD | 16 + .../src/main/thrift/simClustersAnn.thrift | 59 + src/java/com/twitter/search/README.md | 50 + src/java/com/twitter/search/common/README.md | 1 + .../search/common/converter/earlybird/BUILD | 57 + .../earlybird/BasicIndexingConverter.java | 647 +++ .../earlybird/CombinedIndexingConverter.java | 99 + .../earlybird/DelayedIndexingConverter.java | 594 +++ .../earlybird/EncodedFeatureBuilder.java | 531 +++ .../search/common/encoding/docvalues/BUILD | 20 + .../encoding/docvalues/CSFTypeUtil.java | 34 + .../search/common/encoding/features/BUILD | 17 + .../encoding/features/BinByteNormalizer.java | 73 + .../encoding/features/ByteNormalizer.java | 38 + .../features/ClampByteNormalizer.java | 47 + .../encoding/features/EncodedFeatures.java | 58 + .../encoding/features/IntNormalizer.java | 15 + .../features/IntegerEncodedFeatures.java | 159 + .../encoding/features/LogByteNormalizer.java | 53 + .../features/PredictionScoreNormalizer.java | 51 + .../SingleBytePositiveFloatNormalizer.java | 35 + .../features/SingleBytePositiveFloatUtil.java | 164 + .../features/SmartIntegerNormalizer.java | 150 + .../com/twitter/search/common/query/BUILD | 25 + .../search/common/query/BoostUtils.java | 27 + .../query/CollectAnnotationsVisitor.java | 92 + .../common/query/CollectQueryTypeVisitor.java | 89 + .../common/query/CollectVariantVisitor.java | 13 + .../common/query/DefaultFilterWeight.java | 60 + .../search/common/query/DocIdFilter.java | 74 + .../search/common/query/FieldRankHitInfo.java | 48 + .../search/common/query/FieldWeightUtil.java | 205 + .../search/common/query/FilteredQuery.java | 225 + .../search/common/query/FilteredScorer.java | 36 + .../common/query/HitAttributeCollector.java | 101 + .../common/query/HitAttributeHelper.java | 102 + .../common/query/HitAttributeProvider.java | 12 + .../common/query/IDDisjunctionQuery.java | 378 ++ .../common/query/IdentifiableQuery.java | 77 + .../common/query/IdentifiableQueryScorer.java | 60 + .../common/query/IdentifiableQueryWeight.java | 58 + .../search/common/query/MappableField.java | 34 + .../query/MultiTermDisjunctionQuery.java | 61 + .../query/QueryCommonFieldHitsVisitor.java | 160 + .../common/query/QueryHitAttributeHelper.java | 81 + .../search/common/query/QueryRankVisitor.java | 56 + .../query/SingleDocDocIdSetIterator.java | 51 + .../query/StaticHitAttributeProvider.java | 32 + .../com/twitter/search/common/relevance/BUILD | 257 ++ .../search/common/relevance/NGramCache.java | 152 + .../TrendsThriftDataServiceManager.java | 353 ++ .../classifiers/TweetClassifier.java | 118 + .../relevance/classifiers/TweetEvaluator.java | 37 + .../classifiers/TweetOffensiveEvaluator.java | 260 ++ .../TweetQualityFeatureExtractor.java | 105 + .../classifiers/TweetTextClassifier.java | 67 + .../classifiers/TweetTextEvaluator.java | 54 + .../classifiers/TweetTrendsExtractor.java | 165 + .../config/TweetProcessingConfig.java | 114 + .../common/relevance/entities/GeoObject.java | 201 + .../entities/PotentialLocationObject.java | 122 + .../relevance/entities/TwitterMessage.java | 1267 ++++++ .../entities/TwitterMessageUser.java | 231 + .../entities/TwitterMessageUtil.java | 444 ++ .../entities/TwitterQuotedMessage.java | 41 + .../entities/TwitterRetweetMessage.java | 80 + .../common/relevance/features/AgeDecay.java | 88 + .../search/common/relevance/features/BUILD | 24 + .../features/EarlybirdDocumentFeatures.java | 232 + .../relevance/features/FeatureSink.java | 75 + .../relevance/features/IntNormalizers.java | 39 + .../features/MutableFeatureNormalizers.java | 23 + .../relevance/features/QueryFeatureType.java | 9 + .../features/RelevanceSignalConstants.java | 30 + .../relevance/features/ScoringUtils.java | 24 + .../common/relevance/features/TermVector.java | 79 + .../features/TweetEngagementFeatures.java | 57 + .../relevance/features/TweetFeatureType.java | 291 ++ .../relevance/features/TweetFeatures.java | 19 + .../TweetIntegerShingleSignature.java | 201 + .../features/TweetSignatureUtil.java | 15 + .../relevance/features/TweetTextFeatures.java | 225 + .../relevance/features/TweetTextQuality.java | 69 + .../relevance/features/TweetUserFeatures.java | 114 + .../common/relevance/scorers/TweetScorer.java | 65 + .../relevance/scorers/TweetTextScorer.java | 242 ++ .../common/relevance/text/LocationUtils.java | 41 + .../common/relevance/text/TweetParser.java | 190 + .../text/VisibleTokenRatioNormalizer.java | 39 + .../search/common/schema/AnalyzerFactory.java | 142 + .../com/twitter/search/common/schema/BUILD | 34 + .../search/common/schema/DynamicSchema.java | 214 + .../search/common/schema/ImmutableSchema.java | 904 ++++ .../search/common/schema/NumericField.java | 44 + .../search/common/schema/SchemaBuilder.java | 693 +++ .../common/schema/SchemaDocumentFactory.java | 433 ++ .../search/common/schema/SchemaUtil.java | 102 + .../schema/SearchWhitespaceAnalyzer.java | 27 + .../common/schema/ThriftDocumentBuilder.java | 228 + .../twitter/search/common/schema/base/BUILD | 25 + .../schema/base/EarlybirdFieldType.java | 374 ++ .../schema/base/FeatureConfiguration.java | 316 ++ .../schema/base/FieldNameToIdMapping.java | 28 + .../schema/base/FieldWeightDefault.java | 110 + .../schema/base/ImmutableSchemaInterface.java | 14 + .../base/IndexedNumericFieldSettings.java | 37 + .../search/common/schema/base/Schema.java | 231 + .../schema/base/ThriftDocumentUtil.java | 146 + .../search/common/schema/earlybird/BUILD | 93 + .../schema/earlybird/EarlybirdCluster.java | 90 + .../earlybird/EarlybirdEncodedFeatures.java | 148 + .../EarlybirdEncodedFeaturesUtil.java | 36 + .../earlybird/EarlybirdFieldConstants.java | 1132 +++++ .../earlybird/EarlybirdSchemaBuilder.java | 96 + .../earlybird/EarlybirdSchemaCreateTool.java | 702 +++ .../EarlybirdThriftDocumentBuilder.java | 897 ++++ .../EarlybirdThriftDocumentUtil.java | 377 ++ .../common/schema/earlybird/FlushVersion.java | 336 ++ .../common/search/AndNotDocIdSetIterator.java | 71 + .../com/twitter/search/common/search/BUILD | 33 + .../DelegatingEarlyTerminationCollector.java | 75 + .../search/common/search/DocIdTracker.java | 12 + .../common/search/EarlyTerminationState.java | 51 + .../search/GeoQuadTreeQueryBuilderUtil.java | 65 + .../search/IntArrayDocIdSetIterator.java | 76 + .../common/search/PairDocIdSetIterator.java | 82 + .../common/search/QueryCostProvider.java | 9 + .../common/search/TerminationTracker.java | 202 + .../common/search/TwitterCollector.java | 31 + .../TwitterEarlyTerminationCollector.java | 328 ++ .../common/search/TwitterIndexSearcher.java | 189 + .../search/common/search/termination/BUILD | 20 + .../search/termination/QueryTimeout.java | 24 + .../termination/QueryTimeoutFactory.java | 34 + .../search/termination/QueryTimeoutImpl.java | 65 + .../search/termination/TerminationQuery.java | 66 + .../termination/TerminationQueryScorer.java | 91 + .../termination/TerminationQueryWeight.java | 53 + .../search/common/util/earlybird/BUILD | 32 + .../earlybird/EarlybirdResponseMergeUtil.java | 269 ++ .../util/earlybird/EarlybirdResponseUtil.java | 204 + .../util/earlybird/FacetsResultsUtils.java | 495 +++ .../util/earlybird/ResponseMergerUtils.java | 45 + .../common/util/earlybird/ResultsUtil.java | 36 + .../util/earlybird/TermStatisticsUtil.java | 47 + .../util/earlybird/ThriftSearchQueryUtil.java | 29 + .../earlybird/ThriftSearchResultUtil.java | 209 + ...ThriftSearchResultsRelevanceStatsUtil.java | 46 + .../com/twitter/search/common/util/lang/BUILD | 18 + .../common/util/lang/ThriftLanguageUtil.java | 141 + .../com/twitter/search/common/util/ml/BUILD | 16 + .../common/util/ml/EnumBasedLinearModel.java | 141 + .../search/common/util/ml/FeatureUtils.java | 120 + .../common/util/ml/MapBasedLinearModel.java | 32 + .../util/ml/StringMapBasedLinearModel.java | 125 + .../common/util/ml/models_manager/BUILD | 14 + .../ml/models_manager/BaseModelsManager.java | 293 ++ .../common/util/ml/prediction_engine/BUILD | 68 + .../BaseLegacyScoreAccumulator.java | 64 + .../prediction_engine/BaseModelBuilder.java | 111 + .../BaseScoreAccumulator.java | 48 + .../CompositeFeatureContext.java | 35 + .../DecisionForestModelsManager.java | 69 + .../prediction_engine/DiscretizedFeature.java | 47 + .../DiscretizedFeatureRange.java | 33 + .../prediction_engine/LegacyModelBuilder.java | 86 + .../LightweightLinearModel.java | 187 + .../ml/prediction_engine/ModelBuilder.java | 16 + .../ml/prediction_engine/ModelLoader.java | 178 + .../PredictionEngineModelsManager.java | 67 + .../SchemaBasedModelBuilder.java | 105 + .../SchemaBasedScoreAccumulator.java | 64 + .../common/util/ml/tensorflow_engine/BUILD | 21 + .../TensorflowModelsManager.java | 189 + .../com/twitter/search/core/earlybird/BUILD | 38 + .../twitter/search/core/earlybird/README.md | 21 + .../facets/AbstractFacetCountingArray.java | 231 + .../facets/CSFFacetCountIterator.java | 56 + .../facets/CompositeFacetCountIterator.java | 46 + .../facets/DummyFacetAccumulator.java | 28 + .../facets/EarlybirdFacetDocValueSet.java | 153 + .../earlybird/facets/EarlybirdFacets.java | 102 + .../facets/EarlybirdFacetsFactory.java | 48 + .../earlybird/facets/FacetAccumulator.java | 36 + .../facets/FacetCountAggregator.java | 93 + .../earlybird/facets/FacetCountIterator.java | 57 + .../facets/FacetCountIteratorFactory.java | 23 + .../earlybird/facets/FacetCountState.java | 88 + .../earlybird/facets/FacetCountingArray.java | 156 + .../facets/FacetCountingArrayWriter.java | 55 + .../core/earlybird/facets/FacetIDMap.java | 161 + .../earlybird/facets/FacetLabelProvider.java | 206 + .../facets/FacetResponseRewriter.java | 16 + .../earlybird/facets/FacetTermCollector.java | 16 + .../core/earlybird/facets/FacetUtil.java | 106 + .../earlybird/facets/LanguageHistogram.java | 104 + .../facets/OptimizedFacetCountingArray.java | 82 + .../facets/PerfieldFacetCountAggregator.java | 96 + .../SortedSetDocValuesFacetsFactory.java | 45 + .../SortedSetDocValuesReaderStateHelper.java | 14 + .../earlybird/index/DocIDToTweetIDMapper.java | 79 + .../EarlybirdIndexSegmentAtomicReader.java | 139 + .../index/EarlybirdIndexSegmentData.java | 474 +++ .../index/EarlybirdIndexSegmentWriter.java | 130 + .../index/EarlybirdIndexableField.java | 24 + ...rlybirdLuceneIndexSegmentAtomicReader.java | 336 ++ .../EarlybirdLuceneIndexSegmentData.java | 197 + .../EarlybirdLuceneIndexSegmentWriter.java | 170 + ...ybirdRealtimeIndexSegmentAtomicReader.java | 175 + .../EarlybirdRealtimeIndexSegmentData.java | 251 ++ .../EarlybirdRealtimeIndexSegmentWriter.java | 789 ++++ .../index/QueryCacheResultForSegment.java | 39 + .../index/SequentialDocIDMapper.java | 87 + .../core/earlybird/index/TimeMapper.java | 80 + .../AbstractColumnStrideMultiIntIndex.java | 79 + .../index/column/ColumnStrideByteIndex.java | 88 + .../column/ColumnStrideFieldDocValues.java | 76 + .../index/column/ColumnStrideFieldIndex.java | 64 + .../index/column/ColumnStrideIntIndex.java | 88 + .../column/ColumnStrideIntViewIndex.java | 71 + .../index/column/ColumnStrideLongIndex.java | 88 + .../column/ColumnStrideMultiIntIndex.java | 102 + .../ConstantColumnStrideFieldIndex.java | 18 + .../index/column/DocValuesManager.java | 248 ++ .../index/column/DocValuesUpdate.java | 8 + .../OptimizedColumnStrideByteIndex.java | 81 + .../column/OptimizedColumnStrideIntIndex.java | 81 + .../OptimizedColumnStrideLongIndex.java | 81 + .../OptimizedColumnStrideMultiIntIndex.java | 90 + .../column/OptimizedDocValuesManager.java | 97 + .../column/UnoptimizedDocValuesManager.java | 69 + .../EarlybirdIndexExtensionsData.java | 15 + .../EarlybirdIndexExtensionsFactory.java | 19 + .../EarlybirdRealtimeIndexExtensionsData.java | 20 + .../index/inverted/BaseByteBlockPool.java | 373 ++ .../index/inverted/ByteBlockPool.java | 58 + .../index/inverted/ByteTermUtils.java | 126 + .../earlybird/index/inverted/DeletedDocs.java | 245 ++ .../EarlybirdCSFDocValuesProcessor.java | 74 + .../EarlybirdOptimizedPostingsEnum.java | 178 + .../index/inverted/EarlybirdPostingsEnum.java | 26 + .../index/inverted/FSTTermDictionary.java | 299 ++ .../HighDFPackedIntsDocsAndPositionsEnum.java | 156 + .../inverted/HighDFPackedIntsDocsEnum.java | 222 + .../HighDFPackedIntsPostingLists.java | 829 ++++ .../HighDFPackedIntsSkipListReader.java | 200 + .../index/inverted/InMemoryFields.java | 44 + .../index/inverted/IndexOptimizer.java | 201 + .../index/inverted/IntBlockPool.java | 225 + .../IntBlockPoolPackedLongsReader.java | 253 ++ .../IntBlockPoolPackedLongsWriter.java | 166 + .../index/inverted/InvertedIndex.java | 144 + .../index/inverted/InvertedRealtimeIndex.java | 558 +++ .../inverted/InvertedRealtimeIndexWriter.java | 163 + .../inverted/LowDFPackedIntsPostingLists.java | 255 ++ .../inverted/LowDFPackedIntsPostingsEnum.java | 112 + .../index/inverted/MPHTermDictionary.java | 190 + .../index/inverted/MultiPostingLists.java | 135 + .../inverted/MultiSegmentTermDictionary.java | 60 + ...ultiSegmentTermDictionaryWithFastutil.java | 161 + .../MultiSegmentTermDictionaryWithMap.java | 134 + .../index/inverted/OptimizedIndexTerms.java | 57 + .../index/inverted/OptimizedMemoryIndex.java | 434 ++ .../index/inverted/OptimizedPostingLists.java | 41 + .../OptimizingPostingsEnumWrapper.java | 128 + .../PackedLongsReaderPreComputedValues.java | 202 + .../earlybird/index/inverted/PayloadUtil.java | 91 + .../index/inverted/PostingsBufferQueue.java | 155 + .../index/inverted/QueryCostTracker.java | 48 + .../index/inverted/RealtimeIndexTerms.java | 365 ++ .../index/inverted/SkipListComparator.java | 43 + .../index/inverted/SkipListContainer.java | 739 ++++ .../inverted/SkipListIntegerComparator.java | 26 + .../index/inverted/SkipListPostingList.java | 232 + .../index/inverted/SkipListPostingsEnum.java | 255 ++ .../index/inverted/SkipListSearchFinger.java | 45 + .../index/inverted/TermDictionary.java | 47 + .../index/inverted/TermPointerEncoding.java | 38 + .../earlybird/index/inverted/TermsArray.java | 189 + .../earlybird/index/util/AllDocsIterator.java | 82 + .../core/earlybird/index/util/RangeDISI.java | 50 + .../earlybird/index/util/RangeFilterDISI.java | 58 + .../earlybird/index/util/SearchSortUtils.java | 42 + src/java/com/twitter/search/earlybird/BUILD | 222 + .../com/twitter/search/earlybird/CONFIG.ini | 7 + .../twitter/search/earlybird/Earlybird.java | 267 ++ .../earlybird/EarlybirdCPUQualityFactor.java | 181 + .../search/earlybird/EarlybirdDarkProxy.java | 113 + .../EarlybirdFinagleServerManager.java | 53 + .../earlybird/EarlybirdFuturePoolManager.java | 114 + .../earlybird/EarlybirdIndexConfig.java | 190 + .../search/earlybird/EarlybirdMain.java | 10 + ...rlybirdProductionFinagleServerManager.java | 151 + .../search/earlybird/EarlybirdSearcher.java | 1918 +++++++++ .../search/earlybird/EarlybirdServer.java | 1087 +++++ .../earlybird/EarlybirdServerSetManager.java | 275 ++ .../search/earlybird/EarlybirdStatus.java | 204 + .../earlybird/EarlybirdWarmUpManager.java | 100 + .../search/earlybird/QualityFactor.java | 17 + .../com/twitter/search/earlybird/README.md | 83 + .../RealtimeEarlybirdIndexConfig.java | 128 + .../earlybird/RecentTweetRestriction.java | 60 + .../search/earlybird/ServerSetMember.java | 55 + .../UpdateableEarlybirdStateManager.java | 437 ++ .../archive/ArchiveEarlybirdIndexConfig.java | 75 + .../earlybird/archive/ArchiveHDFSUtils.java | 173 + .../ArchiveOnDiskEarlybirdIndexConfig.java | 79 + .../ArchiveSearchPartitionManager.java | 485 +++ .../earlybird/archive/ArchiveSegment.java | 88 + .../archive/ArchiveSegmentDataProvider.java | 84 + .../archive/ArchiveSegmentUpdater.java | 279 ++ .../archive/ArchiveSegmentVerifier.java | 75 + .../earlybird/archive/ArchiveTimeSlicer.java | 322 ++ .../earlybird/archive/DailyStatusBatch.java | 166 + .../earlybird/archive/DailyStatusBatches.java | 702 +++ .../earlybird/archive/PartitionedBatch.java | 333 ++ .../archive/segmentbuilder/BUILD.bazel | 64 + .../BuiltAndFinalizedSegment.java | 29 + .../segmentbuilder/NotYetBuiltSegment.java | 101 + .../RateLimitingSegmentHandler.java | 39 + .../segmentbuilder/SegmentBuilder.java | 540 +++ .../segmentbuilder/SegmentBuilderApp.java | 109 + .../SegmentBuilderCoordinator.java | 200 + .../segmentbuilder/SegmentBuilderMain.java | 10 + .../segmentbuilder/SegmentBuilderModule.java | 58 + .../segmentbuilder/SegmentBuilderSegment.java | 100 + .../archive/segmentbuilder/SegmentConfig.java | 41 + .../SegmentInfoConstructionException.java | 12 + .../SegmentUpdaterException.java | 13 + .../SomeoneElseIsBuildingSegment.java | 69 + .../com/twitter/search/earlybird/common/BUILD | 37 + .../Base64RequestResponseForLogging.java | 120 + .../earlybird/common/CaughtUpMonitor.java | 55 + .../search/earlybird/common/ClientIdUtil.java | 85 + .../common/EarlybirdRequestLogger.java | 365 ++ .../common/EarlybirdRequestPostLogger.java | 37 + .../common/EarlybirdRequestPreLogger.java | 32 + .../common/EarlybirdRequestUtil.java | 244 ++ .../common/EarlybirdThriftBackend.java | 28 + .../earlybird/common/NonPagingAssert.java | 34 + .../common/RequestResponseForLogging.java | 55 + .../earlybird/common/RequestResponsePair.java | 44 + .../UnknownClientRequestForLogging.java | 77 + .../search/earlybird/common/config/BUILD | 21 + .../common/config/EarlybirdConfig.java | 363 ++ .../common/config/EarlybirdProperty.java | 390 ++ .../search/earlybird/common/userupdates/BUILD | 45 + .../common/userupdates/UserScrubGeoMap.java | 100 + .../common/userupdates/UserTable.java | 572 +++ .../UserTableBuilderFromSnapshot.java | 263 ++ .../common/userupdates/UserUpdate.java | 38 + .../userupdates/UserUpdatesChecker.java | 70 + .../com/twitter/search/earlybird/config/BUILD | 21 + .../search/earlybird/config/ServingRange.java | 26 + .../search/earlybird/config/TierConfig.java | 175 + .../search/earlybird/config/TierInfo.java | 180 + .../earlybird/config/TierInfoSource.java | 39 + .../search/earlybird/config/TierInfoUtil.java | 78 + .../earlybird/config/TierInfoWrapper.java | 89 + .../config/TierServingBoundaryEndPoint.java | 146 + .../earlybird/document/DeletedStatus.java | 15 + .../earlybird/document/DocumentFactory.java | 110 + .../document/ThriftDocumentPreprocessor.java | 170 + .../ThriftIndexingEventDocumentFactory.java | 246 ++ .../ThriftIndexingEventUpdateFactory.java | 91 + .../TimeSlicedThriftIndexingEvent.java | 40 + .../document/TruncationTokenStreamWriter.java | 86 + .../earlybird/document/TweetDocument.java | 52 + .../AlreadyInServerSetUpdateException.java | 12 + .../exception/BadRequestException.java | 11 + .../earlybird/exception/ClientException.java | 11 + .../exception/CriticalExceptionHandler.java | 114 + .../exception/EarlybirdException.java | 18 + .../EarlybirdFinagleServerMonitor.java | 25 + .../exception/EarlybirdRuntimeException.java | 7 + .../exception/EarlybirdStartupException.java | 20 + .../FlushVersionMismatchException.java | 17 + .../exception/MissingKafkaTopicException.java | 11 + .../exception/MissingUserException.java | 4 + .../NotInServerSetUpdateException.java | 12 + .../exception/TransientException.java | 15 + .../exception/UncaughtExceptionHandler.java | 23 + .../exception/WrappedKafkaApiException.java | 19 + .../factory/EarlybirdIndexConfigUtil.java | 53 + .../EarlybirdKafkaConsumersFactory.java | 19 + .../factory/EarlybirdServerFactory.java | 353 ++ .../factory/EarlybirdWireModule.java | 901 ++++ .../factory/PartitionConfigUtil.java | 47 + ...ductionEarlybirdKafkaConsumersFactory.java | 41 + ...yCacheUpdaterScheduledExecutorService.java | 57 + .../index/AbstractInMemoryTimeMapper.java | 83 + .../index/DocValuesBasedTimeMapper.java | 146 + .../index/DocValuesBasedTweetIDMapper.java | 149 + .../earlybird/index/DocValuesHelper.java | 70 + .../earlybird/index/EarlybirdSegment.java | 1070 +++++ .../index/EarlybirdSegmentFactory.java | 58 + .../index/EarlybirdSingleSegmentSearcher.java | 423 ++ .../earlybird/index/OptimizedTimeMapper.java | 109 + .../index/OptimizedTweetIDMapper.java | 145 + .../OutOfOrderRealtimeTweetIDMapper.java | 531 +++ .../earlybird/index/RealtimeTimeMapper.java | 149 + .../earlybird/index/TimeMappingWriter.java | 32 + .../search/earlybird/index/TweetIDMapper.java | 183 + .../search/earlybird/index/TweetIDQuery.java | 81 + .../index/TweetIDToInternalIDMap.java | 154 + .../TweetSearchIndexExtensionsFactory.java | 17 + .../TweetSearchLuceneIndexExtensionsData.java | 41 + ...weetSearchRealtimeIndexExtensionsData.java | 33 + .../search/earlybird/index/facets/BUILD | 16 + .../earlybird/index/facets/FacetSkipList.java | 126 + .../earlybird/ml/ScoringModelsManager.java | 155 + .../AudioSpaceEventsStreamIndexer.java | 75 + .../earlybird/partition/AudioSpaceTable.java | 150 + .../partition/BalancingKafkaConsumer.java | 117 + .../partition/CompleteSegmentManager.java | 349 ++ .../partition/DynamicPartitionConfig.java | 69 + .../earlybird/partition/EarlybirdIndex.java | 65 + .../partition/EarlybirdIndexFlusher.java | 371 ++ .../partition/EarlybirdIndexLoader.java | 224 + .../partition/EarlybirdKafkaConsumer.java | 281 ++ .../earlybird/partition/EarlybirdStartup.java | 17 + .../partition/FlowControlException.java | 16 + .../search/earlybird/partition/HdfsUtil.java | 30 + .../earlybird/partition/ISegmentWriter.java | 24 + .../partition/IndexingResultCounts.java | 51 + .../partition/InstrumentedQueue.java | 51 + .../earlybird/partition/KafkaStartup.java | 328 ++ .../MultiSegmentTermDictionaryManager.java | 314 ++ ...timizationAndFlushingCoordinationLock.java | 46 + .../partition/OptimizingSegmentWriter.java | 210 + .../earlybird/partition/PartitionConfig.java | 171 + .../partition/PartitionConfigLoader.java | 45 + .../PartitionConfigLoadingException.java | 12 + .../earlybird/partition/PartitionManager.java | 254 ++ .../partition/PartitionManagerStartup.java | 57 + .../earlybird/partition/PartitionWriter.java | 109 + .../partition/SearchIndexingMetricSet.java | 208 + .../partition/SegmentHdfsFlusher.java | 247 ++ .../partition/SegmentIndexStats.java | 96 + .../partition/SegmentIndexStatsExporter.java | 85 + .../earlybird/partition/SegmentInfo.java | 428 ++ .../earlybird/partition/SegmentLoader.java | 300 ++ .../earlybird/partition/SegmentManager.java | 822 ++++ .../earlybird/partition/SegmentOptimizer.java | 60 + .../partition/SegmentSyncConfig.java | 218 + .../earlybird/partition/SegmentSyncInfo.java | 113 + .../earlybird/partition/SegmentVulture.java | 380 ++ .../earlybird/partition/SegmentWarmer.java | 49 + .../earlybird/partition/SegmentWriter.java | 239 ++ .../partition/SimpleSegmentIndexer.java | 191 + .../partition/SimpleStreamIndexer.java | 187 + .../partition/SimpleUpdateIndexer.java | 140 + .../partition/StartupUserEventIndexer.java | 236 + .../partition/StatusBatchFlushVersion.java | 41 + .../TimeLimitedHadoopExistsCall.java | 90 + .../partition/TweetCreateHandler.java | 526 +++ .../partition/TweetUpdateHandler.java | 175 + .../partition/UserPartitionUtil.java | 32 + .../UserScrubGeoEventStreamIndexer.java | 88 + .../partition/UserUpdatesStreamIndexer.java | 89 + .../freshstartup/FreshStartupHandler.java | 439 ++ .../freshstartup/KafkaOffsetPair.java | 23 + .../PostOptimizationUpdatesIndexer.java | 169 + .../PreOptimizationSegmentIndexer.java | 459 ++ .../freshstartup/SegmentBuildInfo.java | 92 + .../SegmentTweetsIndexingResult.java | 46 + .../freshstartup/SkippedPickedCounter.java | 26 + .../querycache/CachedFilterQuery.java | 310 ++ .../CachedResultDocIdSetIterator.java | 72 + .../querycache/QueryCacheConfig.java | 101 + .../querycache/QueryCacheConversionRules.java | 100 + .../querycache/QueryCacheFilter.java | 302 ++ .../querycache/QueryCacheManager.java | 365 ++ .../querycache/QueryCacheResultCollector.java | 124 + .../querycache/QueryCacheUpdateTask.java | 283 ++ .../querycache/QueryCacheUpdater.java | 242 ++ .../queryparser/DetectAntisocialVisitor.java | 131 + .../DetectFieldAnnotationVisitor.java | 99 + .../EarlybirdLuceneQueryVisitor.java | 1781 ++++++++ .../queryparser/EarlybirdQueryHelper.java | 154 + .../HighFrequencyTermPairExtractor.java | 211 + .../HighFrequencyTermPairRewriteVisitor.java | 477 +++ .../HighFrequencyTermQueryGroup.java | 94 + .../LuceneRelevanceQueryVisitor.java | 69 + .../ProtectedOperatorQueryRewriter.java | 153 + .../search/AbstractResultsCollector.java | 630 +++ .../earlybird/search/AntiGamingFilter.java | 228 + .../search/EarlybirdLuceneSearcher.java | 98 + .../search/EarlybirdMultiSegmentSearcher.java | 254 ++ .../search/GeoQuadTreeQueryBuilder.java | 199 + .../twitter/search/earlybird/search/Hit.java | 59 + .../earlybird/search/SearchRequestInfo.java | 180 + .../search/SearchResultsCollector.java | 188 + .../earlybird/search/SearchResultsInfo.java | 99 + .../earlybird/search/SimpleSearchResults.java | 35 + .../search/earlybird/search/SocialFilter.java | 98 + .../search/SocialSearchResultsCollector.java | 47 + .../facets/AbstractFacetTermCollector.java | 67 + .../search/facets/DefaultFacetScorer.java | 236 + .../facets/EntityAnnotationCollector.java | 48 + .../search/facets/ExpandedUrlCollector.java | 118 + .../facets/ExplainFacetResultsCollector.java | 159 + .../search/facets/FacetLabelCollector.java | 62 + .../search/facets/FacetRankingModule.java | 26 + .../search/facets/FacetResultsCollector.java | 229 + .../earlybird/search/facets/FacetScorer.java | 24 + .../search/facets/FacetSearchRequestInfo.java | 28 + .../HashingAndPruningFacetAccumulator.java | 492 +++ .../search/facets/NamedEntityCollector.java | 49 + .../facets/RetweetFacetCountIterator.java | 36 + .../facets/SimpleCountRankingModule.java | 32 + .../search/facets/SpaceFacetCollector.java | 47 + .../facets/TermStatisticsCollector.java | 487 +++ .../facets/TermStatisticsRequestInfo.java | 94 + .../TweetSearchFacetCountIteratorFactory.java | 41 + .../search/queries/BadUserRepFilter.java | 115 + .../search/queries/CSFDisjunctionFilter.java | 87 + .../search/queries/DocValRangeFilter.java | 195 + ...FeatureValueInAcceptListOrUnsetFilter.java | 113 + .../search/queries/GeoTwoPhaseQuery.java | 255 ++ .../search/queries/MatchAllDocIdSet.java | 44 + .../search/queries/MatchAllDocsQuery.java | 91 + .../queries/RequiredStatusIDsFilter.java | 131 + .../search/queries/SimpleTermQuery.java | 86 + .../search/queries/SinceMaxIDFilter.java | 211 + .../search/queries/SinceUntilFilter.java | 137 + .../queries/TermQueryWithSafeToString.java | 29 + .../search/queries/TimedDocIdSetIterator.java | 128 + .../queries/UserFlagsExcludeFilter.java | 128 + .../queries/UserIdMultiSegmentQuery.java | 528 +++ .../search/queries/UserScrubGeoFilter.java | 82 + .../search/relevance/LinearScoringData.java | 422 ++ .../search/relevance/LinearScoringParams.java | 304 ++ .../relevance/MinFeatureValueFilter.java | 163 + .../search/relevance/RelevanceHit.java | 104 + .../relevance/RelevanceSearchRequestInfo.java | 66 + .../relevance/RelevanceSearchResults.java | 37 + .../search/relevance/ScoreFilterQuery.java | 138 + .../AbstractRelevanceCollector.java | 147 + .../BatchRelevanceTopCollector.java | 118 + .../collectors/RelevanceAllCollector.java | 70 + .../collectors/RelevanceTopCollector.java | 167 + .../search/relevance/scoring/BatchHit.java | 47 + .../scoring/DefaultScoringFunction.java | 37 + .../scoring/FeatureBasedScoringFunction.java | 1360 ++++++ .../scoring/LegacyScoreAccumulator.java | 98 + .../scoring/LinearScoringFunction.java | 237 ++ .../scoring/ModelBasedScoringFunction.java | 151 + .../relevance/scoring/RelevanceQuery.java | 164 + .../RetweetBasedTopTweetsScoringFunction.java | 165 + .../relevance/scoring/ScoringFunction.java | 213 + .../scoring/ScoringFunctionProvider.java | 216 + .../scoring/SpamVectorScoringFunction.java | 85 + .../relevance/scoring/SparseTensor.java | 87 + .../TensorflowBasedScoringFunction.java | 339 ++ .../scoring/TestScoringFunction.java | 52 + .../segment/DLSegmentDataProvider.java | 62 + .../segment/DLSegmentDataReaderSet.java | 237 ++ .../segment/EmptySegmentDataReaderSet.java | 72 + .../segment/SegmentDataProvider.java | 14 + .../segment/SegmentDataReaderSet.java | 79 + .../earlybird/segment/SegmentProvider.java | 13 + .../earlybird/stats/EarlybirdRPCStats.java | 55 + .../stats/EarlybirdSearcherStats.java | 213 + .../earlybird/stats/SegmentSyncStats.java | 59 + ...arlybirdThriftRequestDeserializerUtil.java | 77 + .../search/earlybird/util/ActionLogger.java | 49 + .../util/CoordinatedEarlybirdAction.java | 409 ++ .../CoordinatedEarlybirdActionInterface.java | 50 + .../CoordinatedEarlybirdActionLockFailed.java | 11 + .../earlybird/util/EarlybirdDecider.java | 128 + .../util/EarlybirdSearchResultUtil.java | 182 + .../earlybird/util/FieldTermCounter.java | 304 ++ .../search/earlybird/util/Histogram.java | 160 + .../search/earlybird/util/IndexViewer.java | 798 ++++ .../earlybird/util/JsonViewerWriter.java | 68 + .../util/OneTaskScheduledExecutorManager.java | 91 + .../search/earlybird/util/ParallelUtil.java | 71 + .../earlybird/util/PeriodicActionParams.java | 79 + .../util/ScheduledExecutorManager.java | 150 + .../earlybird/util/ScheduledExecutorTask.java | 27 + .../search/earlybird/util/ScrubGenUtil.java | 28 + .../util/ShutdownWaitTimeParams.java | 40 + .../earlybird/util/TermCountMonitor.java | 338 ++ .../earlybird/util/TweetCountMonitor.java | 447 ++ .../search/earlybird/util/ViewerWriter.java | 47 + .../com/twitter/search/earlybird_root/BUILD | 75 + .../earlybird_root/ClientBackupFilter.java | 90 + .../earlybird_root/ClientLatencyFilter.java | 45 + .../EarlybirdCacheCommonModule.java | 96 + .../EarlybirdChainedScatterGatherService.java | 58 + .../earlybird_root/EarlybirdCommonModule.java | 170 + ...lybirdFullArchiveScatterGatherSupport.java | 21 + ...arlybirdProtectedScatterGatherSupport.java | 25 + .../EarlybirdProtectedValidationBehavior.java | 45 + .../EarlybirdProtectedWarmup.java | 28 + .../EarlybirdQueryRewriteFilter.java | 157 + ...rlybirdRealtimeCgScatterGatherSupport.java | 21 + ...EarlybirdRealtimeScatterGatherSupport.java | 21 + .../EarlybirdRootQueryUtils.java | 53 + .../EarlybirdServiceChainBuilder.java | 278 ++ .../EarlybirdServiceLoggingSupport.java | 60 + ...rlybirdServicePartitionLoggingSupport.java | 42 + .../EarlybirdServiceScatterGatherSupport.java | 202 + .../EarlybirdServiceValidationBehavior.java | 111 + .../EarlybirdTierThrottleDeciders.java | 49 + .../earlybird_root/EarlybirdWarmup.java | 69 + .../earlybird_root/ExceptionHandler.java | 17 + .../FullArchiveRootAppMain.java | 40 + .../earlybird_root/FullArchiveRootModule.java | 241 ++ .../earlybird_root/FullArchiveRootServer.java | 16 + .../FullArchiveRootService.java | 148 + .../earlybird_root/InitializeFilter.java | 36 + .../MultiTierResultsMergeFilter.java | 55 + .../PartitionAccessController.java | 70 + .../earlybird_root/ProtectedRootAppMain.java | 39 + .../ProtectedRootAppModule.java | 78 + .../earlybird_root/ProtectedRootServer.java | 16 + .../earlybird_root/ProtectedRootService.java | 110 + .../ProtectedScatterGatherModule.java | 62 + .../search/earlybird_root/QuotaModule.java | 110 + .../twitter/search/earlybird_root/README.md | 8 + .../earlybird_root/RealtimeCgRootAppMain.java | 40 + .../RealtimeCgRootAppModule.java | 152 + .../earlybird_root/RealtimeCgRootServer.java | 18 + .../earlybird_root/RealtimeCgRootService.java | 132 + .../RealtimeCgScatterGatherModule.java | 71 + .../earlybird_root/RealtimeRootAppMain.java | 39 + .../earlybird_root/RealtimeRootAppModule.java | 151 + .../earlybird_root/RealtimeRootServer.java | 18 + .../earlybird_root/RealtimeRootService.java | 129 + .../RealtimeScatterGatherModule.java | 118 + .../RootResponseClassifier.java | 69 + .../earlybird_root/ScatterGatherModule.java | 167 + .../earlybird_root/SkipPartitionFilter.java | 70 + .../earlybird_root/SuperRootAppMain.java | 48 + .../earlybird_root/SuperRootAppModule.java | 234 + .../SuperRootRequestTypeRouter.java | 79 + .../earlybird_root/SuperRootServer.java | 36 + .../earlybird_root/SuperRootService.java | 121 + .../search/earlybird_root/caching/BUILD | 20 + .../caching/CacheCommonUtil.java | 16 + .../earlybird_root/caching/CacheStats.java | 13 + .../DefaultForcedCacheMissDecider.java | 24 + .../caching/EarlybirdCachePostProcessor.java | 22 + .../EarlybirdRequestPerClientCacheStats.java | 46 + .../earlybird_root/caching/FacetsCache.java | 15 + .../caching/FacetsCacheFilter.java | 32 + .../caching/FacetsCacheRequestNormalizer.java | 18 + .../caching/FacetsQueryCachePredicate.java | 24 + .../caching/FacetsServicePostProcessor.java | 24 + ...RecencyAndRelevanceCachePostProcessor.java | 66 + .../earlybird_root/caching/RecencyCache.java | 15 + .../caching/RecencyCacheFilter.java | 34 + .../RecencyCacheRequestNormalizer.java | 16 + .../caching/RecencyQueryCachePredicate.java | 24 + .../caching/RecencyServicePostProcessor.java | 27 + .../caching/RelevanceCache.java | 15 + .../caching/RelevanceCacheFilter.java | 33 + .../RelevanceCacheRequestNormalizer.java | 40 + .../caching/RelevanceQueryCachePredicate.java | 24 + .../RelevanceServicePostProcessor.java | 24 + .../RelevanceZeroResultsCacheFilter.java | 40 + ...elevanceZeroResultsCachePostProcessor.java | 20 + ...anceZeroResultsCacheRequestNormalizer.java | 31 + ...levanceZeroResultsQueryCachePredicate.java | 31 + ...evanceZeroResultsServicePostProcessor.java | 36 + .../caching/StrictRecencyCache.java | 15 + .../caching/StrictRecencyCacheFilter.java | 34 + .../StrictRecencyQueryCachePredicate.java | 25 + .../caching/TermStatsCache.java | 15 + .../caching/TermStatsCacheFilter.java | 33 + .../TermStatsCacheRequestNormalizer.java | 17 + .../caching/TermStatsQueryCachePredicate.java | 24 + .../TermStatsServicePostProcessor.java | 25 + .../caching/TopTweetsCache.java | 15 + .../caching/TopTweetsCacheFilter.java | 33 + .../TopTweetsCacheRequestNormalizer.java | 18 + .../caching/TopTweetsQueryCachePredicate.java | 24 + .../TopTweetsServicePostProcessor.java | 41 + .../search/earlybird_root/collectors/BUILD | 12 + .../collectors/MultiwayMergeCollector.java | 82 + .../collectors/RecencyMergeCollector.java | 75 + .../collectors/RelevanceMergeCollector.java | 39 + .../search/earlybird_root/common/BUILD | 22 + .../common/ClientErrorException.java | 24 + .../common/EarlybirdFeatureSchemaMerger.java | 377 ++ .../common/EarlybirdRequestContext.java | 227 + .../common/EarlybirdRequestType.java | 68 + .../common/EarlybirdRequestUtil.java | 107 + .../common/EarlybirdServiceResponse.java | 87 + .../earlybird_root/common/InjectionNames.java | 10 + .../common/QueryParsingUtils.java | 86 + .../common/TwitterContextProvider.java | 20 + .../search/earlybird_root/config/BUILD.bazel | 7 + .../config/RootClusterBoundaryInfo.java | 48 + .../search/earlybird_root/filters/BUILD | 40 + .../filters/ClientIdArchiveAccessFilter.java | 56 + .../ClientIdQueryOperatorStatsFilter.java | 129 + .../filters/ClientIdQuotaFilter.java | 274 ++ .../filters/ClientIdTrackingFilter.java | 148 + .../filters/ClientRequestTimeFilter.java | 34 + .../filters/DeadlineTimeoutStatsFilter.java | 188 + .../filters/DisableClientByTierFilter.java | 64 + .../DropAllProtectedOperatorFilter.java | 71 + .../EarlybirdClusterAvailableFilter.java | 85 + .../EarlybirdFeatureSchemaAnnotateFilter.java | 57 + .../EarlybirdResponseExceptionHandler.java | 108 + .../EarlybirdSuccessfulResponseHandler.java | 54 + .../EarlybirdTimeFilterQueryRewriter.java | 133 + .../filters/EarlybirdTimeRangeFilter.java | 205 + .../FullArchiveProtectedOperatorFilter.java | 167 + .../FullArchiveServingRangeProvider.java | 64 + .../InitializeRequestContextFilter.java | 66 + ...IsUserProtectedMetadataTrackingFilter.java | 80 + .../filters/MarkTweetSourceFilter.java | 49 + .../filters/MetadataTrackingFilter.java | 119 + .../NamedMultiTermDisjunctionStatsFilter.java | 45 + .../filters/NullcastTrackingFilter.java | 81 + .../PostCacheRequestTypeCountFilter.java | 10 + .../PreCacheRequestTypeCountFilter.java | 10 + .../filters/QueryLangStatFilter.java | 114 + .../filters/QueryOperatorStatFilter.java | 194 + .../filters/QueryTokenizerFilter.java | 92 + .../filters/RealtimeServingRangeProvider.java | 60 + .../RejectRequestsByQuerySourceFilter.java | 94 + ...equestContextToEarlybirdRequestFilter.java | 33 + .../filters/RequestResultStatsFilter.java | 185 + .../filters/RequestSuccessStatsFilter.java | 79 + .../filters/RequestTypeCountFilter.java | 105 + .../filters/ResponseCodeStatFilter.java | 50 + .../filters/ResultTierCountFilter.java | 114 + ...rGatherWithExperimentRedirectsService.java | 59 + .../SearchPayloadSizeLocalContextFilter.java | 43 + .../SensitiveResultsTrackingFilter.java | 140 + .../ServiceExceptionHandlingFilter.java | 27 + .../ServiceResponseValidationFilter.java | 81 + .../filters/ServingRangeProvider.java | 12 + .../StratoAttributionClientIdFilter.java | 30 + .../TopLevelExceptionHandlingFilter.java | 24 + .../filters/UnsetSuperRootFieldsFilter.java | 30 + .../filters/VeryRecentTweetsFilter.java | 44 + .../search/earlybird_root/img/serving.png | Bin 0 -> 60951 bytes .../mergers/AccumulatedResponses.java | 176 + .../search/earlybird_root/mergers/BUILD | 26 + .../EarlyTerminateTierMergePredicate.java | 9 + .../EarlybirdResponseDebugMessageBuilder.java | 176 + .../mergers/EarlybirdResponseMerger.java | 604 +++ .../mergers/FacetResponseMerger.java | 353 ++ .../mergers/PartitionResponseAccumulator.java | 44 + .../mergers/RecencyResponseMerger.java | 638 +++ .../mergers/RelevanceResponseMerger.java | 268 ++ .../mergers/ResponseAccumulator.java | 356 ++ .../mergers/StrictRecencyResponseMerger.java | 297 ++ .../mergers/SuperRootResponseMerger.java | 688 +++ .../mergers/TermStatisticsResponseMerger.java | 90 + .../mergers/ThriftTermResultsMerger.java | 472 ++ .../mergers/TierResponseAccumulator.java | 97 + .../mergers/TopTweetsResponseMerger.java | 65 + .../earlybird_root/mergers/TrimStats.java | 71 + .../twitter/search/earlybird_root/quota/BUILD | 15 + .../quota/ClientIdQuotaManager.java | 22 + .../quota/ConfigBasedQuotaConfig.java | 161 + .../quota/ConfigRepoBasedQuotaManager.java | 65 + .../earlybird_root/quota/QuotaInfo.java | 78 + ...tractRecencyAndRelevanceRequestRouter.java | 442 ++ .../search/earlybird_root/routers/BUILD | 25 + .../routers/FacetsRequestRouter.java | 35 + .../routers/FacetsRequestRouterModule.java | 33 + .../routers/RecencyRequestRouter.java | 73 + .../routers/RecencyRequestRouterModule.java | 74 + .../routers/RelevanceRequestRouter.java | 100 + .../routers/RelevanceRequestRouterModule.java | 74 + .../earlybird_root/routers/RequestRouter.java | 144 + .../routers/RequestRouterUtil.java | 107 + .../routers/TermStatsRequestRouter.java | 238 ++ .../routers/TermStatsRequestRouterModule.java | 60 + .../routers/TopTweetsRequestRouter.java | 35 + .../routers/TopTweetsRequestRouterModule.java | 32 + .../search/earlybird_root/validators/BUILD | 9 + .../validators/FacetsResponseValidator.java | 39 + .../PassThroughResponseValidator.java | 12 + .../validators/SearchResultsValidator.java | 37 + .../validators/ServiceResponseValidator.java | 10 + .../validators/TermStatsResultsValidator.java | 23 + .../validators/TopTweetsResultsValidator.java | 22 + .../search/earlybird_root/visitors/BUILD | 13 + ...ltiTermDisjunctionPerPartitionVisitor.java | 136 + .../search/feature_update_service/BUILD | 86 + .../FeatureUpdateController.java | 245 ++ .../FeatureUpdateResponseClassifier.java | 43 + .../FeatureUpdateServiceThriftServer.java | 149 + .../FeatureUpdateServiceThriftServerMain.java | 12 + .../search/feature_update_service/README.md | 6 + .../feature_update_service/filters/BUILD | 22 + .../filters/ClientIdWhitelistFilter.java | 60 + .../feature_update_service/modules/BUILD | 48 + .../modules/ClientIdWhitelistModule.java | 30 + .../modules/EarlybirdUtilModule.java | 13 + .../FeatureUpdateServiceDiffyModule.java | 35 + .../modules/FinagleKafkaProducerModule.java | 62 + .../modules/FuturePoolModule.java | 57 + .../modules/TweetypieModule.java | 62 + .../search/feature_update_service/stats/BUILD | 11 + .../stats/FeatureUpdateStats.java | 111 + .../search/feature_update_service/util/BUILD | 11 + .../util/FeatureUpdateValidator.java | 41 + .../feature_update_service/whitelist/BUILD | 13 + .../whitelist/ClientIdWhitelist.java | 77 + src/java/com/twitter/search/img/foryou.png | Bin 0 -> 91995 bytes .../com/twitter/search/img/in-network.png | Bin 0 -> 33193 bytes src/java/com/twitter/search/img/indexing.png | Bin 0 -> 38051 bytes src/java/com/twitter/search/img/serving.png | Bin 0 -> 60951 bytes .../com/twitter/search/img/top-search.png | Bin 0 -> 151902 bytes src/java/com/twitter/search/ingester/BUILD | 30 + .../com/twitter/search/ingester/README.md | 10 + .../com/twitter/search/ingester/model/BUILD | 28 + .../search/ingester/model/IndexerStatus.java | 13 + .../model/IngesterThriftVersionedEvents.java | 50 + .../ingester/model/IngesterTweetEvent.java | 19 + .../model/IngesterTwitterMessage.java | 73 + .../search/ingester/model/KafkaRawRecord.java | 22 + .../ingester/model/PromiseContainer.java | 21 + .../ingester/model/VisibleTokenRatioUtil.java | 42 + .../search/ingester/pipeline/app/BUILD | 31 + .../app/IngesterPipelineApplication.java | 195 + .../pipeline/app/PipelineExceptionImpl.java | 30 + .../pipeline/app/PipelineExceptionImplV2.java | 29 + .../app/RealtimeIngesterPipelineV2.java | 111 + .../AudioSpaceCoreFetcher.java | 56 + .../AudioSpaceParticipantsFetcher.java | 36 + .../ingester/pipeline/strato_fetchers/BUILD | 20 + .../strato_fetchers/NamedEntityFetcher.java | 47 + .../twitter/AsyncPinkUrlsResolver.java | 67 + .../search/ingester/pipeline/twitter/BUILD | 74 + .../CollectComparableObjectsStage.java | 176 + .../twitter/ComputeTweetSignatureStage.java | 38 + .../ConvertDelayedMessageToThriftStage.java | 95 + .../twitter/ConvertMessageToThriftStage.java | 117 + .../ConvertToThriftVersionedEventsStage.java | 83 + .../pipeline/twitter/EventBusReaderStage.java | 185 + .../pipeline/twitter/FieldStatExporter.java | 150 + .../FilterEventsBySafetyTypeStage.java | 279 ++ .../FilterRetweetsAndRepliesStage.java | 79 + .../twitter/FilterTwitterMessageStage.java | 77 + .../LookupUserPropertiesBatchedStage.java | 60 + .../pipeline/twitter/NamedEntityHandler.java | 101 + .../PopulateCodedLocationsBatchedStage.java | 79 + .../ResolveCompressedUrlsBatchedStage.java | 387 ++ .../twitter/ResolveCompressedUrlsPink.java | 113 + .../twitter/ResolveCompressedUrlsUtils.java | 116 + .../twitter/RetrieveCardBatchedStage.java | 288 ++ ...RetrieveNamedEntitiesSingleTweetStage.java | 75 + .../RetrieveSpaceAdminsAndTitleStage.java | 246 ++ .../twitter/RetrieveSpaceIdsStage.java | 99 + ...ngleTweetExtractAndGeocodeLatLonStage.java | 99 + .../TextFeatureExtractionWorkersStage.java | 148 + .../TextQualityEvaluationWorkerStage.java | 181 + .../TextUrlsFeatureExtractionStage.java | 53 + .../twitter/ThriftTweetParserStage.java | 178 + .../ThriftVersionedEventsConverter.java | 132 + .../twitter/TweetEventDeserializerStage.java | 137 + .../pipeline/twitter/TwitterBaseStage.java | 360 ++ .../twitter/TwitterBatchedBaseStage.java | 309 ++ .../ingester/pipeline/twitter/filters/BUILD | 13 + .../filters/IngesterValidMessageFilter.java | 50 + .../ingester/pipeline/twitter/kafka/BUILD | 32 + .../DeleteUpdateEventsKafkaProducerStage.java | 66 + .../twitter/kafka/KafkaConsumerStage.java | 245 ++ .../twitter/kafka/KafkaProducerStage.java | 259 ++ .../kafka/KafkaRawRecordConsumerStage.java | 36 + ...ndReplyUpdateEventsKafkaProducerStage.java | 24 + ...riftVersionedEventsKafkaProducerStage.java | 108 + .../pipeline/twitter/thriftparse/BUILD | 32 + .../ThriftTweetParsingException.java | 7 + .../thriftparse/TweetEventParseHelper.java | 727 ++++ .../pipeline/twitter/userupdates/BUILD | 34 + .../userupdates/UserUpdateIngester.java | 292 ++ .../userupdates/UserUpdatesPipeline.java | 222 + .../userupdates/UserUpdatesPipelineStage.java | 51 + .../search/ingester/pipeline/util/BUILD | 41 + .../pipeline/util/BatchedElement.java | 21 + .../pipeline/util/BatchingClient.java | 105 + .../ingester/pipeline/util/CardFieldUtil.java | 48 + .../pipeline/util/IngesterStageTimer.java | 35 + .../util/ManhattanCodedLocationProvider.java | 110 + .../pipeline/util/PenguinVersionsUtil.java | 47 + .../util/PipelineExceptionHandler.java | 15 + .../pipeline/util/PipelineStageException.java | 19 + .../util/PipelineStageRuntimeException.java | 7 + .../ingester/pipeline/util/PipelineUtil.java | 26 + .../util/PipelineV2CreationException.java | 7 + .../util/ResponseNotReturnedException.java | 7 + .../pipeline/util/UserPropertiesManager.java | 446 ++ .../search/ingester/pipeline/wire/BUILD | 52 + .../pipeline/wire/IngesterPartitioner.java | 27 + .../pipeline/wire/ProductionWireModule.java | 363 ++ .../wire/StratoMetaStoreWireModule.java | 119 + .../pipeline/wire/TweetyPieWireModule.java | 110 + .../ingester/pipeline/wire/WireModule.java | 226 + .../twitter/search/ingester/util/jndi/BUILD | 9 + .../search/ingester/util/jndi/JndiUtil.java | 70 + .../configs/recap_earlybird/feature_config.py | 78 + .../rectweet_earlybird/feature_config.py | 74 + .../timelines/scripts/models/earlybird/BUILD | 23 + .../scripts/models/earlybird/README.md | 63 + .../scripts/models/earlybird/__init__.py | 0 .../scripts/models/earlybird/constants.py | 21 + .../models/earlybird/earlybird_features.png | Bin 0 -> 360811 bytes .../models/earlybird/example_weights.py | 43 + .../scripts/models/earlybird/lolly/BUILD | 18 + .../models/earlybird/lolly/__init__.py | 0 .../models/earlybird/lolly/data_helpers.py | 23 + .../scripts/models/earlybird/lolly/parsers.py | 145 + .../scripts/models/earlybird/lolly/reader.py | 8 + .../scripts/models/earlybird/lolly/score.py | 13 + .../scripts/models/earlybird/lolly/scorer.py | 37 + .../lolly/tf_model_initializer_builder.py | 91 + .../scripts/models/earlybird/metrics.py | 120 + .../scripts/models/earlybird/tf_model/BUILD | 8 + .../models/earlybird/tf_model/__init__.py | 0 .../earlybird/tf_model/discretizer_builder.py | 62 + .../earlybird/tf_model/hashing_utils.py | 29 + .../tf_model/weights_initializer_builder.py | 34 + .../scripts/models/earlybird/train.py | 212 + src/scala/com/twitter/graph/batch/BUILD.bazel | 91 + .../job/tweepcred/ExtractTweepcred.scala | 83 + .../job/tweepcred/PreparePageRankData.scala | 275 ++ .../twitter/graph/batch/job/tweepcred/README | 73 + .../batch/job/tweepcred/Reputation.scala | 50 + .../job/tweepcred/TweepcredBatchJob.scala | 64 + .../graph/batch/job/tweepcred/UserMass.scala | 69 + .../job/tweepcred/WeightedPageRank.scala | 235 + .../com/twitter/interaction_graph/README.md | 19 + .../interaction_graph/bqe/scoring/README.md | 58 + .../bqe/scoring/candidates.sql | 42 + .../bqe/scoring/check_models.sql | 5 + .../bqe/scoring/follow_graph_features.sql | 28 + .../interaction_graph/bqe/scoring/scoring.sql | 52 + .../interaction_graph/bqe/training/README.md | 60 + .../bqe/training/candidates.sql | 18 + .../bqe/training/check_candidates_exist.sql | 43 + .../bqe/training/check_labels_exist.sql | 4 + .../bqe/training/labeled_candidates.sql | 67 + .../bqe/training/train_model.sql | 27 + .../twitter/interaction_graph/injection/BUILD | 25 + .../injection/EdgeListInjection.scala | 14 + .../injection/UserSessionInjection.scala | 14 + .../twitter/interaction_graph/scio/README.md | 7 + .../scio/agg_address_book/BUILD | 62 + .../InteractionGraphAddressBookCounters.scala | 34 + .../InteractionGraphAddressBookJob.scala | 71 + .../InteractionGraphAddressBookOption.scala | 24 + .../InteractionGraphAddressBookSource.scala | 28 + .../InteractionGraphAddressBookUtil.scala | 93 + .../scio/agg_address_book/README.md | 34 + .../interaction_graph/scio/agg_all/BUILD | 175 + .../InteractionGraphAggregationConfig.scala | 14 + .../InteractionGraphAggregationJob.scala | 314 ++ .../InteractionGraphAggregationOption.scala | 36 + .../InteractionGraphAggregationSource.scala | 182 + ...InteractionGraphAggregationTransform.scala | 59 + .../interaction_graph/scio/agg_all/README.md | 38 + .../scio/agg_client_event_logs/BUILD | 61 + ...eractionGraphClientEventLogsCounters.scala | 32 + .../InteractionGraphClientEventLogsJob.scala | 74 + ...nteractionGraphClientEventLogsOption.scala | 24 + ...nteractionGraphClientEventLogsSource.scala | 40 + .../InteractionGraphClientEventLogsUtil.scala | 137 + .../scio/agg_client_event_logs/README.md | 34 + .../scio/agg_direct_interactions/BUILD | 65 + ...ractionGraphAggDirectInteractionsJob.scala | 79 + ...tionGraphAggDirectInteractionsOption.scala | 24 + ...tionGraphAggDirectInteractionsSource.scala | 51 + ...actionGraphAggDirectInteractionsUtil.scala | 168 + .../scio/agg_direct_interactions/README.md | 34 + .../interaction_graph/scio/agg_flock/BUILD | 70 + .../InteractionGraphAggFlockJob.scala | 84 + .../InteractionGraphAggFlockOption.scala | 24 + .../InteractionGraphAggFlockSource.scala | 24 + .../InteractionGraphAggFlockUtil.scala | 63 + .../scio/agg_flock/README.md | 34 + .../interaction_graph/scio/agg_negative/BUILD | 43 + .../InteractionGraphNegativeJob.scala | 155 + .../InteractionGraphNegativeOption.scala | 18 + .../scio/agg_negative/README.md | 35 + .../scio/agg_notifications/BUILD | 65 + .../InteractionGraphNotificationUtil.scala | 132 + .../InteractionGraphNotificationsJob.scala | 86 + .../InteractionGraphNotificationsOption.scala | 24 + .../scio/agg_notifications/README.md | 34 + .../interaction_graph/scio/common/BUILD | 31 + .../scio/common/CaseClasses.scala | 21 + .../scio/common/ConversionUtil.scala | 110 + .../scio/common/DateUtil.scala | 27 + .../scio/common/EdgeFeatureCombiner.scala | 350 ++ .../scio/common/FeatureGeneratorUtil.scala | 263 ++ .../scio/common/FeatureGroups.scala | 30 + .../scio/common/GraphUtil.scala | 93 + .../scio/common/InteractionGraphUtils.scala | 40 + .../scio/common/UserUtil.scala | 76 + .../scio/common/VertexFeatureCombiner.scala | 342 ++ .../interaction_graph/scio/ml/labels/BUILD | 49 + .../ml/labels/InteractionGraphLabelsJob.scala | 123 + .../labels/InteractionGraphLabelsOption.scala | 28 + .../scio/ml/labels/LabelUtil.scala | 63 + .../scio/ml/labels/README.md | 34 + .../interaction_graph/scio/ml/scores/BUILD | 54 + .../InteractionGraphScoreExportJob.scala | 134 + .../InteractionGraphScoreExportOption.scala | 24 + .../scio/ml/scores/README.md | 34 + src/scala/com/twitter/recos/decider/BUILD | 9 + .../twitter/recos/decider/BaseDecider.scala | 110 + .../recos/decider/EndpointLoadShedder.scala | 39 + .../graph_common/ActionEdgeTypeMask.scala | 99 + .../com/twitter/recos/graph_common/BUILD | 12 + .../graph_common/BipartiteGraphHelper.scala | 40 + .../graph_common/FinagleCounterWrapper.scala | 15 + .../FinagleStatsReceiverWrapper.scala | 16 + ...LawMultiSegmentBipartiteGraphBuilder.scala | 59 + ...SegmentPowerLawBipartiteGraphBuilder.scala | 64 + .../recos/graph_common/NodeInfoHandler.scala | 59 + ...LawMultiSegmentBipartiteGraphBuilder.scala | 63 + ...LawMultiSegmentBipartiteGraphBuilder.scala | 63 + src/scala/com/twitter/recos/hose/common/BUILD | 15 + .../hose/common/BufferedEdgeWriter.scala | 48 + .../recos/hose/common/EdgeCollector.scala | 42 + .../hose/common/RecosEdgeProcessor.scala | 41 + .../hose/common/UnifiedGraphWriter.scala | 217 + .../hose/common/UnifiedGraphWriterMulti.scala | 228 + .../recos/user_tweet_entity_graph/BUILD | 67 + .../EntitySocialProofRunner.scala | 167 + .../LoggingUserTweetEntityGraph.scala | 103 + .../recos/user_tweet_entity_graph/Main.scala | 258 ++ .../recos/user_tweet_entity_graph/README.md | 17 + .../RecommendationHandler.scala | 78 + .../user_tweet_entity_graph/RecosConfig.scala | 44 + .../SocialProofHandler.scala | 165 + .../SocialProofHydrator.scala | 111 + .../TweetRecommendationsRunner.scala | 322 ++ .../TweetSocialProofHandler.scala | 73 + .../TweetSocialProofRunner.scala | 168 + .../UserTweetEdgeTypeMask.scala | 95 + .../UserTweetEntityGraph.scala | 46 + .../UserTweetEntityGraphWriter.scala | 105 + .../com/twitter/recos/user_tweet_graph/BUILD | 66 + .../twitter/recos/user_tweet_graph/Main.scala | 291 ++ .../twitter/recos/user_tweet_graph/README.md | 17 + .../user_tweet_graph/UserTweetGraph.scala | 98 + .../UserTweetGraphConfig.scala | 39 + .../UserTweetGraphWriter.scala | 88 + .../relatedTweetHandlers/BUILD | 12 + .../ConsumersBasedRelatedTweetsHandler.scala | 68 + .../ProducerBasedRelatedTweetsHandler.scala | 88 + .../TweetBasedRelatedTweetsHandler.scala | 93 + .../recos/user_tweet_graph/store/BUILD | 9 + .../store/UserRecentFollowersStore.scala | 50 + .../twitter/recos/user_tweet_graph/util/BUILD | 12 + .../util/FetchRHSTweetsUtil.scala | 34 + .../user_tweet_graph/util/FilterUtil.scala | 15 + .../util/GetAllInternalTweetIdsUtil.scala | 33 + .../util/GetRelatedTweetCandidatesUtil.scala | 56 + .../util/SampleLHSUsersUtil.scala | 35 + .../util/UserTweetEdgeTypeMask.scala | 77 + .../com/twitter/recos/user_user_graph/BUILD | 45 + .../recos/user_user_graph/KafkaConfig.scala | 13 + .../LoggingUserUserGraph.scala | 51 + .../twitter/recos/user_user_graph/Main.scala | 255 ++ .../twitter/recos/user_user_graph/README.md | 17 + .../RecommendUsersHandler.scala | 221 + .../recos/user_user_graph/RecosConfig.scala | 37 + .../user_user_graph/UserEdgeTypeMask.scala | 91 + .../recos/user_user_graph/UserUserGraph.scala | 18 + .../user_user_graph/UserUserGraphWriter.scala | 83 + .../com/twitter/recos/user_video_graph/BUILD | 69 + .../LoggingUserVideoGraph.scala | 12 + .../twitter/recos/user_video_graph/Main.scala | 294 ++ .../twitter/recos/user_video_graph/README.md | 14 + .../UserVideoEdgeTypeMask.scala | 62 + .../user_video_graph/UserVideoGraph.scala | 73 + .../UserVideoGraphConfig.scala | 39 + .../UserVideoGraphEdgeHttpHandler.scala | 101 + .../UserVideoGraphWriter.scala | 82 + .../relatedTweetHandlers/BUILD | 12 + .../ConsumersBasedRelatedTweetsHandler.scala | 66 + .../ProducerBasedRelatedTweetsHandler.scala | 86 + .../TweetBasedRelatedTweetsHandler.scala | 91 + .../recos/user_video_graph/store/BUILD | 9 + .../store/UserRecentFollowersStore.scala | 50 + .../twitter/recos/user_video_graph/util/BUILD | 12 + .../util/FetchRHSTweetsUtil.scala | 29 + .../user_video_graph/util/FilterUtil.scala | 15 + .../util/GetAllInternalTweetIdsUtil.scala | 33 + .../util/GetRelatedTweetCandidatesUtil.scala | 56 + .../util/SampleLHSUsersUtil.scala | 35 + .../com/twitter/simclusters_v2/README.md | 112 + .../simclusters_v2/candidate_source/BUILD | 17 + .../candidate_source/ClusterRanker.scala | 56 + .../candidate_source/HeavyRanker.scala | 71 + .../SimClustersANNCandidateSource.scala | 637 +++ ...SimClustersANNWrapperCandidateSource.scala | 53 + .../com/twitter/simclusters_v2/common/BUILD | 12 + .../common/CosineSimilarityUtil.scala | 251 ++ .../DeciderGateBuilderWithIdHashing.scala | 21 + .../simclusters_v2/common/ModelVersions.scala | 48 + .../common/SeqStandardDeviation.scala | 22 + .../common/SimClustersEmbedding.scala | 581 +++ .../common/SimClustersEmbeddingId.scala | 209 + ...imClustersEmbeddingIdCacheKeyBuilder.scala | 19 + .../common/SimClustersEmbeddingMonoid.scala | 18 + .../common/SimClustersMultiEmbedding.scala | 32 + .../common/SimClustersMultiEmbeddingId.scala | 96 + .../simclusters_v2/common/clustering/BUILD | 11 + ...ClusterRepresentativeSelectionMethod.scala | 30 + .../common/clustering/ClusteringMethod.scala | 34 + .../ConnectedComponentsClusteringMethod.scala | 67 + .../LargestDimensionClusteringMethod.scala | 33 + .../clustering/LouvainClusteringMethod.scala | 236 + ...avScoreRepresentativeSelectionMethod.scala | 21 + .../MedoidRepresentativeSelectionMethod.scala | 28 + .../clustering/SimilarityFunctions.scala | 32 + .../twitter/simclusters_v2/common/ml/BUILD | 12 + .../ml/SimClustersEmbeddingAdapter.scala | 39 + .../simclusters_v2/common/package.scala | 17 + .../hdfs_sources/AdhocSources.scala | 164 + .../twitter/simclusters_v2/hdfs_sources/BUILD | 2216 ++++++++++ .../hdfs_sources/DataPaths.scala | 49 + .../hdfs_sources/DataSources.scala | 39 + .../EntityEmbeddingsSources.scala | 222 + .../hdfs_sources/InterestedInSources.scala | 178 + .../ProducerEmbeddingSources.scala | 86 + .../hdfs_sources/injections/BUILD | 13 + .../injections/ClusterDetailsInjection.scala | 16 + .../ClusterTopMediaTweetsInjection.scala | 13 + .../ClusterTopTweetsInjection.scala | 14 + .../injections/ClusteringInjections.scala | 16 + .../EntityEmbeddingsInjections.scala | 47 + .../InferredEntitiesInjections.scala | 27 + .../injections/InterestedInInjection.scala | 13 + .../injections/KnownForInjection.scala | 12 + .../injections/MultiTypeGraphInjections.scala | 31 + .../ProducerEmbeddingsInjections.scala | 45 + .../SemanticCoreEntitiesInjections.scala | 53 + .../SingleSideUserScoresInjection.scala | 12 + .../hdfs_sources/presto_hdfs_sources/BUILD | 60 + .../EntityEmbeddingsPrestoSources.scala | 10 + .../simclusters_v2/images/bipartite_graph.png | Bin 0 -> 62580 bytes .../simclusters_v2/images/interestedin.png | Bin 0 -> 67315 bytes .../simclusters_v2/images/knownfor.png | Bin 0 -> 26378 bytes .../images/producer_embeddings.png | Bin 0 -> 72820 bytes .../images/producer_producer_similarity.png | Bin 0 -> 238661 bytes .../images/topic_embeddings.png | Bin 0 -> 71151 bytes .../com/twitter/simclusters_v2/scalding/BUILD | 521 +++ .../scalding/BipartiteClusterEvaluation.scala | 513 +++ .../BipartiteClusterEvaluationClasses.scala | 316 ++ .../scalding/ClusterDetailsJob.scala | 794 ++++ .../scalding/ClusterEvaluation.scala | 607 +++ .../scalding/CompareClusters.scala | 131 + .../EigenVectorsForSparseSymmetric.scala | 330 ++ ...InFromAggregatableProducerEmbeddings.scala | 332 ++ .../scalding/InterestedInFromKnownFor.scala | 666 +++ .../InterestedInFromKnownForLite.scala | 354 ++ ...stedInFromProducerEmbeddingsAdhocApp.scala | 290 ++ .../scalding/KnownForSources.scala | 275 ++ .../scalding/ProducerNormsAndCounts.scala | 195 + .../scalding/TopUsersSimilarityGraph.scala | 996 +++++ .../scalding/UpdateKnownFor.scala | 311 ++ .../scalding/UpdateKnownForApps.scala | 443 ++ .../scalding/UserUserFavGraph.scala | 445 ++ .../scalding/UserUserGraph.scala | 180 + .../scalding/UserUserNormalizedGraph.scala | 453 ++ .../simclusters_v2/scalding/common/BUILD | 14 + .../PersistentTweetEmbeddingSource.scala | 60 + .../common/QTreeMultiAggregator.scala | 30 + .../scalding/common/TypedRichPipe.scala | 72 + .../simclusters_v2/scalding/common/Util.scala | 305 ++ .../scalding/common/matrix/BUILD | 8 + .../common/matrix/DenseRowMatrix.scala | 73 + .../scalding/common/matrix/SparseMatrix.scala | 423 ++ .../common/matrix/SparseRowMatrix.scala | 366 ++ .../common/matrix/TypedPipeMatrix.scala | 49 + .../simclusters_v2/scalding/embedding/BUILD | 311 ++ ...ityEmbeddingFromProducerEmbeddingJob.scala | 239 ++ .../EntityToSimClustersEmbeddingsJob.scala | 354 ++ .../GlobalSimClustersLanguageEmbedding.scala | 197 + ...ocaleEntitySimClustersEmbeddingV2Job.scala | 248 ++ ...LocaleEntitySimClustersEmbeddingsJob.scala | 437 ++ .../ProducerEmbeddingsFromInterestedIn.scala | 701 +++ .../SimilarUsersBySimClustersEmbedding.scala | 299 ++ .../AbuseSimclusterFeaturesScaldingJob.scala | 178 + ...ocAbuseSimClusterFeaturesScaldingJob.scala | 217 + .../scalding/embedding/abuse/BUILD | 74 + .../CrossSimClusterFeaturesScaldingJob.scala | 149 + .../embedding/abuse/DataSources.scala | 101 + .../abuse/PairedinteractionFeatures.scala | 122 + .../SingleSideInteractionTransformation.scala | 154 + .../embedding/common/EmbeddingUtil.scala | 114 + .../common/EntityEmbeddingUtil.scala | 79 + .../common/ExternalDataSources.scala | 565 +++ .../common/SimClustersEmbeddingJob.scala | 248 ++ ...gregatableFavBasedProducerEmbeddings.scala | 278 ++ ...gatableFollowBasedProducerEmbeddings.scala | 165 + ...gatableLogFavBasedProducerEmbeddings.scala | 368 ++ .../AggregatableProducerEmbeddings.scala | 168 + .../scalding/embedding/producer/BUILD.bazel | 223 + .../scalding/embedding/tfg/BUILD | 196 + ...ntWeightedTfgBasedTopicEmbeddingsJob.scala | 310 ++ ...erredLanguageTfgBasedTopicEmbeddings.scala | 66 + .../tfg/FavTfgBasedTopicEmbeddings.scala | 172 + ...nguageTfgBasedTopicEmbeddingsBaseApp.scala | 194 + .../tfg/LogFavTfgBasedTopicEmbeddings.scala | 70 + .../scalding/embedding/tfg/README | 7 + .../tfg/TfgBasedTopicEmbeddingsBaseApp.scala | 191 + .../scalding/embedding/twice/BUILD.bazel | 166 + .../embedding/twice/InterestedInTwice.scala | 454 ++ .../twice/InterestedInTwiceBaseApp.scala | 495 +++ .../scalding/evaluation/BUILD.bazel | 72 + .../evaluation/CandidateEvaluationBase.scala | 163 + .../evaluation/EvaluationMetricHelper.scala | 540 +++ .../EvaluationReferenceDataExtraction.scala | 270 ++ .../evaluation/LabelCorrelationsHelper.scala | 61 + .../SimClustersEvaluationAdhocApp.scala | 210 + .../scalding/inferred_entities/BUILD.bazel | 74 + .../inferred_entities/InferredEntities.scala | 92 + .../InferredEntitiesFromInterestedIn.scala | 377 ++ ...rredSemanticCoreEntitiesFromKnownFor.scala | 244 ++ .../inferred_entities/ProdSources.scala | 94 + .../scalding/mbcg/AllFeatures.scala | 58 + .../simclusters_v2/scalding/mbcg/BUILD.bazel | 314 ++ .../scalding/mbcg/RecordAdapters.scala | 79 + .../mbcg/TweetEmbeddingGenerationJob.scala | 384 ++ .../mbcg/UserEmbeddingGenerationJob.scala | 270 ++ .../AssembleMultiTypeGraph.scala | 514 +++ .../AssembleMultiTypeGraphApp.scala | 74 + .../AssembleMultiTypeGraphBaseApp.scala | 185 + .../assemble_multi_type_graph/BUILD | 91 + .../assemble_multi_type_graph/Config.scala | 35 + .../scalding/offline_job/BUILD.bazel | 126 + .../OfflineTweetRecommendation.scala | 176 + .../offline_job/SimClustersOfflineJob.scala | 176 + .../SimClustersOfflineJobAdhocApp.scala | 197 + .../SimClustersOfflineJobScheduledApp.scala | 113 + .../SimClustersOfflineJobUtil.scala | 97 + .../scalding/offline_job/adhoc/BUILD.bazel | 81 + .../scalding/offline_job/adhoc/README | 5 + .../SimClustersTweetEmbeddingAdhocApp.scala | 211 + .../TweetSimilarityEvaluationAdhocApp.scala | 362 ++ .../scalding/offline_tweets/BUILD.bazel | 27 + .../ClusterTopMediaTweetsJob.scala | 267 ++ .../scalding/optout/BUILD.bazel | 81 + .../scalding/optout/InterestedInOptOut.scala | 269 ++ .../scalding/optout/KnownForOptOut.scala | 198 + .../optout/SimClustersOptOutUtil.scala | 166 + .../scalding/topic_recommendations/BUILD | 168 + .../GeoPopularTopicsApp.scala | 165 + ...oducersForTopicsFromTopicFollowGraph.scala | 206 + ...SimilarTopicsFromTopicFollowGraphApp.scala | 222 + .../TopicsForProducersFromEM.scala | 261 ++ .../TopicsForProducersUtils.scala | 103 + .../model_based_topic_recommendations/BUILD | 70 + .../DataSources.scala | 74 + .../UserFeatures.scala | 57 + .../UserTopicDataRecordAdapter.scala | 64 + ...icModellingTrainingDataCollectionJob.scala | 449 ++ .../scalding/tweet_similarity/BUILD | 234 + .../DatasetTopKAnalysisJob.scala | 255 ++ .../TrainingDataCollectionJob.scala | 228 + .../TrainingDataCollectionUtil.scala | 138 + .../TweetPairFeatureHydrationUtil.scala | 289 ++ .../TweetPairLabelCollectionUtil.scala | 490 +++ .../UnhydratedPairsCollectionJob.scala | 209 + .../tweet_similarity/evaluation/BUILD.bazel | 40 + .../evaluation/ModelEvalAdhocApp.scala | 91 + .../RUXLandingDdgAnalysisAdhocApp.scala | 82 + .../scalding/update_known_for/BUILD.bazel | 59 + .../UpdateKnownFor20M145K2020.scala | 256 ++ .../UpdateKnownForSBFRunner.scala | 685 +++ .../common/BQGenerationUtil.scala | 255 ++ .../scio/bq_generation/common/BUILD | 10 + .../common/IndexGenerationUtil.scala | 63 + .../scio/bq_generation/ftr_tweet/BUILD | 250 ++ .../scio/bq_generation/ftr_tweet/Config.scala | 43 + .../scio/bq_generation/ftr_tweet/FTRJob.scala | 242 ++ .../FtrClusterToTweetIndexGenerationJob.scala | 264 ++ .../scio/bq_generation/ftr_tweet/README.md | 212 + ...based-simclusters-index-generation-job.d6w | 44 + .../ftr_tweet/ftr-tweets-ann-adhoc-job.d6w | 36 + .../iikf2020-decayed-sum-ann-batch-job.d6w | 35 + .../iikf2020-ftrat5-pop1000-ann-batch-job.d6w | 35 + ...iikf2020-ftrat5-pop10000-ann-batch-job.d6w | 35 + .../scio/bq_generation/ftr_tweet/sql/BUILD | 3 + .../ftr_tweet/sql/ftr_tweet_embeddings.sql | 280 ++ .../simclusters_index_generation/BUILD | 167 + .../simclusters_index_generation/Config.scala | 82 + ...tEventBasedClusterToTweetIndexFromBQ.scala | 177 + ...asedClusterToTweetIndexGenerationJob.scala | 659 +++ .../simclusters_index_generation/README | 146 + ...based-simclusters-index-generation-job.d6w | 44 + .../scio/bq_generation/sql/BUILD | 3 + .../ads_user_tweet_action_pair_generation.sql | 38 + .../bq_generation/sql/cluster_top_tweets.sql | 15 + ...eets_intersection_with_fav_based_index.sql | 59 + ...ined_user_tweet_action_pair_generation.sql | 68 + .../sql/engagement_based_index_generation.sql | 85 + ...tent_user_tweet_action_pair_generation.sql | 62 + .../bq_generation/sql/nsfw_tweet_denylist.sql | 43 + .../sql/tweet_embeddings_generation.sql | 104 + .../bq_generation/sql/tweet_fav_count.sql | 38 + .../scio/bq_generation/sql/tweets_ann.sql | 64 + ...fied_user_tweet_action_pair_generation.sql | 45 + ..._video_tweet_fav_engagement_generation.sql | 69 + .../scio/bq_generation/tweets_ann/BUILD | 110 + .../bq_generation/tweets_ann/Config.scala | 33 + .../scio/bq_generation/tweets_ann/README | 95 + .../tweets_ann/TweetsANNFromBQ.scala | 120 + .../tweets_ann/TweetsANNJob.scala | 297 ++ .../iikf-hl-0-el-15-tweets-ann-batch-job.d6w | 39 + .../iikf-hl-2-el-15-tweets-ann-batch-job.d6w | 39 + .../iikf-hl-2-el-50-tweets-ann-batch-job.d6w | 39 + .../iikf-hl-8-el-50-tweets-ann-adhoc-job.d6w | 39 + .../iikf-hl-8-el-50-tweets-ann-batch-job.d6w | 39 + .../tweets_ann/iikf-tweets-ann-adhoc-job.d6w | 34 + .../tweets_ann/iikf-tweets-ann-batch-job.d6w | 39 + ...nsumer-embeddings-tweets-ann-adhoc-job.d6w | 34 + ...nsumer-embeddings-tweets-ann-batch-job.d6w | 39 + .../twitter/simclusters_v2/scio/common/BUILD | 21 + .../scio/common/ExternalDataSources.scala | 301 ++ .../AssembleMultiTypeGraphScioApp.scala | 39 + .../AssembleMultiTypeGraphScioBaseApp.scala | 574 +++ .../assemble_multi_type_graph/BUILD | 73 + .../assemble_multi_type_graph/Config.scala | 37 + .../assemble_multi_type_graph/README.md | 49 + .../assemble-multi-type-graph-scio-adhoc.d6w | 36 + .../assemble-multi-type-graph-scio-batch.d6w | 41 + .../scio/multi_type_graph/common/BUILD | 13 + .../common/MultiTypeGraphUtil.scala | 69 + .../multi_type_graph_sims/BUILD | 92 + .../multi_type_graph_sims/Config.scala | 18 + .../RightNodeCosineSimilarityScioApp.scala | 55 + ...RightNodeCosineSimilarityScioBaseApp.scala | 96 + .../RightNodeSimHashScioApp.scala | 43 + .../RightNodeSimHashScioBaseApp.scala | 65 + .../cosine-similarity-scio-adhoc.d6w | 33 + .../cosine-similarity-scio-batch.d6w | 39 + .../sim-hash-scio-adhoc.d6w | 33 + .../sim-hash-scio-batch.d6w | 38 + .../score/AggregatedScoreStore.scala | 24 + .../com/twitter/simclusters_v2/score/BUILD | 9 + .../twitter/simclusters_v2/score/Score.scala | 22 + .../score/ScoreFacadeStore.scala | 103 + .../simclusters_v2/score/ScoreId.scala | 129 + .../simclusters_v2/score/ScoreStore.scala | 72 + .../SimClustersEmbeddingPairScoreStore.scala | 201 + .../WeightedSumAggregatedScoreStore.scala | 84 + .../com/twitter/simclusters_v2/stores/BUILD | 14 + ...geFilteredLocaleEntityEmbeddingStore.scala | 96 + .../stores/MultiTypeGraphStore.scala | 287 ++ .../stores/SimClustersEmbeddingStore.scala | 120 + .../SimClustersMultiEmbeddingStore.scala | 74 + .../stores/TopicTopProducersStore.scala | 87 + .../simclusters_v2/stores/WtfMbcgStore.scala | 34 + .../twitter/simclusters_v2/summingbird/BUILD | 118 + .../simclusters_v2/summingbird/README.md | 4 + .../simclusters_v2/summingbird/common/BUILD | 62 + .../summingbird/common/ClientConfigs.scala | 81 + .../summingbird/common/Configs.scala | 70 + .../summingbird/common/EntityUtil.scala | 46 + .../summingbird/common/Implicits.scala | 140 + .../common/ModelVersionProfile.scala | 40 + .../summingbird/common/Monoids.scala | 478 +++ ...mClustersEmbeddingWithMetadataMonoid.scala | 59 + .../common/SimClustersHashUtil.scala | 14 + .../common/SimClustersInterestedInUtil.scala | 72 + .../common/SimClustersProfile.scala | 212 + .../summingbird/common/StatsUtil.scala | 22 + .../common/SummerWithSumValues.scala | 40 + .../common/ThriftDecayedValueMonoid.scala | 57 + .../common/TweetEntityExtractor.scala | 65 + .../stores/ApeTopicEmbeddingStore.scala | 43 + .../simclusters_v2/summingbird/stores/BUILD | 32 + .../stores/ClusterDetailsReadableStore.scala | 67 + .../EntityClusterScoreReadableStore.scala | 62 + .../stores/ManhattanFromStratoStore.scala | 108 + .../PersistentTweetEmbeddingStore.scala | 104 + ...oducerClusterEmbeddingReadableStores.scala | 101 + .../SemanticCoreEntityEmbeddingStore.scala | 49 + ...ttanReadableStoreForReadWriteDataset.scala | 65 + .../stores/TfgTopicEmbeddingsStore.scala | 46 + .../TopKClustersForEntityReadableStore.scala | 36 + .../TopKClustersForTweetReadableStore.scala | 176 + .../TopKTweetsForClusterReadableStore.scala | 298 ++ .../stores/TweetStatusCountsStore.scala | 29 + .../UserInterestedInReadableStore.scala | 263 ++ .../stores/UserKnownForReadableStore.scala | 75 + .../simclusters_v2/summingbird/storm/BUILD | 27 + .../storm/PersistentTweetJob.scala | 151 + .../storm/PersistentTweetJobRunner.scala | 227 + .../summingbird/storm/TweetJob.scala | 232 + .../summingbird/storm/TweetJobRunner.scala | 242 ++ .../storm/persistent_tweet_job_deploy.sh | 77 + .../summingbird/storm/tweet_alt_job_deploy.sh | 78 + .../summingbird/storm/tweet_job_deploy.sh | 77 + .../simclusters_v2/tweet_similarity/BUILD | 11 + ...imilaritySimClustersEmbeddingAdapter.scala | 37 + .../TweetSimilarityFeatures.scala | 54 + .../com/twitter/interaction_graph/BUILD | 15 + .../interaction_graph.thrift | 98 + src/thrift/com/twitter/recos/recos.thrift | 176 + .../com/twitter/recos/recos_common.thrift | 54 + .../com/twitter/recos/recos_injector.thrift | 22 + .../recos/user_tweet_entity_graph/BUILD | 19 + .../recos/user_tweet_entity_graph/CONFIG.ini | 7 + .../user_tweet_entity_graph.thrift | 187 + .../com/twitter/recos/user_tweet_graph/BUILD | 22 + .../twitter/recos/user_tweet_graph/CONFIG.ini | 7 + .../user_tweet_graph/user_tweet_graph.thrift | 172 + .../com/twitter/recos/user_user_graph/BUILD | 19 + .../twitter/recos/user_user_graph/CONFIG.ini | 7 + .../user_user_graph/user_user_graph.thrift | 45 + .../com/twitter/recos/user_video_graph/BUILD | 22 + .../twitter/recos/user_video_graph/CONFIG.ini | 7 + .../user_video_graph/user_video_graph.thrift | 64 + .../search/common/ranking/ranking.thrift | 366 ++ .../search/earlybird/thrift/earlybird.thrift | 1416 ++++++ src/thrift/com/twitter/simclusters_v2/BUILD | 23 + .../com/twitter/simclusters_v2/abuse.thrift | 53 + .../twitter/simclusters_v2/clustering.thrift | 18 + .../twitter/simclusters_v2/embedding.thrift | 137 + .../com/twitter/simclusters_v2/entity.thrift | 51 + .../twitter/simclusters_v2/evaluation.thrift | 65 + .../com/twitter/simclusters_v2/graph.thrift | 61 + .../twitter/simclusters_v2/identifier.thrift | 205 + .../simclusters_v2/inferred_entities.thrift | 38 + .../twitter/simclusters_v2/interests.thrift | 259 ++ .../simclusters_v2/multi_type_graph.thrift | 110 + .../offline_job_internal.thrift | 63 + .../simclusters_v2/online_store.thrift | 92 + .../online_store_internal.thrift | 30 + .../com/twitter/simclusters_v2/score.thrift | 71 + .../simclusters_v2/simclusters_presto.thrift | 59 + .../twitter/simclusters_v2/top_k_map.thrift | 14 + .../simclusters_v2/tweet_similarity.thrift | 16 + timelineranker/README.md | 36 + timelineranker/client/builder/BUILD | 6 + timelineranker/client/builder/README.md | 4 + .../client/builder/src/main/scala/BUILD | 16 + .../client/TimelineRankerClient.scala | 195 + .../client/TimelineRankerClientBuilder.scala | 89 + timelineranker/common/BUILD | 17 + timelineranker/common/src/main/scala/BUILD | 6 + .../com/twitter/timelineranker/adapter/BUILD | 14 + .../adapter/TimelineServiceAdapter.scala | 139 + .../com/twitter/timelineranker/model/BUILD | 23 + .../timelineranker/model/CandidateTweet.scala | 35 + .../model/CandidateTweetsResult.scala | 37 + .../model/HydratedTweetEntry.scala | 21 + .../timelineranker/model/Language.scala | 31 + .../timelineranker/model/LanguageScope.scala | 46 + .../model/PartiallyHydratedTweet.scala | 184 + .../model/PriorSeenEntries.scala | 23 + .../model/RankedTimelineQuery.scala | 14 + .../model/RankedTimelineQueryOptions.scala | 29 + .../timelineranker/model/RecapQuery.scala | 278 ++ .../model/ReverseChronTimelineQuery.scala | 23 + .../ReverseChronTimelineQueryOptions.scala | 31 + .../timelineranker/model/TimeRange.scala | 39 + .../timelineranker/model/Timeline.scala | 46 + .../timelineranker/model/TimelineEntry.scala | 18 + .../model/TimelineEntryEnvelope.scala | 24 + .../timelineranker/model/TimelineQuery.scala | 82 + .../model/TimelineQueryOptions.scala | 20 + .../timelineranker/model/TimelineRange.scala | 18 + .../twitter/timelineranker/model/Tweet.scala | 62 + .../timelineranker/model/TweetIdRange.scala | 53 + .../model/UtegLikedByTweetsOptions.scala | 8 + timelineranker/server/BUILD.bazel | 17 + timelineranker/server/config/BUILD | 14 + timelineranker/server/config/decider.yml | 153 + .../server/src/main/resources/BUILD.bazel | 5 + .../main/resources/logback-timelineranker.xml | 124 + .../server/src/main/scala/BUILD.bazel | 32 + .../com/twitter/timelineranker/clients/BUILD | 22 + .../CortexTweetQueryServiceClient.scala | 113 + .../clients/MemcacheFactory.scala | 48 + .../clients/content_features_cache/BUILD | 24 + .../ContentFeaturesMemcacheBuilder.scala | 39 + .../com/twitter/timelineranker/common/BUILD | 43 + .../common/CandidateGenerationTransform.scala | 40 + .../ContentFeaturesHydrationTransform.scala | 112 + .../CreateCandidateEnvelopeTransform.scala | 15 + .../FeatureHydrationDataTransform.scala | 33 + ...FollowAndRealGraphCombiningTransform.scala | 198 + .../common/FollowGraphDataTransform.scala | 23 + ...tsAndSourceTweetsInParallelTransform.scala | 31 + .../HydratedTweetsFilterTransform.scala | 96 + ...eetsSearchFeaturesHydrationTransform.scala | 38 + .../common/MarkRandomTweetTransform.scala | 55 + ...epliesToUserIdSearchResultsTransform.scala | 49 + ...eetsSearchFeaturesHydrationTransform.scala | 31 + ...pHydrationSearchResultsTransformBase.scala | 29 + .../common/RecapSearchResultsTransform.scala | 89 + ...ecapSearchResultsTruncationTransform.scala | 67 + ...SearchResultDedupAndSortingTransform.scala | 23 + .../SourceTweetsSearchResultsTransform.scala | 62 + .../TrimToMatchHydratedTweetsTransform.scala | 37 + .../TrimToMatchSearchResultsTransform.scala | 57 + .../common/TweetHydrationTransform.scala | 62 + ...dOptionHydratedTweetsFilterTransform.scala | 85 + .../common/UserLanguagesTransform.scala | 30 + .../common/UserProfileInfoTransform.scala | 29 + .../common/VisibilityEnforcingTransform.scala | 22 + .../com/twitter/timelineranker/config/BUILD | 65 + .../timelineranker/config/CallInfo.scala | 119 + .../config/ClientAccessPermissions.scala | 287 ++ .../config/ClientWrapperFactories.scala | 86 + .../config/ClientWrappers.scala | 11 + ...DefaultUnderlyingClientConfiguration.scala | 158 + .../timelineranker/config/RequestScopes.scala | 13 + .../config/RuntimeConfiguration.scala | 133 + .../StagingUnderlyingConfiguration.scala | 6 + .../config/TimelineRankerConstants.scala | 8 + .../config/TimelineRankerFlags.scala | 72 + .../UnderlyingClientConfiguration.scala | 107 + .../timelineranker/contentfeatures/BUILD | 13 + .../contentfeatures/package.scala | 10 + .../com/twitter/timelineranker/core/BUILD | 24 + .../core/CandidateEnvelope.scala | 24 + .../timelineranker/core/FollowGraphData.scala | 34 + .../core/FollowGraphDataFuture.scala | 53 + ...ydratedCandidatesAndFeaturesEnvelope.scala | 18 + .../timelineranker/core/HydratedTweets.scala | 7 + .../twitter/timelineranker/core/package.scala | 13 + .../com/twitter/timelineranker/decider/BUILD | 9 + .../timelineranker/decider/DeciderKey.scala | 83 + .../timelineranker/entity_tweets/BUILD.bazel | 39 + .../EntityTweetsRepository.scala | 20 + .../EntityTweetsRepositoryBuilder.scala | 60 + .../EntityTweetsSearchResultsTransform.scala | 71 + .../entity_tweets/EntityTweetsSource.scala | 146 + .../timelineranker/in_network_tweets/BUILD | 41 + .../InNetworkTweetRepository.scala | 31 + .../InNetworkTweetRepositoryBuilder.scala | 109 + .../InNetworkTweetSource.scala | 271 ++ .../twitter/timelineranker/monitoring/BUILD | 15 + ...UsersSearchResultMonitoringTransform.scala | 52 + .../timelineranker/observe/BUILD.bazel | 15 + .../observe/DebugObserverBuilder.scala | 36 + .../observe/ObservedRequests.scala | 35 + .../twitter/timelineranker/parameters/BUILD | 18 + .../parameters/ConfigBuilder.scala | 60 + .../parameters/entity_tweets/BUILD | 12 + .../entity_tweets/EntityTweetsParams.scala | 65 + .../EntityTweetsProduction.scala | 42 + .../parameters/in_network_tweets/BUILD | 15 + .../InNetworkTweetParams.scala | 133 + .../InNetworkTweetProduction.scala | 71 + .../parameters/monitoring/BUILD | 11 + .../monitoring/MonitoringParams.scala | 13 + .../monitoring/MonitoringProduction.scala | 14 + .../timelineranker/parameters/recap/BUILD | 18 + .../parameters/recap/RecapParams.scala | 231 + .../parameters/recap/RecapProduction.scala | 115 + .../parameters/recap/RecapQueryContext.scala | 79 + .../parameters/recap_author/BUILD | 12 + .../recap_author/RecapAuthorParams.scala | 53 + .../recap_author/RecapAuthorProduction.scala | 46 + .../parameters/recap_hydration/BUILD | 12 + .../RecapHydrationParams.scala | 48 + .../RecapHydrationProduction.scala | 45 + .../timelineranker/parameters/revchron/BUILD | 18 + .../revchron/ReverseChronParams.scala | 45 + .../revchron/ReverseChronProduction.scala | 28 + .../ReverseChronTimelineQueryContext.scala | 114 + ...erseChronTimelineQueryContextBuilder.scala | 72 + .../parameters/uteg_liked_by_tweets/BUILD | 13 + .../UtegLikedByTweetsParams.scala | 174 + .../UtegLikedByTweetsProduction.scala | 87 + .../timelineranker/parameters/util/BUILD | 24 + .../util/CommonRequestContext.scala | 50 + .../parameters/util/ConfigHelper.scala | 41 + .../util/RecapQueryParamInitializer.scala | 20 + .../com/twitter/timelineranker/recap/BUILD | 8 + .../twitter/timelineranker/recap/model/BUILD | 15 + .../recap/model/ContentFeatures.scala | 222 + .../timelineranker/recap_author/BUILD.bazel | 41 + .../recap_author/RecapAuthorRepository.scala | 29 + .../RecapAuthorRepositoryBuilder.scala | 90 + .../RecapAuthorSearchResultsTransform.scala | 69 + .../recap_author/RecapAuthorSource.scala | 212 + .../recap_hydration/BUILD.bazel | 36 + .../RecapHydrationRepository.scala | 20 + .../RecapHydrationRepositoryBuilder.scala | 47 + ...RecapHydrationSearchResultsTransform.scala | 15 + .../RecapHydrationSource.scala | 123 + .../twitter/timelineranker/repository/BUILD | 32 + .../CandidatesRepositoryBuilder.scala | 89 + .../RankedHomeTimelineRepository.scala | 16 + .../repository/RepositoryBuilder.scala | 17 + .../ReverseChronHomeTimelineRepository.scala | 20 + ...seChronHomeTimelineRepositoryBuilder.scala | 72 + .../RoutingTimelineRepository.scala | 25 + .../RoutingTimelineRepositoryBuilder.scala | 18 + .../repository/TimelineRepository.scala | 16 + .../twitter/timelineranker/server/BUILD.bazel | 65 + .../twitter/timelineranker/server/Main.scala | 182 + .../server/TimelineRanker.scala | 255 ++ .../server/TimelineRankerBuilder.scala | 127 + .../server/TimelineRankerThriftWebForms.scala | 46 + .../timelineranker/server/Warmup.scala | 53 + .../com/twitter/timelineranker/source/BUILD | 22 + .../ReverseChronHomeTimelineSource.scala | 327 ++ .../source/TimelineSource.scala | 16 + .../uteg_liked_by_tweets/BUILD.bazel | 51 + .../CombinedScoreAndTruncateTransform.scala | 95 + ...horFavoritedByUserIdsFilterTransform.scala | 39 + ...uthoredByWeightedFollowingsTransform.scala | 29 + ...lProofAndUTEGScoreHydrationTransform.scala | 28 + .../UTEGResultsTransform.scala | 74 + .../UtegLikedByTweetsRepository.scala | 18 + .../UtegLikedByTweetsRepositoryBuilder.scala | 46 + ...gLikedByTweetsSearchResultsTransform.scala | 36 + .../UtegLikedByTweetsSource.scala | 306 ++ .../com/twitter/timelineranker/util/BUILD | 39 + .../util/CachingContentFeaturesProvider.scala | 120 + ...tFeaturesIntoHydratedTweetsTransform.scala | 59 + ...uresIntoThriftTweetFeaturesTransform.scala | 106 + .../util/ExtendedRepliesFilter.scala | 72 + .../util/LatentRepository.scala | 31 + .../util/RecommendedRepliesFilter.scala | 28 + .../util/ReverseExtendedRepliesFilter.scala | 32 + .../util/SearchResultUtil.scala | 123 + .../SearchResultWithVisibilityActors.scala | 61 + .../timelineranker/util/SnowflakeUtils.scala | 19 + .../util/SourceTweetsUtil.scala | 88 + .../TweetAnnotationFeaturesExtractor.scala | 19 + .../timelineranker/util/TweetHydrator.scala | 76 + .../util/TweetMediaFeatureExtractor.scala | 272 ++ .../util/TweetTextFeaturesExtractor.scala | 199 + .../util/TweetsPostFilter.scala | 493 +++ ...weetsPostFilterBasedOnSearchMetadata.scala | 170 + .../TweetypieContentFeaturesProvider.scala | 114 + .../twitter/timelineranker/visibility/BUILD | 17 + .../visibility/FollowGraphDataProvider.scala | 25 + .../RealGraphFollowGraphDataProvider.scala | 134 + .../SgsFollowGraphDataProvider.scala | 266 ++ .../earlybird_ranking/earlybird_ranking/BUILD | 8 + .../earlybird_ranking/common/BUILD | 24 + .../EarlybirdTrainingConfiguration.scala | 271 ++ .../EarlybirdTrainingRecapConfiguration.scala | 17 + ...rlybirdTrainingRectweetConfiguration.scala | 100 + .../earlybird_ranking/model_evaluation/BUILD | 36 + .../EarlybirdEvaluationMetric.scala | 203 + .../EarlybirdModelEvaluationJob.scala | 214 + .../training_data_generation/BUILD | 89 + .../EarlybirdExampleSampler.scala | 65 + .../EarlybirdStatsJob.scala | 63 + .../EarlybirdTrainingDataJob.scala | 92 + trust_and_safety_models/README.md | 10 + .../abusive/abusive_model.py | 276 ++ trust_and_safety_models/nsfw/nsfw_media.py | 466 ++ trust_and_safety_models/nsfw/nsfw_text.py | 152 + trust_and_safety_models/toxicity/__init__.py | 0 .../toxicity/data/__init__.py | 0 .../toxicity/data/data_preprocessing.py | 118 + .../toxicity/data/dataframe_loader.py | 348 ++ .../toxicity/data/mb_generator.py | 284 ++ .../toxicity/load_model.py | 227 + .../toxicity/optim/__init__.py | 0 .../toxicity/optim/callbacks.py | 220 + .../toxicity/optim/losses.py | 56 + .../toxicity/optim/schedulers.py | 44 + trust_and_safety_models/toxicity/rescoring.py | 54 + .../toxicity/settings/__init__.py | 0 .../toxicity/settings/default_settings_tox.py | 38 + trust_and_safety_models/toxicity/train.py | 401 ++ .../toxicity/utils/__init__.py | 0 .../toxicity/utils/helpers.py | 99 + twml/BUILD | 186 + twml/README.md | 13 + twml/libtwml/BUILD | 8 + twml/libtwml/include/twml.h | 21 + .../include/twml/BatchPredictionRequest.h | 45 + .../include/twml/BatchPredictionResponse.h | 58 + twml/libtwml/include/twml/BlockFormatReader.h | 32 + twml/libtwml/include/twml/BlockFormatWriter.h | 61 + twml/libtwml/include/twml/DataRecord.h | 108 + twml/libtwml/include/twml/DataRecordReader.h | 61 + twml/libtwml/include/twml/DataRecordWriter.h | 39 + twml/libtwml/include/twml/Error.h | 48 + twml/libtwml/include/twml/HashedDataRecord.h | 70 + .../include/twml/HashedDataRecordReader.h | 70 + twml/libtwml/include/twml/Hashmap.h | 110 + twml/libtwml/include/twml/RawTensor.h | 92 + twml/libtwml/include/twml/Tensor.h | 82 + twml/libtwml/include/twml/TensorRecord.h | 47 + .../libtwml/include/twml/TensorRecordReader.h | 34 + .../libtwml/include/twml/TensorRecordWriter.h | 35 + twml/libtwml/include/twml/ThriftReader.h | 56 + twml/libtwml/include/twml/ThriftWriter.h | 59 + twml/libtwml/include/twml/Type.h | 69 + twml/libtwml/include/twml/common.h | 42 + twml/libtwml/include/twml/defines.h | 36 + twml/libtwml/include/twml/discretizer_impl.h | 22 + twml/libtwml/include/twml/functions.h | 26 + .../include/twml/hashing_discretizer_impl.h | 22 + twml/libtwml/include/twml/io/IOError.h | 45 + twml/libtwml/include/twml/optim.h | 51 + twml/libtwml/include/twml/utilities.h | 18 + twml/libtwml/setup.cfg | 9 + twml/libtwml/setup.py | 12 + .../src/lib/BatchPredictionRequest.cpp | 52 + .../src/lib/BatchPredictionResponse.cpp | 125 + twml/libtwml/src/lib/BlockFormatReader.cpp | 145 + twml/libtwml/src/lib/BlockFormatWriter.cpp | 163 + twml/libtwml/src/lib/CMakeLists.txt | 36 + twml/libtwml/src/lib/CPPLINT.cfg | 1 + twml/libtwml/src/lib/DataRecord.cpp | 72 + twml/libtwml/src/lib/DataRecordReader.cpp | 230 + twml/libtwml/src/lib/DataRecordWriter.cpp | 162 + twml/libtwml/src/lib/HashedDataRecord.cpp | 80 + .../src/lib/HashedDataRecordReader.cpp | 218 + twml/libtwml/src/lib/Hashmap.cpp | 380 ++ twml/libtwml/src/lib/Tensor.cpp | 191 + twml/libtwml/src/lib/TensorRecordReader.cpp | 323 ++ twml/libtwml/src/lib/TensorRecordWriter.cpp | 162 + twml/libtwml/src/lib/ThriftReader.cpp | 33 + twml/libtwml/src/lib/ThriftWriter.cpp | 91 + twml/libtwml/src/lib/discretizer_impl.cpp | 167 + twml/libtwml/src/lib/functions.cpp | 158 + .../src/lib/hashing_discretizer_impl.cpp | 241 ++ twml/libtwml/src/lib/internal/endianutils.h | 137 + twml/libtwml/src/lib/internal/error.h | 29 + twml/libtwml/src/lib/internal/interpolate.h | 74 + twml/libtwml/src/lib/internal/khash.h | 627 +++ twml/libtwml/src/lib/internal/linear_search.h | 17 + twml/libtwml/src/lib/internal/murmur_hash3.h | 37 + twml/libtwml/src/lib/internal/thrift.h | 69 + twml/libtwml/src/lib/internal/utf_converter.h | 10 + twml/libtwml/src/lib/io/IOError.cpp | 61 + twml/libtwml/src/lib/murmur_hash3.cpp | 335 ++ twml/libtwml/src/lib/optim.cpp | 274 ++ twml/libtwml/src/lib/utf_converter.cpp | 53 + twml/libtwml/src/ops/CMakeLists.txt | 79 + twml/libtwml/src/ops/add1.cpp | 92 + .../src/ops/batch_prediction_request.cpp | 183 + .../src/ops/batch_prediction_request_v2.cpp | 224 + .../ops/batch_prediction_response_writer.cpp | 82 + ...atch_prediction_tensor_response_writer.cpp | 81 + .../src/ops/binary_sparse_dense_matmul.cpp | 330 ++ .../src/ops/binary_sparse_dense_matmul.h | 75 + .../src/ops/binary_sparse_dense_matmul_impl.h | 145 + twml/libtwml/src/ops/block_format_dataset.cpp | 243 ++ twml/libtwml/src/ops/block_format_reader.h | 50 + twml/libtwml/src/ops/compress_sample_ids.cpp | 138 + .../src/ops/contrib/get_substrings.cpp | 116 + twml/libtwml/src/ops/data_record.cpp | 1891 +++++++++ .../src/ops/data_record_tensor_writer.cpp | 81 + twml/libtwml/src/ops/discretizer.cpp | 293 ++ twml/libtwml/src/ops/feature_extractor.cpp | 134 + twml/libtwml/src/ops/feature_id.cpp | 58 + twml/libtwml/src/ops/feature_mask.cpp | 83 + twml/libtwml/src/ops/fixed_length_tensor.cpp | 190 + twml/libtwml/src/ops/hashed_data_record.cpp | 520 +++ twml/libtwml/src/ops/hashing_discretizer.cpp | 260 ++ twml/libtwml/src/ops/hashmap.cpp | 84 + twml/libtwml/src/ops/isotonic_calibration.cpp | 81 + twml/libtwml/src/ops/num_intra_op_threads.cpp | 39 + twml/libtwml/src/ops/par_add.cpp | 75 + .../src/ops/partition_sparse_tensor.cpp | 125 + .../src/ops/percentile_discretizer_v2.cpp | 241 ++ twml/libtwml/src/ops/resource_utils.h | 126 + twml/libtwml/src/ops/scripts/get_inc.py | 5 + twml/libtwml/src/ops/scripts/get_inc.sh | 2 + twml/libtwml/src/ops/scripts/get_lib.py | 5 + twml/libtwml/src/ops/scripts/get_lib.sh | 2 + twml/libtwml/src/ops/scripts/symlink.sh | 12 + twml/libtwml/src/ops/sleep_op.cpp | 51 + twml/libtwml/src/ops/sparse_normalization.cpp | 378 ++ twml/libtwml/src/ops/tensor_record.cpp | 692 +++ twml/libtwml/src/ops/tensorflow_utils.cpp | 87 + twml/libtwml/src/ops/tensorflow_utils.h | 13 + twml/libtwml/src/ops/var_length_reader.cpp | 46 + twml/setup.cfg | 8 + twml/setup.py | 29 + twml/twml/__init__.py | 61 + twml/twml/argument_parser.py | 561 +++ twml/twml/array.py | 101 + twml/twml/block_format_writer.py | 65 + twml/twml/constants.py | 11 + twml/twml/contrib/__init__.py | 21 + twml/twml/contrib/build_graphs_fns.py | 32 + twml/twml/contrib/calibrators/__init__.py | 18 + twml/twml/contrib/calibrators/calibrator.py | 157 + .../contrib/calibrators/common_calibrators.py | 707 +++ .../hashed_percentile_discretizer.py | 22 + .../calibrators/hashing_discretizer.py | 35 + twml/twml/contrib/calibrators/isotonic.py | 317 ++ twml/twml/contrib/calibrators/mdl.py | 118 + .../calibrators/percentile_discretizer.py | 577 +++ twml/twml/contrib/eventbus/input_fn.py | 59 + twml/twml/contrib/eventbus/reader.py | 119 + twml/twml/contrib/export/__init__.py | 2 + twml/twml/contrib/export/export_fn.py | 264 ++ twml/twml/contrib/export/exporters.py | 145 + twml/twml/contrib/feature_config.py | 85 + twml/twml/contrib/feature_config_parsers.py | 224 + .../contrib/feature_importances/__init__.py | 0 .../feature_importances.py | 414 ++ .../feature_permutation.py | 129 + .../contrib/feature_importances/helpers.py | 96 + twml/twml/contrib/hooks.py | 42 + twml/twml/contrib/initializers.py | 61 + twml/twml/contrib/layers/__init__.py | 11 + twml/twml/contrib/layers/embedding_lookup.py | 419 ++ .../contrib/layers/factorization_machine.py | 179 + twml/twml/contrib/layers/full_dense.py | 380 ++ .../layers/hashed_percentile_discretizer.py | 217 + .../contrib/layers/hashing_discretizer.py | 156 + twml/twml/contrib/layers/mask_layer.py | 29 + twml/twml/contrib/layers/stacked_rnn.py | 189 + .../contrib/layers/zscore_normalization.py | 247 ++ twml/twml/contrib/metrics/__init__.py | 5 + twml/twml/contrib/metrics/metrics.py | 209 + twml/twml/contrib/metrics/search_metrics.py | 292 ++ twml/twml/contrib/optimizers/__init__.py | 4 + .../deep_gradient_compression_optimizer.py | 180 + .../contrib/optimizers/pruning_optimizer.py | 164 + twml/twml/contrib/parsers.py | 21 + twml/twml/contrib/pruning.py | 363 ++ twml/twml/contrib/readers/__init__.py | 5 + .../readers/batch_prediction_request.py | 8 + twml/twml/contrib/readers/data_record.py | 10 + .../hashed_batch_prediction_request.py | 8 + twml/twml/contrib/trainers/__init__.py | 5 + .../batch_prediction_request_trainer.py | 180 + .../trainers/pruning_data_record_trainer.py | 59 + twml/twml/contrib/trainers/trainer_utils.py | 111 + twml/twml/contrib/utils/__init__.py | 18 + twml/twml/contrib/utils/datasets.py | 93 + twml/twml/contrib/utils/device.py | 27 + twml/twml/contrib/utils/interp.py | 94 + twml/twml/contrib/utils/loss_fns.py | 302 ++ twml/twml/contrib/utils/masks.py | 38 + twml/twml/contrib/utils/math_fns.py | 171 + twml/twml/contrib/utils/normalizer.py | 39 + twml/twml/contrib/utils/scores.py | 33 + twml/twml/contrib/utils/similarities.py | 17 + twml/twml/dataset.py | 372 ++ twml/twml/errors.py | 13 + twml/twml/export_output_fns.py | 17 + twml/twml/feature_config.py | 54 + twml/twml/filters.py | 9 + twml/twml/hooks.py | 562 +++ twml/twml/input_fns.py | 129 + twml/twml/layers/__init__.py | 21 + .../layers/batch_prediction_tensor_writer.py | 51 + twml/twml/layers/batch_prediction_writer.py | 51 + twml/twml/layers/data_record_tensor_writer.py | 50 + twml/twml/layers/full_dense.py | 259 ++ twml/twml/layers/full_sparse.py | 370 ++ twml/twml/layers/isotonic.py | 76 + twml/twml/layers/layer.py | 50 + twml/twml/layers/mdl.py | 256 ++ twml/twml/layers/partition.py | 74 + twml/twml/layers/percentile_discretizer.py | 209 + twml/twml/layers/sequential.py | 160 + twml/twml/layers/sparse_max_norm.py | 221 + twml/twml/layers/stitch.py | 54 + twml/twml/learning_rate_decay.py | 168 + twml/twml/lookup/__init__.py | 11 + twml/twml/metrics.py | 1380 ++++++ twml/twml/optimizers/__init__.py | 4 + twml/twml/parsers.py | 20 + twml/twml/readers/__init__.py | 7 + twml/twml/readers/batch_prediction_request.py | 8 + twml/twml/readers/data_record.py | 15 + .../hashed_batch_prediction_request.py | 8 + twml/twml/readers/hashed_data_record.py | 12 + twml/twml/saved_model_cli/__init__.py | 0 twml/twml/saved_model_cli/__main__.py | 9 + twml/twml/summary/__init__.py | 6 + twml/twml/tensorboard/__init__.py | 0 twml/twml/tensorboard/__main__.py | 16 + twml/twml/tensorio.py | 161 + twml/twml/tracking/__init__.py | 5 + twml/twml/tracking/experiment_tracker.py | 543 +++ twml/twml/trainers/__init__.py | 10 + twml/twml/trainers/data_record_trainer.py | 821 ++++ twml/twml/trainers/trainer.py | 1777 ++++++++ twml/twml/util.py | 942 ++++ twml/twml_common/__init__.py | 0 twml/twml_common/initializer.py | 14 + twml/twml_common/serialize.py | 16 + twml/twml_common/sparse_inputs.py | 24 + visibilitylib/BUILD | 29 + visibilitylib/README.md | 51 + visibilitylib/src/main/resources/config/BUILD | 6 + .../config/com/twitter/visibility/decider.yml | 906 ++++ .../main/scala/com/twitter/visibility/BUILD | 42 + .../visibility/VisibilityLibrary.scala | 387 ++ .../com/twitter/visibility/builder/BUILD | 29 + .../builder/FeatureMapBuilder.scala | 64 + .../visibility/builder/VerdictLogger.scala | 187 + .../visibility/builder/VisibilityResult.scala | 112 + .../builder/VisibilityResultBuilder.scala | 114 + .../twitter/visibility/builder/common/BUILD | 32 + .../builder/common/MutedKeywordFeatures.scala | 228 + .../com/twitter/visibility/builder/dms/BUILD | 23 + .../builder/dms/DmConversationFeatures.scala | 196 + .../builder/dms/DmEventFeatures.scala | 341 ++ .../twitter/visibility/builder/media/BUILD | 31 + .../builder/media/MediaFeatures.scala | 90 + .../builder/media/MediaMetadataFeatures.scala | 79 + .../twitter/visibility/builder/spaces/BUILD | 25 + .../builder/spaces/SpaceFeatures.scala | 131 + .../twitter/visibility/builder/tweets/BUILD | 38 + .../tweets/BlenderContextFeatures.scala | 45 + .../CommunityNotificationFeatures.scala | 64 + .../tweets/CommunityTweetFeatures.scala | 70 + .../CommunityTweetFeaturesPartitioned.scala | 26 + .../tweets/CommunityTweetFeaturesV2.scala | 129 + .../tweets/ConversationControlFeatures.scala | 178 + .../builder/tweets/EditTweetFeatures.scala | 71 + .../tweets/ExclusiveTweetFeatures.scala | 65 + ...rPefetchedLabelsRelationshipFeatures.scala | 81 + .../tweets/FosnrRelationshipFeatures.scala | 82 + .../tweets/MisinformationPolicyFeatures.scala | 86 + .../builder/tweets/ModerationFeatures.scala | 23 + .../tweets/SearchContextFeatures.scala | 44 + .../tweets/ToxicReplyFilterFeature.scala | 57 + .../tweets/TrustedFriendsFeatures.scala | 57 + .../builder/tweets/TweetFeatures.scala | 210 + .../builder/tweets/TweetIdFeatures.scala | 76 + .../tweets/TweetMediaMetadataFeatures.scala | 130 + .../tweets/TweetPerspectiveFeatures.scala | 54 + .../TweetVisibilityNudgeSourceWrapper.scala | 39 + .../UnmentionNotificationFeatures.scala | 75 + .../builder/users/AuthorDeviceFeatures.scala | 39 + .../builder/users/AuthorFeatures.scala | 221 + .../twitter/visibility/builder/users/BUILD | 22 + .../builder/users/QuotedTweetFeatures.scala | 52 + .../builder/users/RelationshipFeatures.scala | 176 + .../users/RelationshipVerbHelpers.scala | 79 + .../builder/users/SearchFeatures.scala | 26 + .../users/UserUnavailableFeatures.scala | 145 + .../ViewerAdvancedFilteringFeatures.scala | 92 + .../builder/users/ViewerFeatures.scala | 245 ++ .../users/ViewerSearchSafetyFeatures.scala | 49 + ...ViewerSensitiveMediaSettingsFeatures.scala | 41 + .../com/twitter/visibility/configapi/BUILD | 19 + .../visibility/configapi/ConfigBuilder.scala | 43 + .../configapi/VisibilityParams.scala | 61 + .../configapi/VisibilityRequestContext.scala | 14 + .../VisibilityRequestContextFactory.scala | 64 + .../visibility/configapi/configs/BUILD | 17 + .../configapi/configs/DeciderKey.scala | 1064 +++++ .../configapi/configs/ExperimentsHelper.scala | 26 + .../configs/VisibilityDeciderGates.scala | 73 + .../configs/VisibilityDeciders.scala | 389 ++ .../configs/VisibilityExperimentsConfig.scala | 33 + .../configs/VisibilityFeatureSwitches.scala | 74 + .../configapi/configs/overrides/BUILD | 9 + .../VisibilityLibraryDeciderOverrides.scala | 24 + .../twitter/visibility/configapi/params/BUILD | 15 + .../configapi/params/FSRuleParams.scala | 213 + .../configapi/params/GlobalParams.scala | 11 + .../configapi/params/LabelSourceParams.scala | 15 + .../configapi/params/RuleParams.scala | 164 + .../configapi/params/SafetyLevelParams.scala | 214 + ...nversationsDownrankingSpecificParams.scala | 13 + .../params/VisibilityExperiment.scala | 19 + .../params/VisibilityExperiments.scala | 16 + .../scala/com/twitter/visibility/engine/BUILD | 22 + .../DeciderableVisibilityRuleEngine.scala | 26 + .../VisibilityResultsMetricRecorder.scala | 179 + .../engine/VisibilityRuleEngine.scala | 266 ++ .../engine/VisibilityRulePreprocessor.scala | 156 + .../features/AdvancedFilteringFeatures.scala | 24 + .../com/twitter/visibility/features/BUILD | 17 + .../twitter/visibility/features/Feature.scala | 11 + .../visibility/features/FeatureMap.scala | 121 + .../visibility/features/Features.scala | 269 ++ .../com/twitter/visibility/generators/BUILD | 30 + .../generators/CountryNameGenerator.scala | 58 + .../EpitaphToLocalizedMessage.scala | 66 + ...InterstitialReasonToLocalizedMessage.scala | 47 + .../LocalizedInterstitialGenerator.scala | 151 + .../generators/TombstoneGenerator.scala | 94 + .../visibility/interfaces/blender/BUILD | 34 + .../blender/BlenderVisibilityLibrary.scala | 416 ++ .../blender/BlenderVisibilityRequest.scala | 42 + .../blender/CombinedVisibilityResult.scala | 7 + .../twitter/visibility/interfaces/cards/BUILD | 17 + .../cards/CardVisibilityLibrary.scala | 187 + .../CardVisibilityLibraryParityTest.scala | 35 + .../cards/CardVisibilityRequest.scala | 13 + .../visibility/interfaces/common/BUILD.bazel | 15 + .../interfaces/common/blender/BUILD | 14 + .../blender/BlenderVFRequestContext.scala | 19 + .../visibility/interfaces/common/search/BUILD | 14 + .../search/SearchVFRequestContext.scala | 19 + .../visibility/interfaces/common/tweets/BUILD | 17 + .../tweets/StratoSafetyLabelFetcher.scala | 18 + .../tweets/StratoSafetyLabelMapFetcher.scala | 17 + .../interfaces/common/tweets/package.scala | 13 + .../conversations/AdAvoidanceLibrary.scala | 158 + .../visibility/interfaces/conversations/BUILD | 46 + ...melineConversationsVisibilityLibrary.scala | 260 ++ ...melineConversationsVisibilityRequest.scala | 20 + ...elineConversationsVisibilityResponse.scala | 7 + .../interfaces/conversations/Tombstone.scala | 35 + .../TombstoneVisibilityLibrary.scala | 633 +++ .../interfaces/conversations/package.scala | 11 + .../twitter/visibility/interfaces/des/BUILD | 23 + .../des/DESRealtimeVisibilityLibrary.scala | 99 + .../interfaces/des/DESVisibilityLibrary.scala | 72 + .../twitter/visibility/interfaces/dms/BUILD | 32 + .../dms/DmConversationVisibilityLibrary.scala | 94 + .../dms/DmConversationVisibilityRequest.scala | 9 + .../dms/DmEventVisibilityLibrary.scala | 80 + .../dms/DmEventVisibilityRequest.scala | 9 + .../interfaces/dms/DmVisibilityLibrary.scala | 88 + .../visibility/interfaces/dms/package.scala | 12 + .../visibility/interfaces/media/BUILD.bazel | 16 + .../media/MediaVisibilityLibrary.scala | 89 + .../media/MediaVisibilityRequest.scala | 10 + .../visibility/interfaces/notifications/BUILD | 31 + .../notifications/NotificationVFRequest.scala | 9 + .../NotificationsFilteringResponse.scala | 13 + ...tificationsPlatformFilteringResponse.scala | 13 + ...tificationsPlatformVisibilityLibrary.scala | 157 + .../NotificationsVisibilityLibrary.scala | 181 + .../interfaces/push_service/BUILD.bazel | 31 + .../PushServiceSafetyLabelMapFetcher.scala | 21 + .../PushServiceVisibilityLibrary.scala | 179 + .../PushServiceVisibilityLibraryParity.scala | 74 + .../PushServiceVisibilityLibraryUtil.scala | 57 + .../PushServiceVisibilityRequest.scala | 19 + .../PushServiceVisibilityResponse.scala | 52 + .../visibility/interfaces/search/BUILD | 34 + .../search/BatchSearchVisibilityRequest.scala | 9 + .../BatchSearchVisibilityResponse.scala | 5 + .../search/CombinedVisibilityResult.scala | 7 + .../search/SearchVisibilityLibrary.scala | 466 ++ .../interfaces/search/TweetContext.scala | 10 + .../visibility/interfaces/spaces/BUILD | 37 + .../spaces/SpaceVisibilityLibrary.scala | 117 + .../spaces/SpaceVisibilityRequest.scala | 10 + .../visibility/interfaces/tweets/BUILD | 47 + .../DeletedTweetVisibilityLibrary.scala | 59 + .../tweets/QuotedTweetVisibilityLibrary.scala | 150 + .../tweets/TweetVisibilityLibrary.scala | 421 ++ .../TweetVisibilityLibraryParityTest.scala | 109 + .../tweets/TweetVisibilityRequest.scala | 14 + .../interfaces/tweets/TweetypieContext.scala | 59 + ...serUnavailableStateVisibilityLibrary.scala | 138 + ...serUnavailableStateVisibilityRequest.scala | 14 + .../interfaces/tweets/enrichments/BUILD | 25 + .../ComplianceTweetNoticeEnrichment.scala | 55 + .../LimitedActionsPolicyEnrichment.scala | 173 + .../TweetVisibilityNudgeEnrichment.scala | 96 + .../twitter/visibility/interfaces/users/BUILD | 21 + .../users/UserVisibilityLibrary.scala | 111 + .../scala/com/twitter/visibility/models/BUILD | 32 + .../visibility/models/CommunityTweet.scala | 23 + .../twitter/visibility/models/ContentId.scala | 22 + .../visibility/models/LabelSource.scala | 61 + .../models/MediaSafetyLabelType.scala | 89 + .../models/MisinformationPolicy.scala | 179 + .../visibility/models/MutedKeyword.scala | 3 + .../visibility/models/SafetyLabel.scala | 90 + .../models/SafetyLabelMetadata.scala | 25 + .../visibility/models/SafetyLabelType.scala | 7 + .../visibility/models/SafetyLevel.scala | 851 ++++ .../visibility/models/SafetyLevelGroup.scala | 557 +++ .../models/SemanticCoreAnnotation.scala | 3 + .../models/SpaceSafetyLabelType.scala | 95 + .../visibility/models/TweetDeleteReason.scala | 6 + .../models/TweetModelMetadata.scala | 23 + .../visibility/models/TweetSafetyLabel.scala | 360 ++ .../visibility/models/UnitOfDiversion.scala | 16 + .../twitter/visibility/models/UserAge.scala | 15 + .../twitter/visibility/models/UserLabel.scala | 244 ++ .../models/UserSensitiveMediaSettings.scala | 13 + .../models/UserUnavailableStateEnum.scala | 22 + .../visibility/models/ViewerContext.scala | 53 + .../visibility/models/ViolationLevel.scala | 51 + .../twitter/visibility/models/package.scala | 5 + .../com/twitter/visibility/rules/Action.scala | 916 ++++ .../rules/AdvancedFilteringRules.scala | 71 + .../scala/com/twitter/visibility/rules/BUILD | 37 + .../twitter/visibility/rules/CardRules.scala | 52 + .../visibility/rules/ComposableActions.scala | 45 + .../twitter/visibility/rules/Condition.scala | 2401 +++++++++++ .../rules/DmConversationRules.scala | 50 + .../visibility/rules/DmEventRules.scala | 90 + .../rules/DmVisibilityPolicies.scala | 130 + .../visibility/rules/DownrankingRules.scala | 207 + .../visibility/rules/EvaluationContext.scala | 68 + .../visibility/rules/ExperimentBase.scala | 18 + .../rules/FailClosedException.scala | 41 + .../visibility/rules/FollowerRelations.scala | 20 + .../rules/ForEmergencyUseOnly.scala | 100 + .../rules/FreedomOfSpeechNotReach.scala | 705 +++ .../visibility/rules/InterstitialIf.scala | 43 + .../rules/PublicInterestRules.scala | 327 ++ .../com/twitter/visibility/rules/Rule.scala | 215 + .../rules/RuleActionSourceBuilder.scala | 97 + .../twitter/visibility/rules/RuleBase.scala | 238 ++ .../com/twitter/visibility/rules/Rules.scala | 315 ++ .../visibility/rules/SafeSearchRules.scala | 332 ++ .../visibility/rules/SearchBlenderRules.scala | 37 + .../rules/SensitiveMediaSettingsRules.scala | 277 ++ .../twitter/visibility/rules/SpaceRules.scala | 219 + .../visibility/rules/TombstoneIf.scala | 44 + .../rules/ToxicityReplyFilterRules.scala | 28 + .../visibility/rules/TweetLabelRules.scala | 862 ++++ .../twitter/visibility/rules/TweetRules.scala | 594 +++ .../visibility/rules/UserLabelRules.scala | 361 ++ .../UserUnavailableStateTombstoneRules.scala | 120 + .../visibility/rules/VisibilityPolicy.scala | 3778 +++++++++++++++++ .../twitter/visibility/rules/generators/BUILD | 37 + .../rules/generators/RuleGenerator.scala | 8 + .../rules/generators/TweetRuleGenerator.scala | 321 ++ .../generators/TweetVisibilityPolicy.scala | 74 + .../twitter/visibility/rules/package.scala | 5 + .../twitter/visibility/rules/providers/BUILD | 38 + .../providers/InjectedPolicyProvider.scala | 27 + .../rules/providers/PolicyProvider.scala | 8 + .../providers/ProvidedEvaluationContext.scala | 50 + .../com/twitter/visibility/rules/utils/BUILD | 38 + .../visibility/rules/utils/ShimUtils.scala | 60 + .../scala/com/twitter/visibility/util/BUILD | 18 + .../twitter/visibility/util/DeciderUtil.scala | 45 + .../visibility/util/FeatureSwitchUtil.scala | 22 + .../twitter/visibility/util/LoggingUtil.scala | 35 + .../twitter/visibility/util/NamingUtils.scala | 6 + 5364 files changed, 460239 insertions(+) create mode 100644 .gitignore create mode 100644 COPYING create mode 100644 README.md create mode 100644 ann/src/main/java/com/twitter/ann/faiss/BUILD create mode 100644 ann/src/main/java/com/twitter/ann/faiss/NativeUtils.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/AlignedTableFloat32.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/AlignedTableUint16.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/AlignedTableUint8.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/ArrayInvertedLists.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/AutoTuneCriterion.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/BUILD create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/BitstringReader.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/BitstringWriter.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/BufferList.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/ByteVector.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/ByteVectorVector.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/CenteringTransform.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/CharVector.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/Clustering.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/Clustering1D.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/ClusteringIterationStats.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/ClusteringParameters.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/DistanceComputer.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/DoubleVector.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/FloatVector.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/FloatVectorVector.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/GenHammingComputer16.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/GenHammingComputer32.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/GenHammingComputer8.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/GenHammingComputerM8.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/HNSW.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/HNSWStats.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/HStackInvertedLists.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/HammingComputer16.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/HammingComputer20.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/HammingComputer32.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/HammingComputer4.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/HammingComputer64.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/HammingComputer8.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/HammingComputerDefault.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/HammingComputerM4.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/HammingComputerM8.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/IDSelector.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/IDSelectorArray.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/IDSelectorBatch.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/IDSelectorRange.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/ITQMatrix.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/ITQTransform.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/IVFPQSearchParameters.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/IVFSearchParameters.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/Index.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/Index2Layer.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/IndexBinary.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/IndexBinaryFlat.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/IndexBinaryFromFloat.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/IndexBinaryHNSW.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/IndexBinaryIVF.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/IndexFlat.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/IndexFlat1D.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/IndexFlatCodes.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/IndexFlatIP.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/IndexFlatL2.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/IndexHNSW.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/IndexHNSW2Level.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/IndexHNSWFlat.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/IndexHNSWPQ.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/IndexHNSWSQ.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/IndexIDMap.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/IndexIVF.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/IndexIVFFlat.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/IndexIVFFlatDedup.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/IndexIVFPQ.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/IndexIVFPQStats.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/IndexIVFScalarQuantizer.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/IndexIVFStats.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/IndexLSH.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/IndexPQ.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/IndexPQStats.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/IndexRefine.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/IndexRefineFlat.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/IndexScalarQuantizer.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/IndexShards.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/IndexSplitVectors.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/IntVector.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/InterruptCallback.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/IntersectionCriterion.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/InvertedLists.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/InvertedListsPtrVector.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/Level1Quantizer.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/LinearTransform.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/LongVector.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/LongVectorVector.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/MapLong2Long.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/MaskedInvertedLists.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/MetricType.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/MultiIndexQuantizer.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/MultiIndexQuantizer2.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/NormalizationTransform.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/OPQMatrix.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/OnDiskInvertedLists.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/OnDiskInvertedListsIOHook.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/OnDiskOneList.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/OneRecallAtRCriterion.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/OperatingPoint.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/OperatingPointVector.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/OperatingPoints.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/PCAMatrix.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/PQDecoder16.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/PQDecoder8.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/PQDecoderGeneric.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/PQEncoder16.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/PQEncoder8.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/PQEncoderGeneric.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/ParameterRange.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/ParameterSpace.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/PartitionStats.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/PermutationObjective.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/PolysemousTraining.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/ProductQuantizer.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/ProgressiveDimClustering.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/ProgressiveDimClusteringParameters.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/ProgressiveDimIndexFactory.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/RandomRotationMatrix.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/RangeQueryResult.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/RangeSearchPartialResult.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/RangeSearchResult.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/ReadOnlyInvertedLists.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/ReconstructFromNeighbors.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/RemapDimensionsTransform.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/ReproduceDistancesObjective.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_AlignedTableT_float_32_t.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_AlignedTableT_float_t.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_DirectMap.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_DirectMap__Type.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_FILE.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_IOReader.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_IOWriter.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_ScalarQuantizer.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_ScalarQuantizer__QuantizerType.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_double.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_faiss__AlignedTableTightAllocT_float_32_t.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_faiss__AlignedTableTightAllocT_uint16_t_32_t.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_faiss__AlignedTableTightAllocT_unsigned_char_32_t.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_faiss__BinaryInvertedListScanner.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_faiss__HeapArrayT_faiss__CMaxT_float_int64_t_t_t.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_faiss__HeapArrayT_faiss__CMaxT_int_int64_t_t_t.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_faiss__HeapArrayT_faiss__CMinT_float_int64_t_t_t.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_faiss__IOReader.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_faiss__IOWriter.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_faiss__InvertedListScanner.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_faiss__LockLevels.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_faiss__OnDiskInvertedLists__OngoingPrefetch.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_faiss__RandomGenerator.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_float.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_int.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_long.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_long_long.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_omp_lock_t.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_p_faiss__Index.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_p_faiss__InvertedLists.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_p_faiss__VectorTransform.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_std__listT_faiss__OnDiskInvertedLists__Slot_t.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_std__pairT_float_int_t.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_std__priority_queueT_faiss__HNSW__NodeDistFarther_t.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_std__priority_queueT_std__pairT_float_int_t_t.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_std__unordered_mapT_long_long_t.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_std__unordered_multimapT_int64_t_int64_t_t.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_std__vectorT_faiss__BufferList__Buffer_t.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_std__vectorT_faiss__ClusteringIterationStats_t.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_std__vectorT_faiss__HNSW__NodeDistFarther_t.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_std__vectorT_faiss__Index_p_t.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_std__vectorT_faiss__InvertedLists_const_p_t.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_std__vectorT_faiss__OnDiskOneList_t.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_std__vectorT_faiss__ParameterRange_t.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_std__vectorT_faiss__RangeQueryResult_t.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_std__vectorT_faiss__RangeSearchPartialResult_p_t.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_std__vectorT_int64_t_t.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_std__vectorT_long_t.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_std__vectorT_omp_lock_t_t.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_std__vectorT_std__vectorT_int64_t_t_t.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_std__vectorT_std__vectorT_unsigned_long_t_t.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_uint16_t.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_uint32_t.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_unsigned_char.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_unsigned_long.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_void.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/SimulatedAnnealingOptimizer.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/SimulatedAnnealingParameters.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/SliceInvertedLists.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/SlidingIndexWindow.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/StopWordsInvertedLists.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/Uint64Vector.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/VStackInvertedLists.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/VectorTransform.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/VectorTransformVector.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/VisitedTable.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/doubleArray.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/floatArray.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/float_maxheap_array_t.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/float_minheap_array_t.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/intArray.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/int_maxheap_array_t.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/int_minheap_array_t.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/longArray.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/resources/.gitignore create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/resources/.gitkeep create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/resources/BUILD create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/swigfaiss.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/swigfaissConstants.java create mode 100644 ann/src/main/java/com/twitter/ann/faiss/swig/swigfaissJNI.java create mode 100644 ann/src/main/java/com/twitter/ann/hnsw/BUILD create mode 100644 ann/src/main/java/com/twitter/ann/hnsw/DistanceFunction.java create mode 100644 ann/src/main/java/com/twitter/ann/hnsw/DistancedItem.java create mode 100644 ann/src/main/java/com/twitter/ann/hnsw/DistancedItemQueue.java create mode 100644 ann/src/main/java/com/twitter/ann/hnsw/HnswIndex.java create mode 100644 ann/src/main/java/com/twitter/ann/hnsw/HnswIndexIOUtil.java create mode 100644 ann/src/main/java/com/twitter/ann/hnsw/HnswMeta.java create mode 100644 ann/src/main/java/com/twitter/ann/hnsw/HnswNode.java create mode 100644 ann/src/main/java/com/twitter/ann/hnsw/IllegalDuplicateInsertException.java create mode 100644 ann/src/main/python/dataflow/BUILD.bazel create mode 100644 ann/src/main/python/dataflow/bq.sql create mode 100644 ann/src/main/python/dataflow/faiss_index_bq_dataset.py create mode 100644 ann/src/main/python/dataflow/worker_harness/Dockerfile create mode 100644 ann/src/main/python/dataflow/worker_harness/cloudbuild.yml create mode 100644 ann/src/main/scala/com/twitter/ann/annoy/AnnoyCommon.scala create mode 100644 ann/src/main/scala/com/twitter/ann/annoy/BUILD create mode 100644 ann/src/main/scala/com/twitter/ann/annoy/RawAnnoyIndexBuilder.scala create mode 100644 ann/src/main/scala/com/twitter/ann/annoy/RawAnnoyQueryIndex.scala create mode 100644 ann/src/main/scala/com/twitter/ann/annoy/TypedAnnoyIndex.scala create mode 100644 ann/src/main/scala/com/twitter/ann/annoy/TypedAnnoyIndexBuilderWithFile.scala create mode 100644 ann/src/main/scala/com/twitter/ann/annoy/TypedAnnoyQueryIndexWithFile.scala create mode 100644 ann/src/main/scala/com/twitter/ann/brute_force/BUILD create mode 100644 ann/src/main/scala/com/twitter/ann/brute_force/BruteForceDeserialization.scala create mode 100644 ann/src/main/scala/com/twitter/ann/brute_force/BruteForceIndex.scala create mode 100644 ann/src/main/scala/com/twitter/ann/common/AnnInjections.scala create mode 100644 ann/src/main/scala/com/twitter/ann/common/Api.scala create mode 100644 ann/src/main/scala/com/twitter/ann/common/BUILD create mode 100644 ann/src/main/scala/com/twitter/ann/common/EmbeddingProducer.scala create mode 100644 ann/src/main/scala/com/twitter/ann/common/IndexOutputFile.scala create mode 100644 ann/src/main/scala/com/twitter/ann/common/IndexTransformer.scala create mode 100644 ann/src/main/scala/com/twitter/ann/common/MemoizedInEpochs.scala create mode 100644 ann/src/main/scala/com/twitter/ann/common/Metric.scala create mode 100644 ann/src/main/scala/com/twitter/ann/common/QueryableById.scala create mode 100644 ann/src/main/scala/com/twitter/ann/common/QueryableByIdImplementation.scala create mode 100644 ann/src/main/scala/com/twitter/ann/common/QueryableOperations.scala create mode 100644 ann/src/main/scala/com/twitter/ann/common/ReadWriteFuturePool.scala create mode 100644 ann/src/main/scala/com/twitter/ann/common/Serialization.scala create mode 100644 ann/src/main/scala/com/twitter/ann/common/ServiceClientQueryable.scala create mode 100644 ann/src/main/scala/com/twitter/ann/common/ShardApi.scala create mode 100644 ann/src/main/scala/com/twitter/ann/common/ShardedSerialization.scala create mode 100644 ann/src/main/scala/com/twitter/ann/common/Task.scala create mode 100644 ann/src/main/scala/com/twitter/ann/dataflow/offline/ANNIndexBuilderBeamJob.scala create mode 100644 ann/src/main/scala/com/twitter/ann/dataflow/offline/BUILD create mode 100644 ann/src/main/scala/com/twitter/ann/dataflow/offline/BaseEmbeddingData.scala create mode 100644 ann/src/main/scala/com/twitter/ann/dataflow/offline/FlatEmbeddingData.scala create mode 100644 ann/src/main/scala/com/twitter/ann/dataflow/offline/GroupedEmbeddingData.scala create mode 100644 ann/src/main/scala/com/twitter/ann/experimental/BUILD.bazel create mode 100644 ann/src/main/scala/com/twitter/ann/experimental/Runner.scala create mode 100644 ann/src/main/scala/com/twitter/ann/faiss/BUILD create mode 100644 ann/src/main/scala/com/twitter/ann/faiss/FaissCommon.scala create mode 100644 ann/src/main/scala/com/twitter/ann/faiss/FaissIndex.scala create mode 100644 ann/src/main/scala/com/twitter/ann/faiss/FaissIndexer.scala create mode 100644 ann/src/main/scala/com/twitter/ann/faiss/HourlyDirectoryWithSuccessFileListing.scala create mode 100644 ann/src/main/scala/com/twitter/ann/faiss/HourlyShardedIndex.scala create mode 100644 ann/src/main/scala/com/twitter/ann/faiss/QueryableIndexAdapter.scala create mode 100644 ann/src/main/scala/com/twitter/ann/featurestore/BUILD create mode 100644 ann/src/main/scala/com/twitter/ann/featurestore/FeatureStoreEmbeddingProducer.scala create mode 100644 ann/src/main/scala/com/twitter/ann/file_store/BUILD create mode 100644 ann/src/main/scala/com/twitter/ann/file_store/ReadableIndexIdFileStore.scala create mode 100644 ann/src/main/scala/com/twitter/ann/file_store/WritableIndexIdFileStore.scala create mode 100644 ann/src/main/scala/com/twitter/ann/hnsw/BUILD create mode 100644 ann/src/main/scala/com/twitter/ann/hnsw/DistanceFunctionGenerator.scala create mode 100644 ann/src/main/scala/com/twitter/ann/hnsw/Hnsw.scala create mode 100644 ann/src/main/scala/com/twitter/ann/hnsw/HnswCommon.scala create mode 100644 ann/src/main/scala/com/twitter/ann/hnsw/HnswIOUtil.scala create mode 100644 ann/src/main/scala/com/twitter/ann/hnsw/IdEmbeddingMap.scala create mode 100644 ann/src/main/scala/com/twitter/ann/hnsw/JMapBasedIdEmbeddingMap.scala create mode 100644 ann/src/main/scala/com/twitter/ann/hnsw/MapDbBasedIdEmbeddingMap.scala create mode 100644 ann/src/main/scala/com/twitter/ann/hnsw/SerializableHnsw.scala create mode 100644 ann/src/main/scala/com/twitter/ann/hnsw/TypedHnswIndex.scala create mode 100644 ann/src/main/scala/com/twitter/ann/manhattan/BUILD create mode 100644 ann/src/main/scala/com/twitter/ann/manhattan/ManhattanEmbeddingProducer.scala create mode 100644 ann/src/main/scala/com/twitter/ann/manhattan/README create mode 100644 ann/src/main/scala/com/twitter/ann/scalding/benchmark/BUILD create mode 100644 ann/src/main/scala/com/twitter/ann/scalding/benchmark/Knn.scala create mode 100644 ann/src/main/scala/com/twitter/ann/scalding/offline/BUILD.bazel create mode 100644 ann/src/main/scala/com/twitter/ann/scalding/offline/IndexingStrategy.scala create mode 100644 ann/src/main/scala/com/twitter/ann/scalding/offline/KnnDebug.scala create mode 100644 ann/src/main/scala/com/twitter/ann/scalding/offline/KnnEntityRecoDebugJob.scala create mode 100644 ann/src/main/scala/com/twitter/ann/scalding/offline/KnnHelper.scala create mode 100644 ann/src/main/scala/com/twitter/ann/scalding/offline/KnnOfflineJob.scala create mode 100644 ann/src/main/scala/com/twitter/ann/scalding/offline/KnnTruthSetGenerator.scala create mode 100644 ann/src/main/scala/com/twitter/ann/scalding/offline/ParameterlessQueryable.scala create mode 100644 ann/src/main/scala/com/twitter/ann/scalding/offline/README create mode 100644 ann/src/main/scala/com/twitter/ann/scalding/offline/faissindexbuilder/BUILD.bazel create mode 100644 ann/src/main/scala/com/twitter/ann/scalding/offline/faissindexbuilder/IndexBuilder.scala create mode 100644 ann/src/main/scala/com/twitter/ann/scalding/offline/faissindexbuilder/IndexBuilderApp.scala create mode 100644 ann/src/main/scala/com/twitter/ann/scalding/offline/indexbuilder/BUILD.bazel create mode 100644 ann/src/main/scala/com/twitter/ann/scalding/offline/indexbuilder/IndexBuilder.scala create mode 100644 ann/src/main/scala/com/twitter/ann/scalding/offline/indexbuilder/IndexBuilderApp.scala create mode 100644 ann/src/main/scala/com/twitter/ann/scalding/offline/indexbuilder/README.rst create mode 100644 ann/src/main/scala/com/twitter/ann/scalding/offline/indexbuilderfrombq/BUILD.bazel create mode 100644 ann/src/main/scala/com/twitter/ann/scalding/offline/indexbuilderfrombq/IndexBuilderFromBQ.scala create mode 100644 ann/src/main/scala/com/twitter/ann/scalding/offline/indexbuilderfrombq/IndexBuilderFromBQApp.scala create mode 100644 ann/src/main/scala/com/twitter/ann/serialization/BUILD create mode 100644 ann/src/main/scala/com/twitter/ann/serialization/DummyANNIndexInjection.scala create mode 100644 ann/src/main/scala/com/twitter/ann/serialization/PersistedEmbeddingInjection.scala create mode 100644 ann/src/main/scala/com/twitter/ann/serialization/ThriftIteratorIO.scala create mode 100644 ann/src/main/scala/com/twitter/ann/service/loadtest/AnnLoadTest.scala create mode 100644 ann/src/main/scala/com/twitter/ann/service/loadtest/AnnLoadTestMain.scala create mode 100644 ann/src/main/scala/com/twitter/ann/service/loadtest/AnnLoadTestWorker.scala create mode 100644 ann/src/main/scala/com/twitter/ann/service/loadtest/BUILD create mode 100644 ann/src/main/scala/com/twitter/ann/service/loadtest/EmbeddingIndexer.scala create mode 100644 ann/src/main/scala/com/twitter/ann/service/loadtest/LoadTestRecorder.scala create mode 100644 ann/src/main/scala/com/twitter/ann/service/loadtest/LoadTestUtils.scala create mode 100644 ann/src/main/scala/com/twitter/ann/service/loadtest/README.md create mode 100644 ann/src/main/scala/com/twitter/ann/service/query_server/common/BUILD create mode 100644 ann/src/main/scala/com/twitter/ann/service/query_server/common/BaseQueryIndexServer.scala create mode 100644 ann/src/main/scala/com/twitter/ann/service/query_server/common/Exceptions.scala create mode 100644 ann/src/main/scala/com/twitter/ann/service/query_server/common/FaissIndexPathProvider.scala create mode 100644 ann/src/main/scala/com/twitter/ann/service/query_server/common/IndexPathProvider.scala create mode 100644 ann/src/main/scala/com/twitter/ann/service/query_server/common/QueryIndexThriftController.scala create mode 100644 ann/src/main/scala/com/twitter/ann/service/query_server/common/QueryServerUtil.scala create mode 100644 ann/src/main/scala/com/twitter/ann/service/query_server/common/QueryableProvider.scala create mode 100644 ann/src/main/scala/com/twitter/ann/service/query_server/common/RefreshableQueryable.scala create mode 100644 ann/src/main/scala/com/twitter/ann/service/query_server/common/UnsafeQueryIndexServer.scala create mode 100644 ann/src/main/scala/com/twitter/ann/service/query_server/common/throttling/AuroraCPUStatsReader.scala create mode 100644 ann/src/main/scala/com/twitter/ann/service/query_server/common/throttling/BUILD create mode 100644 ann/src/main/scala/com/twitter/ann/service/query_server/common/throttling/ThrottlingBasedQualityTask.scala create mode 100644 ann/src/main/scala/com/twitter/ann/service/query_server/common/throttling/WindowedStats.scala create mode 100644 ann/src/main/scala/com/twitter/ann/service/query_server/common/throttling/WindowedThrottlingInstrument.scala create mode 100644 ann/src/main/scala/com/twitter/ann/service/query_server/common/warmup/BUILD create mode 100644 ann/src/main/scala/com/twitter/ann/service/query_server/common/warmup/Warmup.scala create mode 100644 ann/src/main/scala/com/twitter/ann/service/query_server/faiss/BUILD create mode 100644 ann/src/main/scala/com/twitter/ann/service/query_server/faiss/FaissQueryIndexServer.scala create mode 100644 ann/src/main/scala/com/twitter/ann/service/query_server/hnsw/BUILD create mode 100644 ann/src/main/scala/com/twitter/ann/service/query_server/hnsw/HnswQueryIndexServer.scala create mode 100644 ann/src/main/scala/com/twitter/ann/util/BUILD create mode 100644 ann/src/main/scala/com/twitter/ann/util/IndexBuilderUtils.scala create mode 100644 ann/src/main/thrift/com/twitter/ann/common/BUILD create mode 100644 ann/src/main/thrift/com/twitter/ann/common/ann_common.thrift create mode 100644 ann/src/main/thrift/com/twitter/ann/knn/BUILD create mode 100644 ann/src/main/thrift/com/twitter/ann/knn/knn.thrift create mode 100644 ann/src/main/thrift/com/twitter/ann/serialization/BUILD create mode 100644 ann/src/main/thrift/com/twitter/ann/serialization/serialization.thrift create mode 100755 ci/ci.sh create mode 100644 cr-mixer/BUILD.bazel create mode 100644 cr-mixer/README.md create mode 100644 cr-mixer/server/src/main/resources/BUILD.bazel create mode 100644 cr-mixer/server/src/main/resources/config/decider.yml create mode 100644 cr-mixer/server/src/main/resources/logback.xml create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/BUILD.bazel create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/CrMixerHttpServerWarmupHandler.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/CrMixerServer.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/CrMixerThriftServerWarmupHandler.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/blender/AdsBlender.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/blender/BUILD create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/blender/BlendedCandidatesBuilder.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/blender/ContentSignalBlender.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/blender/CountWeightedInterleaveBlender.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/blender/InterleaveBlender.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/blender/SourceTypeBackFillBlender.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/blender/SwitchBlender.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/candidate_generation/AdsCandidateGenerator.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/candidate_generation/AdsCandidateSourcesRouter.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/candidate_generation/BUILD create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/candidate_generation/CandidateSourcesRouter.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/candidate_generation/CrCandidateGenerator.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/candidate_generation/CustomizedRetrievalCandidateGeneration.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/candidate_generation/FrsTweetCandidateGenerator.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/candidate_generation/RelatedTweetCandidateGenerator.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/candidate_generation/RelatedVideoTweetCandidateGenerator.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/candidate_generation/SimClustersInterestedInCandidateGeneration.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/candidate_generation/TopicTweetCandidateGenerator.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/candidate_generation/UtegTweetCandidateGenerator.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/config/BUILD create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/config/SimClustersANNConfig.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/config/TimeoutConfig.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/controller/BUILD.bazel create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/controller/CrMixerThriftController.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/exception/BUILD create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/exception/InvalidSANNConfigException.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/featureswitch/BUILD create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/featureswitch/CrMixerLoggingABDecider.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/featureswitch/ParamsBuilder.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/featureswitch/SetImpressedBucketsLocalContextFilter.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/filter/BUILD create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/filter/FilterBase.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/filter/ImpressedTweetlistFilter.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/filter/InNetworkFilter.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/filter/PostRankFilterRunner.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/filter/PreRankFilterRunner.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/filter/ReplyFilter.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/filter/RetweetFilter.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/filter/TweetAgeFilter.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/filter/TweetInfoHealthFilterBase.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/filter/UtegFilterRunner.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/filter/UtegHealthFilter.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/filter/VideoTweetFilter.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/logging/AdsRecommendationsScribeLogger.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/logging/BUILD create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/logging/CrMixerScribeLogger.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/logging/RelatedTweetScribeLogger.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/logging/ScribeLoggerUtils.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/logging/ScribeMetadata.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/logging/TopLevelDdgMetricsMetadata.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/logging/UtegTweetScribeLogger.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/model/BUILD create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/model/Candidate.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/model/CandidateGenerationInfo.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/model/CandidateGeneratorQuery.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/model/EarlybirdSimilarityEngineType.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/model/HealthThreshold.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/model/ModelConfig.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/model/ModuleNames.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/model/TopicTweetWithScore.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/model/TweetWithAuthor.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/model/TweetWithScore.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/model/TweetWithScoreAndSocialProof.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/ActivePromotedTweetStoreModule.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/BUILD.bazel create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/BlueVerifiedAnnotationStoreModule.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/CertoStratoStoreModule.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/ConsumersBasedUserAdGraphStoreModule.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/ConsumersBasedUserTweetGraphStoreModule.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/ConsumersBasedUserVideoGraphStoreModule.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/CrMixerParamConfigModule.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/DiffusionStoreModule.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/EarlybirdRecencyBasedCandidateStoreModule.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/EmbeddingStoreModule.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/FrsStoreModule.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/MHMtlsParamsModule.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/OfflineCandidateStoreModule.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/RealGraphOonStoreModule.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/RealGraphStoreMhModule.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/RepresentationManagerModule.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/RepresentationScorerModule.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/SampleSimilarityEngineModule.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/SimClustersANNServiceNameToClientMapper.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/SkitStratoStoreModule.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/StrongTiePredictionStoreModule.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/TripCandidateStoreModule.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/TweetInfoStoreModule.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/TweetRecentEngagedUserStoreModule.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/TweetRecommendationResultsStoreModule.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/TwhinCollabFilterStratoStoreModule.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/TwiceClustersMembersStoreModule.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/UnifiedCacheClient.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/UserSignalServiceColumnModule.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/UserSignalServiceStoreModule.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/UserStateStoreModule.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/core/ABDeciderModule.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/core/CrMixerFlagModule.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/core/CrMixerLoggingABDeciderModule.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/core/FeatureContextBuilderModule.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/core/FeatureSwitchesModule.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/core/KafkaProducerModule.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/core/LoggerFactoryModule.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/core/MemoizingStatsReceiverModule.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/core/TimeoutConfigModule.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/grpc_client/NaviGRPCClientModule.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/CertoTopicTweetSimilarityEngineModule.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/ConsumerBasedWalsSimilarityEngineModule.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/ConsumerEmbeddingBasedTripSimilarityEngineModule.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/ConsumerEmbeddingBasedTwHINSimilarityEngineModule.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/ConsumerEmbeddingBasedTwoTowerSimilarityEngineModule.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/ConsumersBasedUserAdGraphSimilarityEngineModule.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/ConsumersBasedUserVideoGraphSimilarityEngineModule.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/DiffusionBasedSimilarityEngineModule.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/EarlybirdSimilarityEngineModule.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/ProducerBasedUnifiedSimilarityEngineModule.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/ProducerBasedUserAdGraphSimilarityEngineModule.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/ProducerBasedUserTweetGraphSimilarityEngineModule.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/SimClustersANNSimilarityEngineModule.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/SkitTopicTweetSimilarityEngineModule.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/TweetBasedQigSimilarityEngineModule.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/TweetBasedTwHINSimlarityEngineModule.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/TweetBasedUnifiedSimilarityEngineModule.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/TweetBasedUserAdGraphSimilarityEngineModule.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/TweetBasedUserTweetGraphSimilarityEngineModule.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/TweetBasedUserVideoGraphSimilarityEngineModule.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/TwhinCollabFilterLookupSimilarityEngineModule.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/UserTweetEntityGraphSimilarityEngineModule.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/thrift_client/AnnQueryServiceClientModule.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/thrift_client/EarlybirdSearchClientModule.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/thrift_client/FrsClientModule.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/thrift_client/HydraPartitionClientModule.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/thrift_client/HydraRootClientModule.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/thrift_client/QigServiceClientModule.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/thrift_client/SimClustersAnnServiceClientModule.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/thrift_client/TweetyPieClientModule.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/thrift_client/UserAdGraphClientModule.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/thrift_client/UserTweetEntityGraphClientModule.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/thrift_client/UserTweetGraphClientModule.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/thrift_client/UserTweetGraphPlusClientModule.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/thrift_client/UserVideoGraphClientModule.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/AdsParams.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/BUILD create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/BlenderParams.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/BypassInterleaveAndRankParams.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/ConsumerBasedWalsParams.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/ConsumerEmbeddingBasedCandidateGenerationParams.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/ConsumerEmbeddingBasedTripParams.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/ConsumerEmbeddingBasedTwHINParams.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/ConsumerEmbeddingBasedTwoTowerParams.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/ConsumersBasedUserAdGraphParams.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/ConsumersBasedUserTweetGraphParams.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/ConsumersBasedUserVideoGraphParams.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/CrMixerParamConfig.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/CustomizedRetrievalBasedCandidateGenerationParams.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/CustomizedRetrievalBasedFTROfflineInterestedInParams.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/CustomizedRetrievalBasedOfflineInterestedInParams.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/CustomizedRetrievalBasedTwhinParams.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/EarlybirdFrsBasedCandidateGenerationParams.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/FrsParams.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/GlobalParams.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/GoodProfileClickParams.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/GoodTweetClickParams.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/InterestedInParams.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/ProducerBasedCandidateGenerationParams.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/ProducerBasedUserAdGraphParams.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/ProducerBasedUserTweetGraphParams.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/RankerParams.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/RealGraphInParams.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/RealGraphOonParams.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/RecentFollowsParams.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/RecentNegativeSignalParams.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/RecentNotificationsParams.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/RecentOriginalTweetsParams.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/RecentReplyTweetsParams.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/RecentRetweetsParams.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/RecentTweetFavoritesParams.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/RelatedTweetGlobalParams.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/RelatedTweetProducerBasedParams.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/RelatedTweetTweetBasedParams.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/RelatedVideoTweetGlobalParams.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/RelatedVideoTweetTweetBasedParams.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/RepeatedProfileVisitsParams.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/SimClustersANNParams.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/TopicTweetParams.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/TweetBasedCandidateGenerationParams.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/TweetBasedTwHINParams.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/TweetBasedUserAdGraphParams.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/TweetBasedUserTweetGraphParams.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/TweetBasedUserVideoGraphParams.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/TweetSharesParams.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/UnifiedSETweetCombinationMethod.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/UnifiedUSSSignalParams.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/UtegTweetGlobalParams.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/VideoTweetFilterParams.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/VideoViewTweetsParams.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/decider/BUILD create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/decider/CrMixerDecider.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/decider/DeciderKey.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/decider/EndpointLoadShedder.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/ranker/BUILD create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/ranker/DefaultRanker.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/ranker/SwitchRanker.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/scribe/BUILD create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/scribe/ScribeCategory.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/service/BUILD.bazel create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/service/CrMixerAlertNotificationConfig.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/BUILD create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/CertoTopicTweetSimilarityEngine.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/ConsumerBasedWalsSimilarityEngine.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/ConsumerEmbeddingBasedTripSimilarityEngine.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/ConsumerEmbeddingBasedTwHINSimilarityEngine.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/ConsumerEmbeddingBasedTwoTowerSimilarityEngine.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/ConsumersBasedUserAdGraphSimilarityEngine.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/ConsumersBasedUserVideoGraphSimilarityEngine.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/DiffusionBasedSimilarityEngine.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/EarlybirdModelBasedSimilarityEngine.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/EarlybirdRecencyBasedSimilarityEngine.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/EarlybirdSimilarityEngine.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/EarlybirdSimilarityEngineBase.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/EarlybirdSimilarityEngineRouter.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/EarlybirdTensorflowBasedSimilarityEngine.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/FilterUtil.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/HnswANNSimilarityEngine.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/LookupSimilarityEngine.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/ModelBasedANNStore.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/ProducerBasedUnifiedSimilarityEngine.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/ProducerBasedUserAdGraphSimilarityEngine.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/ProducerBasedUserTweetGraphSimilarityEngine.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/SimClustersANNSimilarityEngine.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/SimilarityEngine.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/SimilaritySourceOrderingUtil.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/SkitHighPrecisionTopicTweetSimilarityEngine.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/SkitTopicTweetSimilarityEngine.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/StandardSimilarityEngine.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/TweetBasedQigSimilarityEngine.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/TweetBasedUnifiedSimilarityEngine.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/TweetBasedUserAdGraphSimilarityEngine.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/TweetBasedUserTweetGraphSimilarityEngine.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/TweetBasedUserVideoGraphSimilarityEngine.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/TwhinCollabFilterSimilarityEngine.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/UserTweetEntityGraphSimilarityEngine.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/source_signal/BUILD create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/source_signal/FrsSourceGraphFetcher.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/source_signal/FrsSourceSignalFetcher.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/source_signal/FrsStore.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/source_signal/RealGraphInSourceGraphFetcher.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/source_signal/RealGraphOonSourceGraphFetcher.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/source_signal/SourceFetcher.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/source_signal/SourceGraphFetcher.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/source_signal/SourceInfoRouter.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/source_signal/SourceSignalFetcher.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/source_signal/UssSourceSignalFetcher.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/source_signal/UssStore.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/util/BUILD create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/util/CandidateGenerationKeyUtil.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/util/CountWeightedInterleaveUtil.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/util/EarlybirdSearchUtil.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/util/InterleaveUtil.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/util/MetricTagUtil.scala create mode 100644 cr-mixer/server/src/main/scala/com/twitter/cr_mixer/util/SignalTimestampStatsUtil.scala create mode 100644 cr-mixer/thrift/src/main/thrift/BUILD create mode 100644 cr-mixer/thrift/src/main/thrift/ads.thrift create mode 100644 cr-mixer/thrift/src/main/thrift/candidate_generation_key.thrift create mode 100644 cr-mixer/thrift/src/main/thrift/cr_mixer.thrift create mode 100644 cr-mixer/thrift/src/main/thrift/frs_based_tweet.thrift create mode 100644 cr-mixer/thrift/src/main/thrift/metric_tags.thrift create mode 100644 cr-mixer/thrift/src/main/thrift/product.thrift create mode 100644 cr-mixer/thrift/src/main/thrift/product_context.thrift create mode 100644 cr-mixer/thrift/src/main/thrift/related_tweet.thrift create mode 100644 cr-mixer/thrift/src/main/thrift/related_video_tweet.thrift create mode 100644 cr-mixer/thrift/src/main/thrift/scribe.thrift create mode 100644 cr-mixer/thrift/src/main/thrift/source_type.thrift create mode 100644 cr-mixer/thrift/src/main/thrift/topic_tweet.thrift create mode 100644 cr-mixer/thrift/src/main/thrift/uteg.thrift create mode 100644 cr-mixer/thrift/src/main/thrift/validation.thrift create mode 100644 docs/system-diagram.png create mode 100644 follow-recommendations-service/BUILD create mode 100644 follow-recommendations-service/CONFIG.ini create mode 100644 follow-recommendations-service/FRS_architecture.png create mode 100644 follow-recommendations-service/README.md create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base/BUILD create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base/CandidateSourceRegistry.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base/EnrichedCandidateSource.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base/ParamPredicate.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base/Predicate.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base/PredicateResult.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base/Ranker.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base/RecommendationFlow.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base/SideEffectsUtil.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base/StatsUtil.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base/Transform.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/addressbook/AddressBookParams.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/addressbook/BUILD create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/addressbook/ForwardEmailBookSource.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/addressbook/ForwardPhoneBookSource.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/addressbook/README.md create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/addressbook/ReverseEmailBookSource.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/addressbook/ReversePhoneBookSource.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/base/BUILD create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/base/CachedCandidateSource.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/base/ExperimentalCandidateSource.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/base/RealGraphExpansionRepository.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/base/SimilarUserExpanderParams.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/base/SimilarUserExpanderRepository.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/base/SocialProofEnforcedCandidateSource.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/base/SocialProofEnforcedCandidateSourceFSConfig.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/base/SocialProofEnforcedCandidateSourceParams.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/base/StratoFetcherSource.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/base/StratoFetcherWithUnitViewSource.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/base/TweetAuthorsCandidateSource.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/base/TwoHopExpansionCandidateSource.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/crowd_search_accounts/BUILD create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/crowd_search_accounts/CrowdSearchAccountsFSConfig.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/crowd_search_accounts/CrowdSearchAccountsParams.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/crowd_search_accounts/CrowdSearchAccountsSource.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/crowd_search_accounts/README.md create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/geo/BUILD create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/geo/BasePopGeoHashSource.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/geo/PopCountryBackFillSource.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/geo/PopCountrySource.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/geo/PopGeoQualityFollowSource.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/geo/PopGeoQualityFollowSourceFSConfig.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/geo/PopGeoQualityFollowSourceParams.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/geo/PopGeoSource.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/geo/PopGeoSourceFSConfig.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/geo/PopGeoSourceParams.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/geo/PopGeohashSource.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/geo/README.md create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/ppmi_locale_follow/BUILD create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/ppmi_locale_follow/PPMILocaleFollowSource.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/ppmi_locale_follow/PPMILocaleFollowSourceFSConfig.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/ppmi_locale_follow/PPMILocaleFollowSourceParams.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/ppmi_locale_follow/README.md create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/promoted_accounts/BUILD create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/promoted_accounts/PromotedAccountsCandidateSource.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/promoted_accounts/README.md create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/real_graph/BUILD create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/real_graph/README.md create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/real_graph/RealGraphOonFSConfig.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/real_graph/RealGraphOonParams.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/real_graph/RealGraphOonV2Source.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/real_graph/RealGraphSource.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/recent_engagement/BUILD create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/recent_engagement/README.md create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/recent_engagement/RecentEngagementDirectFollowSource.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/recent_engagement/RecentEngagementNonDirectFollowSource.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/recent_engagement/RepeatedProfileVisitsFSConfig.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/recent_engagement/RepeatedProfileVisitsParams.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/recent_engagement/RepeatedProfileVisitsSource.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/salsa/BUILD create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/salsa/README.md create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/salsa/RecentEngagementDirectFollowSalsaExpansionSource.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/salsa/SalsaExpander.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/salsa/SalsaExpansionBasedCandidateSource.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims/BUILD create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims/CacheBasedSimsStore.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims/DBV2SimsRefreshStore.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims/DBV2SimsStore.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims/Follow2vecNearestNeighborsStore.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims/README.md create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims/SimsExperimentalStore.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims/SimsSourceFSConfig.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims/SimsSourceParams.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims/SimsStore.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims/StratoBasedSimsCandidateSource.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims/StratoBasedSimsCandidateSourceWithUnitView.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims/SwitchingSimsSource.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims_expansion/BUILD create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims_expansion/DBV2SimsExpansionParams.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims_expansion/README.md create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims_expansion/RecentEngagementSimilarUsersFSConfig.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims_expansion/RecentEngagementSimilarUsersParams.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims_expansion/RecentEngagementSimilarUsersSource.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims_expansion/RecentFollowingSimilarUsersParams.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims_expansion/RecentFollowingSimilarUsersSource.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims_expansion/RecentStrongEngagementDirectFollowSimilarUsersSource.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims_expansion/SimsExpansionBasedCandidateSource.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims_expansion/SimsExpansionFSConfig.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims_expansion/SimsExpansionSourceParams.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/socialgraph/BUILD create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/socialgraph/README.md create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/socialgraph/RecentFollowingRecentFollowingExpansionSource.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/socialgraph/RecentFollowingRecentFollowingExpansionSourceFSConfig.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/socialgraph/RecentFollowingRecentFollowingExpansionSourceParams.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/BUILD create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/BaseOnlineSTPSource.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/Dbv2StpScorer.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/EpStpScorer.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/MutualFollowStrongTiePredictionSource.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/OfflineMutualFollowExpansionSource.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/OfflineStpSourceFsConfig.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/OfflineStpSourceParams.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/OfflineStpSourceWithDensePmiMatrix.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/OfflineStpSourceWithLegacyPmiMatrix.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/OfflineStrongTiePredictionBaseSource.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/OfflineStrongTiePredictionSource.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/OnlineSTPSourceFSConfig.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/OnlineSTPSourceParams.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/OnlineSTPSourceScorer.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/OnlineSTPSourceWithDeepbirdV2Scorer.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/OnlineSTPSourceWithEPScorer.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/README.md create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/STPFirstDegreeFetcher.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/STPGraphBuilder.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/STPSecondDegreeFetcher.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/SocialProofEnforcedOfflineStrongTiePredictionSource.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/img.png create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/top_organic_follows_accounts/BUILD create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/top_organic_follows_accounts/README.md create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/top_organic_follows_accounts/TopOrganicFollowsAccountsFSConfig.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/top_organic_follows_accounts/TopOrganicFollowsAccountsParams.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/top_organic_follows_accounts/TopOrganicFollowsAccountsSource.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/triangular_loops/BUILD create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/triangular_loops/README.md create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/triangular_loops/TriangularLoopsFSConfig.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/triangular_loops/TriangularLoopsParams.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/triangular_loops/TriangularLoopsSource.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/two_hop_random_walk/BUILD create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/two_hop_random_walk/README.md create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/two_hop_random_walk/TwoHopRandomWalkSource.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/user_user_graph/BUILD create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/user_user_graph/README.md create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/user_user_graph/UserUserGraphCandidateSource.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/user_user_graph/UserUserGraphFSConfig.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/user_user_graph/UserUserGraphParams.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/addressbook/AddressbookClient.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/addressbook/AddressbookModule.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/addressbook/BUILD create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/addressbook/models/BUILD create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/addressbook/models/Contact.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/addressbook/models/EdgeType.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/addressbook/models/QueryOption.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/addressbook/models/RecordIdentifier.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/adserver/AdRequest.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/adserver/AdserverClient.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/adserver/AdserverModule.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/adserver/BUILD create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/cache/BUILD create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/cache/MemcacheClient.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/cache/MemcacheModule.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/cache/ThriftBijection.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/common/BUILD create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/common/BaseClientModule.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/deepbirdv2/BUILD create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/deepbirdv2/DeepBirdV2PredictionServiceClientModule.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/dismiss_store/BUILD create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/dismiss_store/DismissStore.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/email_storage_service/BUILD create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/email_storage_service/EmailStorageServiceClient.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/email_storage_service/EmailStorageServiceModule.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/geoduck/BUILD create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/geoduck/LocationServiceClient.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/geoduck/LocationServiceModule.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/geoduck/ReverseGeocodeClient.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/geoduck/UserLocationFetcher.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/gizmoduck/BUILD create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/gizmoduck/GizmoduckClient.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/gizmoduck/GizmoduckModule.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/graph_feature_service/BUILD create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/graph_feature_service/GraphFeatureServiceClient.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/graph_feature_service/GraphFeatureStoreModule.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/impression_store/BUILD create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/impression_store/ImpressionStoreModule.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/impression_store/WtfImpressionStore.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/interests_service/BUILD create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/interests_service/InterestServiceClient.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/phone_storage_service/BUILD create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/phone_storage_service/PhoneStorageServiceClient.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/phone_storage_service/PhoneStorageServiceModule.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/real_time_real_graph/BUILD create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/real_time_real_graph/Engagement.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/real_time_real_graph/EngagementScorer.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/real_time_real_graph/RealTimeRealGraphClient.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/socialgraph/BUILD create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/socialgraph/SocialGraphClient.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/socialgraph/SocialGraphModule.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/strato/BUILD create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/strato/StratoClientModule.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/user_state/BUILD create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/user_state/UserStateClient.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/constants/BUILD create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/constants/CandidateAlgorithmTypeConstants.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/constants/GuiceNamedConstants.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/constants/ServiceConstants.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/adapters/BUILD create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/adapters/CandidateAlgorithmAdapter.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/adapters/ClientContextAdapter.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/adapters/PostNuxAlgorithmAdapter.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/adapters/PreFetchedFeatureAdapter.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/common/BUILD create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/common/FeatureSource.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/common/FeatureSourceId.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/common/HasPreFetchedFeature.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/sources/BUILD create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/sources/CandidateAlgorithmSource.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/sources/ClientContextSource.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/sources/FeatureHydrationSourcesFSConfig.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/sources/FeatureHydrationSourcesFeatureSwitchKeys.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/sources/FeatureStoreFeatures.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/sources/FeatureStoreGizmoduckSource.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/sources/FeatureStoreParameters.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/sources/FeatureStorePostNuxAlgorithmSource.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/sources/FeatureStoreSource.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/sources/FeatureStoreSourceParams.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/sources/FeatureStoreTimelinesAuthorSource.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/sources/FeatureStoreUserMetricCountsSource.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/sources/HydrationSourcesModule.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/sources/PreFetchedFeatureSource.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/sources/UserScoringFeatureSource.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/sources/Utils.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/features/BUILD create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/features/LocationFeature.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/features/TrackingTokenFeature.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/features/UserStateFeature.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/AddressBookMetadata.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/AlgorithmType.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/BUILD create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/CandidateUser.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/ClientContextConverter.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/DisplayLocation.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/EngagementType.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/FilterReason.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/FlowContext.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/FlowRecommendation.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/GeohashAndCountryCode.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasAdMetadata.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasByfSeedUserIds.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasDataRecord.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasDebugOptions.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasDismissedUserIds.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasDisplayLocation.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasEngagements.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasExcludedUserIds.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasGeohashAndCountryCode.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasInfoPerRankingStage.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasInterestIds.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasInvalidRelationshipUserIds.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasIsSoftUser.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasMutualFollowedUserIds.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasPreviousRecommendationsContext.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasProfileId.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasQualityFactor.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasRecentFollowedByUserIds.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasRecentFollowedUserIds.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasRecentFollowedUserIdsWithTime.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasRecentlyEngagedUserIds.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasRecommendationFlowIdentifier.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasScores.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasSimilarToContext.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasTopicId.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasUserCandidateSourceDetails.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasUserState.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasWtfImpressions.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/OptimusRequest.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/Product.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/RankingInfo.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/Reason.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/RecentlyEngagedUserId.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/RecommendationStep.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/STPGraph.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/SafetyLevel.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/Score.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/Session.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/SignalData.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/TrackingToken.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/TweetCandidate.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/UserCandidateSourceDetails.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/UserIdAndTimestamp.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/WtfImpression.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/BUILD create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/CandidateParamPredicate.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/CandidateSourceParamPredicate.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/CuratedCompetitorListPredicate.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/ExcludedUserIdPredicate.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/InactivePredicate.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/InactivePredicateParams.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/PreviouslyRecommendedUserIdsPredicate.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/dismiss/BUILD create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/dismiss/DismissedCandidatePredicate.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/dismiss/DismissedCandidatePredicateParams.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/gizmoduck/BUILD create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/gizmoduck/GizmoduckPredicate.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/gizmoduck/GizmoduckPredicateCache.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/gizmoduck/GizmoduckPredicateFSConfig.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/gizmoduck/GizmoduckPredicateParams.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/health/BUILD create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/health/HssPredicate.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/health/HssPredicateFSConfig.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/health/HssPredicateParams.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/sgs/BUILD create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/sgs/InvalidRelationshipPredicate.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/sgs/RecentFollowingPredicate.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/sgs/SgsPredicateFSConfig.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/sgs/SgsPredicateParams.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/sgs/SgsRelationshipsByUserIdPredicate.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/sgs/SgsRelationshipsPredicate.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/user_activity/BUILD create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/user_activity/UserActivityPredicate.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/user_activity/UserActivityPredicateParams.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/common/AdhocScoreModificationType.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/common/BUILD create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/common/DedupCandidates.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/common/RankerId.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/fatigue_ranker/BUILD create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/fatigue_ranker/ImpressionBasedFatigueRanker.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/fatigue_ranker/ImpressionBasedFatigueRankerFSConfig.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/fatigue_ranker/ImpressionBasedFatigueRankerParams.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/first_n_ranker/BUILD create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/first_n_ranker/FirstNRanker.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/first_n_ranker/FirstNRankerFSConfig.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/first_n_ranker/FirstNRankerFeatureSwitchKeys.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/first_n_ranker/FirstNRankerParams.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/interleave_ranker/BUILD create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/interleave_ranker/InterleaveRanker.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/interleave_ranker/InterleaveRankerFSConfig.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/interleave_ranker/InterleaveRankerParams.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/ml_ranker/ranking/BUILD create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/ml_ranker/ranking/HydrateFeaturesTransform.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/ml_ranker/ranking/MlRanker.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/ml_ranker/ranking/MlRankerFSConfig.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/ml_ranker/ranking/MlRankerParams.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/ml_ranker/scoring/AdhocScorer.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/ml_ranker/scoring/BUILD create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/ml_ranker/scoring/DeepbirdScorer.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/ml_ranker/scoring/PostnuxDeepbirdProdScorer.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/ml_ranker/scoring/RandomScorer.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/ml_ranker/scoring/Scorer.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/ml_ranker/scoring/ScorerFactory.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/utils/BUILD create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/utils/Utils.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/weighted_candidate_source_ranker/BUILD create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/weighted_candidate_source_ranker/CandidateShuffle.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/weighted_candidate_source_ranker/WeightMethod.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/weighted_candidate_source_ranker/WeightedCandidateSourceBaseRanker.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/weighted_candidate_source_ranker/WeightedCandidateSourceRanker.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/weighted_candidate_source_ranker/WeightedCandidateSourceRankerFSConfig.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/weighted_candidate_source_ranker/WeightedCandidateSourceRankerParams.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/stores/BUILD create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/stores/LowTweepCredFollowStore.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/transforms/dedup/BUILD create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/transforms/dedup/DedupTransform.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/transforms/modify_social_proof/BUILD create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/transforms/modify_social_proof/ModifySocialProofTransform.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/transforms/modify_social_proof/RemoveAccountProofTransform.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/transforms/ranker_id/BUILD create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/transforms/ranker_id/RandomRankerIdTransform.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/transforms/recommendation_flow_identifier/AddRecommendationFlowIdentifierTransform.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/transforms/recommendation_flow_identifier/BUILD create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/transforms/tracking_token/BUILD create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/transforms/tracking_token/TrackingTokenTransform.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/transforms/weighted_sampling/BUILD create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/transforms/weighted_sampling/SamplingTransform.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/transforms/weighted_sampling/SamplingTransformFSConfig.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/transforms/weighted_sampling/SamplingTransformParams.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/utils/BUILD create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/utils/CollectionUtil.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/utils/DisplayLocationProductConverterUtil.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/utils/MergeUtil.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/utils/RandomUtil.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/utils/RescueWithStatsUtils.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/utils/UserSignupUtil.scala create mode 100644 follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/utils/Weighted.scala create mode 100644 follow-recommendations-service/server/src/main/resources/BUILD create mode 100644 follow-recommendations-service/server/src/main/resources/config/decider.yml create mode 100644 follow-recommendations-service/server/src/main/resources/logback.xml create mode 100644 follow-recommendations-service/server/src/main/resources/quality/stp_models/20141223/epModel create mode 100644 follow-recommendations-service/server/src/main/resources/quality/stp_models/20141223/trainingConfig create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/BUILD create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/FollowRecommendationsServiceThriftServer.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/assembler/models/Action.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/assembler/models/BUILD create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/assembler/models/Config.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/assembler/models/FeedbackAction.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/assembler/models/Footer.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/assembler/models/Header.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/assembler/models/Layout.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/assembler/models/RecommendationOptions.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/assembler/models/SocialProof.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/assembler/models/Title.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/assembler/models/WTFPresentation.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/blenders/BUILD create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/blenders/PromotedAccountsBlender.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/BUILD create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/ConfigBuilder.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/DeciderConfigs.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/FeatureSwitchConfigs.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/GlobalFeatureSwitchConfig.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/ParamsFactory.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/RequestContext.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/RequestContextFactory.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/candidates/BUILD create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/candidates/CandidateUserContext.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/candidates/CandidateUserContextFactory.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/candidates/CandidateUserParamsFactory.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/candidates/HydrateCandidateParamsTransform.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/common/BUILD create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/common/FeatureSwitchConfig.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/deciders/BUILD create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/deciders/DeciderKey.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/deciders/DeciderParams.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/params/BUILD create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/params/GlobalParams.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/controllers/BUILD create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/controllers/CandidateUserDebugParamsBuilder.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/controllers/RecommendationRequestBuilder.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/controllers/RequestBuilderUserFetcher.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/controllers/ScoringUserRequestBuilder.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/controllers/ThriftController.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/ads/BUILD create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/ads/PromotedAccountsFlow.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/ads/PromotedAccountsFlowParams.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/ads/PromotedAccountsFlowRequest.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/ads/PromotedAccountsUtil.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/content_recommender_flow/BUILD create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/content_recommender_flow/ContentRecommenderFlow.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/content_recommender_flow/ContentRecommenderFlowCandidateSourceRegistry.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/content_recommender_flow/ContentRecommenderFlowCandidateSourceWeights.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/content_recommender_flow/ContentRecommenderFlowCandidateSourceWeightsParams.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/content_recommender_flow/ContentRecommenderFlowFSConfig.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/content_recommender_flow/ContentRecommenderFlowFeatureSwitchKeys.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/content_recommender_flow/ContentRecommenderParams.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/content_recommender_flow/ContentRecommenderRequest.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/content_recommender_flow/ContentRecommenderRequestBuilder.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/post_nux_ml/BUILD create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/post_nux_ml/PostNuxMlCandidateSourceRegistry.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/post_nux_ml/PostNuxMlCandidateSourceWeightParams.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/post_nux_ml/PostNuxMlCombinedRankerBuilder.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/post_nux_ml/PostNuxMlFlow.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/post_nux_ml/PostNuxMlFlowCandidateSourceWeights.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/post_nux_ml/PostNuxMlFlowCandidateSourceWeightsFeatureSwitchKeys.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/post_nux_ml/PostNuxMlFlowFSConfig.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/post_nux_ml/PostNuxMlFlowFeatureSwitchKeys.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/post_nux_ml/PostNuxMlParams.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/post_nux_ml/PostNuxMlRequest.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/post_nux_ml/PostNuxMlRequestBuilder.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/post_nux_ml/PostNuxMlRequestBuilderParams.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/logging/BUILD create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/logging/FrsLogger.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/models/BUILD create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/models/CandidateSourceType.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/models/CandidateUserDebugParams.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/models/DebugParams.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/models/DisplayContext.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/models/FeatureValue.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/models/RecommendationFlowData.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/models/RecommendationRequest.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/models/RecommendationResponse.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/models/Request.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/models/ScoringUserRequest.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/models/ScoringUserResponse.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/models/failures/BUILD create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/models/failures/TimeoutPipelineFailure.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/modules/ABDeciderModule.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/modules/BUILD create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/modules/ConfigApiModule.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/modules/DiffyModule.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/modules/FeatureSwitchesModule.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/modules/FlagsModule.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/modules/ProductRegistryModule.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/modules/ScorerModule.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/modules/ScribeModule.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/modules/TimerModule.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/BUILD create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/ProdProductRegistry.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/common/BUILD create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/common/Exceptions.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/common/Product.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/common/ProductRegistry.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/explore_tab/BUILD create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/explore_tab/ExploreTabProduct.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/explore_tab/configapi/BUILD create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/explore_tab/configapi/ExploreTabFSConfig.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/explore_tab/configapi/ExploreTabParams.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/home_timeline/BUILD create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/home_timeline/HTLProductMixer.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/home_timeline/HomeTimelineProduct.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/home_timeline/HomeTimelineStrings.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/home_timeline/configapi/BUILD create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/home_timeline/configapi/HomeTimelineFSConfig.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/home_timeline/configapi/HomeTimelineParams.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/home_timeline_tweet_recs/BUILD create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/home_timeline_tweet_recs/HomeTimelineTweetRecsProduct.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/home_timeline_tweet_recs/configapi/BUILD create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/home_timeline_tweet_recs/configapi/HomeTimelineTweetRecsParams.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/sidebar/BUILD create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/sidebar/SidebarProduct.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/sidebar/configapi/BUILD create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/sidebar/configapi/SidebarParams.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/services/BUILD create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/services/FollowRecommendationsServiceWarmupHandler.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/services/ProductMixerRecommendationService.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/services/ProductPipelineSelector.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/services/ProductPipelineSelectorConfig.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/services/ProductRecommenderService.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/services/RecommendationsService.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/services/UserScoringService.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/services/exceptions/BUILD create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/services/exceptions/UnknownExceptionMapper.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/utils/BUILD create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/utils/CandidateSourceHoldbackUtil.scala create mode 100644 follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/utils/RecommendationFlowBaseSideEffectsUtil.scala create mode 100644 follow-recommendations-service/thrift/src/main/thrift/BUILD create mode 100644 follow-recommendations-service/thrift/src/main/thrift/assembler.thrift create mode 100644 follow-recommendations-service/thrift/src/main/thrift/client_context.thrift create mode 100644 follow-recommendations-service/thrift/src/main/thrift/debug.thrift create mode 100644 follow-recommendations-service/thrift/src/main/thrift/display_context.thrift create mode 100644 follow-recommendations-service/thrift/src/main/thrift/display_location.thrift create mode 100644 follow-recommendations-service/thrift/src/main/thrift/engagementType.thrift create mode 100644 follow-recommendations-service/thrift/src/main/thrift/flows.thrift create mode 100644 follow-recommendations-service/thrift/src/main/thrift/follow-recommendations-service.thrift create mode 100644 follow-recommendations-service/thrift/src/main/thrift/follow_recommendations_serving_history.thrift create mode 100644 follow-recommendations-service/thrift/src/main/thrift/logging/BUILD create mode 100644 follow-recommendations-service/thrift/src/main/thrift/logging/client_context.thrift create mode 100644 follow-recommendations-service/thrift/src/main/thrift/logging/debug.thrift create mode 100644 follow-recommendations-service/thrift/src/main/thrift/logging/display_context.thrift create mode 100644 follow-recommendations-service/thrift/src/main/thrift/logging/display_location.thrift create mode 100644 follow-recommendations-service/thrift/src/main/thrift/logging/engagementType.thrift create mode 100644 follow-recommendations-service/thrift/src/main/thrift/logging/flows.thrift create mode 100644 follow-recommendations-service/thrift/src/main/thrift/logging/logs.thrift create mode 100644 follow-recommendations-service/thrift/src/main/thrift/logging/reasons.thrift create mode 100644 follow-recommendations-service/thrift/src/main/thrift/logging/recently_engaged_user_id.thrift create mode 100644 follow-recommendations-service/thrift/src/main/thrift/logging/recommendations.thrift create mode 100644 follow-recommendations-service/thrift/src/main/thrift/logging/scoring.thrift create mode 100644 follow-recommendations-service/thrift/src/main/thrift/logging/tracking.thrift create mode 100644 follow-recommendations-service/thrift/src/main/thrift/reasons.thrift create mode 100644 follow-recommendations-service/thrift/src/main/thrift/recently_engaged_user_id.thrift create mode 100644 follow-recommendations-service/thrift/src/main/thrift/recommendations.thrift create mode 100644 follow-recommendations-service/thrift/src/main/thrift/scoring.thrift create mode 100644 follow-recommendations-service/thrift/src/main/thrift/tracking.thrift create mode 100644 graph-feature-service/BUILD.bazel create mode 100644 graph-feature-service/README.md create mode 100644 graph-feature-service/doc/common.md create mode 100644 graph-feature-service/doc/getintersection.md create mode 100644 graph-feature-service/src/main/scala/com/twitter/graph_feature_service/common/BUILD.bazel create mode 100644 graph-feature-service/src/main/scala/com/twitter/graph_feature_service/common/Configs.scala create mode 100644 graph-feature-service/src/main/scala/com/twitter/graph_feature_service/server/BUILD.bazel create mode 100644 graph-feature-service/src/main/scala/com/twitter/graph_feature_service/server/Main.scala create mode 100644 graph-feature-service/src/main/scala/com/twitter/graph_feature_service/server/controllers/ServerController.scala create mode 100644 graph-feature-service/src/main/scala/com/twitter/graph_feature_service/server/handlers/ServerGetIntersectionHandler.scala create mode 100644 graph-feature-service/src/main/scala/com/twitter/graph_feature_service/server/handlers/ServerWarmupHandler.scala create mode 100644 graph-feature-service/src/main/scala/com/twitter/graph_feature_service/server/modules/GetIntersectionStoreModule.scala create mode 100644 graph-feature-service/src/main/scala/com/twitter/graph_feature_service/server/modules/GraphFeatureServiceWorkerClientsModule.scala create mode 100644 graph-feature-service/src/main/scala/com/twitter/graph_feature_service/server/modules/LZ4Injection.scala create mode 100644 graph-feature-service/src/main/scala/com/twitter/graph_feature_service/server/modules/ServerFlagModule.scala create mode 100644 graph-feature-service/src/main/scala/com/twitter/graph_feature_service/server/stores/FeatureTypesEncoder.scala create mode 100644 graph-feature-service/src/main/scala/com/twitter/graph_feature_service/server/stores/GetIntersectionStore.scala create mode 100644 graph-feature-service/src/main/scala/com/twitter/graph_feature_service/util/BUILD create mode 100644 graph-feature-service/src/main/scala/com/twitter/graph_feature_service/util/FeatureTypesCalculator.scala create mode 100644 graph-feature-service/src/main/scala/com/twitter/graph_feature_service/util/IntersectionValueCalculator.scala create mode 100644 graph-feature-service/src/main/scala/com/twitter/graph_feature_service/worker/BUILD.bazel create mode 100644 graph-feature-service/src/main/scala/com/twitter/graph_feature_service/worker/Main.scala create mode 100644 graph-feature-service/src/main/scala/com/twitter/graph_feature_service/worker/controllers/WorkerController.scala create mode 100644 graph-feature-service/src/main/scala/com/twitter/graph_feature_service/worker/handlers/WorkerGetIntersectionHandler.scala create mode 100644 graph-feature-service/src/main/scala/com/twitter/graph_feature_service/worker/handlers/WorkerWarmupHandler.scala create mode 100644 graph-feature-service/src/main/scala/com/twitter/graph_feature_service/worker/modules/GraphContainerProviderModule.scala create mode 100644 graph-feature-service/src/main/scala/com/twitter/graph_feature_service/worker/modules/WorkerFlagModule.scala create mode 100644 graph-feature-service/src/main/scala/com/twitter/graph_feature_service/worker/util/AutoUpdatingGraph.scala create mode 100644 graph-feature-service/src/main/scala/com/twitter/graph_feature_service/worker/util/GfsQuery.scala create mode 100644 graph-feature-service/src/main/scala/com/twitter/graph_feature_service/worker/util/GraphContainer.scala create mode 100644 graph-feature-service/src/main/scala/com/twitter/graph_feature_service/worker/util/GraphKey.scala create mode 100644 graph-feature-service/src/main/scala/com/twitter/graph_feature_service/worker/util/GraphType.scala create mode 100644 graph-feature-service/src/main/scalding/com/twitter/graph_feature_service/scalding/BUILD.bazel create mode 100644 graph-feature-service/src/main/scalding/com/twitter/graph_feature_service/scalding/EdgeFeature.scala create mode 100644 graph-feature-service/src/main/scalding/com/twitter/graph_feature_service/scalding/GraphFeatureServiceAppBase.scala create mode 100644 graph-feature-service/src/main/scalding/com/twitter/graph_feature_service/scalding/GraphFeatureServiceApps.scala create mode 100644 graph-feature-service/src/main/scalding/com/twitter/graph_feature_service/scalding/GraphFeatureServiceMainJob.scala create mode 100644 graph-feature-service/src/main/scalding/com/twitter/graph_feature_service/scalding/adhoc/BUILD.bazel create mode 100644 graph-feature-service/src/main/scalding/com/twitter/graph_feature_service/scalding/adhoc/RandomRequestGenerationApp.scala create mode 100644 graph-feature-service/src/main/thrift/com/twitter/graph_feature_service/BUILD create mode 100644 graph-feature-service/src/main/thrift/com/twitter/graph_feature_service/graph_feature_service.thrift create mode 100644 home-mixer/BUILD.bazel create mode 100644 home-mixer/README.md create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/BUILD.bazel create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/HomeMixerHttpServerWarmupHandler.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/HomeMixerServer.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/HomeMixerThriftServerWarmupHandler.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/candidate_pipeline/BUILD.bazel create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/candidate_pipeline/ConversationServiceCandidatePipelineConfig.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/candidate_pipeline/ConversationServiceCandidatePipelineConfigBuilder.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/candidate_pipeline/ConversationServiceResponseFeatureTransformer.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/candidate_pipeline/EditedTweetsCandidatePipelineConfig.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/candidate_pipeline/NewTweetsPillCandidatePipelineConfig.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/candidate_pipeline/TimelineServiceResponseFeatureTransformer.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/controller/BUILD.bazel create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/controller/HomeThriftController.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/candidate_source/BUILD.bazel create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/candidate_source/EarlybirdCandidateSource.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/candidate_source/SimilarityBasedUsersCandidateSource.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/candidate_source/StaleTweetsCacheCandidateSource.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/AuthorChildFeedbackActionBuilder.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/BUILD.bazel create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/BlockUserChildFeedbackActionBuilder.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/DontLikeFeedbackActionBuilder.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/EngagerSocialContextBuilder.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/ExtendedReplySocialContextBuilder.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/FeedbackUtil.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/FollowedBySocialContextBuilder.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/HomeAdsClientEventDetailsBuilder.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/HomeClientEventDetailsBuilder.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/HomeConversationServiceCandidateDecorator.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/HomeFeedbackActionInfoBuilder.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/HomeQueryTypePredicates.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/HomeTimelinesScoreInfoBuilder.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/HomeTweetSocialContextBuilder.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/HomeTweetTypePredicates.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/LikedBySocialContextBuilder.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/ListConversationServiceCandidateDecorator.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/MuteUserChildFeedbackActionBuilder.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/NotInterestedTopicFeedbackActionBuilder.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/NotRelevantChildFeedbackActionBuilder.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/ReceivedReplySocialContextBuilder.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/ReportTweetChildFeedbackActionBuilder.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/RetweeterChildFeedbackActionBuilder.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/TopicSocialContextBuilder.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/UnfollowUserChildFeedbackActionBuilder.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/YouMightLikeSocialContextBuilder.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/builder/BUILD.bazel create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/builder/HomeClientEventInfoBuilder.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/builder/HomeConversationModuleMetadataBuilder.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/builder/ListClientEventDetailsBuilder.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/urt/builder/AddEntriesWithReplaceAndShowAlertAndShowCoverInstructionBuilder.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/urt/builder/BUILD.bazel create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/urt/builder/HomeWhoToFollowFeedbackActionInfoBuilder.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/AncestorFeatureHydrator.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/AuthorFeatureHydrator.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/BUILD.bazel create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/DismissInfoQueryFeatureHydrator.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/EarlybirdFeatureHydrator.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/FeedbackHistoryQueryFeatureHydrator.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/FocalTweetFeatureHydrator.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/FollowedTopicsQueryFeatureHydrator.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/GizmoduckAuthorSafetyFeatureHydrator.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/GizmoduckUserQueryFeatureHydrator.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/GraphTwoHopFeatureHydrator.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/ImpressionBloomFilterQueryFeatureHydrator.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/LastNonPollingTimeQueryFeatureHydrator.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/ListMembersQueryFeatureHydrator.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/MetricCenterUserCountingFeatureHydrator.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/NamesFeatureHydrator.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/PersistenceStoreQueryFeatureHydrator.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/PerspectiveFilteredSocialContextFeatureHydrator.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/RealGraphInNetworkScoresQueryFeatureHydrator.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/RealGraphQueryFeatureHydrator.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/RealGraphViewerAuthorFeatureHydrator.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/RealGraphViewerRelatedUsersFeatureHydrator.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/RealTimeInteractionGraphEdgeFeatureHydrator.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/RealTimeInteractionGraphUserVertexQueryFeatureHydrator.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/ReplyFeatureHydrator.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/RequestQueryFeatureHydrator.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/RetweetSourceTweetFeatureHydrator.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/SGSFollowedUsersQueryFeatureHydrator.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/SGSValidSocialContextFeatureHydrator.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/SimClustersEngagementSimilarityFeatureHydrator.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/SocialGraphServiceFeatureHydrator.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/TSPInferredTopicFeatureHydrator.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/TimeFeaturesHydrator.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/TimelineServiceTweetsQueryFeatureHydrator.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/TweetImpressionsQueryFeatureHydrator.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/TweetMetaDataFeatureHydrator.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/TweetypieContentFeatureHydrator.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/TweetypieFeatureHydrator.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/TweetypieStaticEntitiesFeatureHydrator.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/TwhinAuthorFollow20220101FeatureHydrator.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/TwhinUserEngagementQueryFeatureHydrator.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/TwhinUserFollowQueryFeatureHydrator.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/UserFollowedTopicIdsFeatureHydrator.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/UserLanguagesFeatureHydrator.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/UserStateQueryFeatureHydrator.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/UtegFeatureHydrator.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/adapters/author_features/AuthorFeaturesAdapter.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/adapters/author_features/BUILD.bazel create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/adapters/content/BUILD.bazel create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/adapters/content/ContentFeatureAdapter.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/adapters/earlybird/BUILD.bazel create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/adapters/earlybird/EarlybirdAdapter.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/adapters/inferred_topic/BUILD.bazel create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/adapters/inferred_topic/InferredTopicAdapter.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/adapters/non_ml_features/BUILD.bazel create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/adapters/non_ml_features/NonMLCandidateFeaturesAdapter.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/adapters/non_ml_features/NonMLCommonFeaturesAdapter.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/adapters/offline_aggregates/BUILD.bazel create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/adapters/offline_aggregates/PassThroughAdapter.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/adapters/offline_aggregates/SparseAggregatesToDenseAdapter.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/adapters/twhin_embeddings/BUILD.bazel create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/adapters/twhin_embeddings/TwhinEmbeddingsAdapter.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/offline_aggregates/AggregateFeatureInfo.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/offline_aggregates/AggregateFeaturesToDecodeWithMetadata.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/offline_aggregates/BUILD.bazel create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/offline_aggregates/BaseAggregateQueryFeatureHydrator.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/offline_aggregates/BaseEdgeAggregateFeatureHydrator.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/offline_aggregates/EdgeAggregateFeatures.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/offline_aggregates/PartAAggregateQueryFeatureHydrator.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/offline_aggregates/PartBAggregateQueryFeatureHydrator.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/offline_aggregates/Phase1EdgeAggregateFeatureHydrator.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/offline_aggregates/Phase2EdgeAggregateFeatureHydrator.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/offline_aggregates/Utils.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/real_time_aggregates/BUILD.bazel create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/real_time_aggregates/BaseRealTimeAggregateBulkCandidateFeatureHydrator.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/real_time_aggregates/BaseRealTimeAggregateQueryFeatureHydrator.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/real_time_aggregates/BaseRealtimeAggregateHydrator.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/real_time_aggregates/EngagementsReceivedByAuthorRealTimeAggregateFeatureHydrator.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/real_time_aggregates/RealTimeAggregateTimeDecay.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/real_time_aggregates/TopicCountryEngagementRealTimeAggregateFeatureHydrator.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/real_time_aggregates/TopicEngagementRealTimeAggregateFeatureHydrator.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/real_time_aggregates/TweetCountryEngagementRealTimeAggregateFeatureHydrator.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/real_time_aggregates/TweetEngagementRealTimeAggregateFeatureHydrator.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/real_time_aggregates/TwitterListEngagementRealTimeAggregateFeatureHydrator.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/real_time_aggregates/UserAuthorEngagementRealTimeAggregateFeatureHydrator.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/real_time_aggregates/UserEngagementRealTimeAggregatesFeatureHydrator.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/filter/BUILD.bazel create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/filter/DropMaxCandidatesFilter.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/filter/FeedbackFatigueFilter.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/filter/InvalidConversationModuleFilter.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/filter/KeepBestOutOfNetworkTweetPerAuthorFilter.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/filter/OutOfNetworkCompetitorFilter.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/filter/OutOfNetworkCompetitorURLFilter.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/filter/PredicateFeatureFilter.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/filter/PredicateGatedFilter.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/filter/PreviouslySeenTweetsFilter.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/filter/PreviouslyServedAncestorsFilter.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/filter/PreviouslyServedTweetsFilter.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/filter/RejectTweetFromViewerFilter.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/filter/RetweetDeduplicationFilter.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/filter/RetweetSourceTweetRemovingFilter.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/filter/SocialContextFilter.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/gate/BUILD.bazel create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/gate/DismissFatigueGate.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/gate/ExcludeSoftUserGate.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/gate/MinCachedTweetsGate.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/gate/NonEmptySeqFeatureGate.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/gate/RequestContextGate.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/gate/RequestContextNotGate.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/gate/SupportedLanguagesGate.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/gate/TimelinesPersistenceStoreLastInjectionGate.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/gate/ViewerIsListOwnerGate.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/query_transformer/BUILD.bazel create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/query_transformer/EditedTweetsCandidatePipelineQueryTransformer.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/scorer/BUILD.bazel create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/scorer/FeedbackFatigueScorer.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/scorer/OONTweetScalingScorer.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/scorer/VerifiedAuthorScalingScorer.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/selector/BUILD.bazel create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/selector/DebunchCandidates.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/selector/UpdateConversationModuleId.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/selector/UpdateHomeClientEventDetails.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/selector/UpdateNewTweetsPillDecoration.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/side_effect/BUILD.bazel create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/side_effect/ClientEventsBuilder.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/side_effect/HomeScribeClientEventSideEffect.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/side_effect/HomeScribeServedEntriesSideEffect.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/side_effect/PublishClientSentImpressionsEventBusSideEffect.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/side_effect/PublishClientSentImpressionsManhattanSideEffect.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/side_effect/ServedCandidateFeatureKeysKafkaSideEffect.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/side_effect/ServedCandidateFeatureKeysKafkaSideEffectBuilder.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/side_effect/ServedCandidateKafkaSideEffect.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/side_effect/ServedCandidateKeysKafkaSideEffect.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/side_effect/ServedCandidateKeysKafkaSideEffectBuilder.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/side_effect/ServedStatsSideEffect.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/side_effect/TruncateTimelinesPersistenceStoreSideEffect.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/side_effect/UpdateImpressionBloomFilterSideEffect.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/side_effect/UpdateLastNonPollingTimeSideEffect.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/side_effect/UpdateTimelinesPersistenceStoreSideEffect.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/marshaller/request/BUILD.bazel create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/marshaller/request/DeviceContextUnmarshaller.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/marshaller/request/HomeMixerDebugParamsUnmarshaller.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/marshaller/request/HomeMixerProductContextUnmarshaller.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/marshaller/request/HomeMixerProductUnmarshaller.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/marshaller/request/HomeMixerRequestUnmarshaller.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/marshaller/timeline_logging/BUILD.bazel create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/marshaller/timeline_logging/ConversationEntryMarshaller.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/marshaller/timeline_logging/PromotedTweetEntryMarshaller.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/marshaller/timeline_logging/TweetEntryMarshaller.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/marshaller/timeline_logging/WhoToFollowEntryMarshaller.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/marshaller/timelines/BUILD.bazel create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/marshaller/timelines/ChronologicalCursorMarshaller.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/marshaller/timelines/ChronologicalCursorUnmarshaller.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/marshaller/timelines/DeviceContextMarshaller.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/marshaller/timelines/RecommendedUsersCursorUnmarshaller.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/marshaller/timelines/TimelineServiceCursorMarshaller.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/marshaller/timelines/TopicContextFunctionalityTypeUnmarshaller.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/model/BUILD.bazel create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/model/ClearCacheIncludeInstruction.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/model/ContentFeatures.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/model/GapIncludeInstruction.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/model/HomeAdsQuery.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/model/HomeFeatures.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/model/request/BUILD.bazel create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/model/request/DeviceContext.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/model/request/HasListId.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/model/request/HasSeenTweetIds.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/model/request/HomeMixerDebugOptions.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/model/request/HomeMixerProduct.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/model/request/HomeMixerProductContext.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/model/request/HomeMixerRequest.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/module/AdvertiserBrandSafetySettingsStoreModule.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/module/BUILD.bazel create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/module/ClientSentImpressionsPublisherModule.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/module/ConversationServiceModule.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/module/FeedbackHistoryClientModule.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/module/HomeAdsCandidateSourceModule.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/module/HomeMixerFlagsModule.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/module/HomeMixerResourcesModule.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/module/HomeNaviModelClientModule.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/module/ImpressionBloomFilterModule.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/module/InjectionHistoryClientModule.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/module/ManhattanClientsModule.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/module/ManhattanFeatureRepositoryModule.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/module/ManhattanTweetImpressionStoreModule.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/module/MemcachedFeatureRepositoryModule.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/module/OptimizedStratoClientModule.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/module/PeopleDiscoveryServiceModule.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/module/PipelineFailureExceptionMapper.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/module/RealGraphInNetworkScoresModule.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/module/RealtimeAggregateFeatureRepositoryModule.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/module/ScoredTweetsMemcacheModule.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/module/ScribeEventPublisherModule.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/module/SimClustersRecentEngagementsClientModule.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/module/StaleTweetsCacheModule.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/module/ThriftFeatureRepositoryModule.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/module/TimelinesPersistenceStoreClientModule.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/module/TweetyPieClientModule.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/module/TweetypieStaticEntitiesCacheClientModule.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/module/UserMetadataStoreModule.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/param/BUILD.bazel create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/param/GlobalParamConfigModule.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/param/HomeGlobalParamConfig.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/param/HomeGlobalParams.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/param/HomeMixerFlagName.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/param/HomeMixerInjectionNames.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/param/decider/BUILD.bazel create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/param/decider/DeciderKey.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/BUILD.bazel create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/HomeMixerProductModule.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/HomeProductPipelineRegistryConfig.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/following/BUILD.bazel create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/following/FollowingAdsCandidatePipelineBuilder.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/following/FollowingEarlybirdCandidatePipelineConfig.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/following/FollowingEarlybirdQueryTransformer.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/following/FollowingEarlybirdResponseFeatureTransformer.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/following/FollowingMixerPipelineConfig.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/following/FollowingProductPipelineConfig.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/following/FollowingWhoToFollowArmCandidatePipelineConfigBuilder.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/following/model/BUILD.bazel create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/following/model/FollowingQuery.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/following/model/HomeMixerExternalStrings.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/following/param/BUILD.bazel create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/following/param/FollowingParam.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/following/param/FollowingParamConfig.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/for_you/BUILD.bazel create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/for_you/ForYouAdsCandidatePipelineBuilder.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/for_you/ForYouConversationServiceCandidatePipelineConfig.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/for_you/ForYouProductPipelineConfig.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/for_you/ForYouScoredTweetsCandidatePipelineConfig.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/for_you/ForYouScoredTweetsMixerPipelineConfig.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/for_you/ForYouScoredTweetsResponseFeatureTransformer.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/for_you/ForYouTimelineScorerCandidatePipelineConfig.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/for_you/ForYouTimelineScorerMixerPipelineConfig.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/for_you/ForYouTimelineScorerResponseFeatureTransformer.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/for_you/ForYouWhoToFollowCandidatePipelineConfigBuilder.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/for_you/candidate_source/BUILD.bazel create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/for_you/candidate_source/ScoredTweetsProductCandidateSource.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/for_you/model/BUILD.bazel create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/for_you/model/ForYouQuery.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/for_you/model/ForYouTweetsResponse.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/for_you/param/BUILD.bazel create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/for_you/param/ForYouParam.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/for_you/param/ForYouParamConfig.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_recommended_users/BUILD.bazel create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_recommended_users/ListMemberBasedUsersCandidatePipelineConfig.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_recommended_users/ListMemberBasedUsersResponseFeatureTransfromer.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_recommended_users/ListRecommendedUsersMixerPipelineConfig.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_recommended_users/ListRecommendedUsersProductPipelineConfig.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_recommended_users/feature_hydrator/BUILD.bazel create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_recommended_users/feature_hydrator/GizmoduckUserFeatureHydrator.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_recommended_users/feature_hydrator/IsListMemberFeatureHydrator.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_recommended_users/filter/BUILD.bazel create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_recommended_users/filter/DropMaxCandidatesByScoreFilter.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_recommended_users/filter/PreviouslyServedUsersFilter.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_recommended_users/model/BUILD.bazel create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_recommended_users/model/ListFeatures.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_recommended_users/model/ListRecommendedUsersQuery.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_recommended_users/param/BUILD.bazel create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_recommended_users/param/ListRecommendedUsersParam.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_recommended_users/param/ListRecommendedUsersParamConfig.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_tweets/BUILD.bazel create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_tweets/ListTweetsAdsCandidatePipelineBuilder.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_tweets/ListTweetsMixerPipelineConfig.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_tweets/ListTweetsProductPipelineConfig.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_tweets/ListTweetsTimelineServiceCandidatePipelineConfig.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_tweets/model/BUILD.bazel create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_tweets/model/ListTweetsQuery.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_tweets/param/BUILD.bazel create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_tweets/param/ListTweetsParam.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_tweets/param/ListTweetsParamConfig.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/BUILD.bazel create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/ScoredTweetsProductPipelineConfig.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/ScoredTweetsRecommendationPipelineConfig.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/candidate_pipeline/BUILD.bazel create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/candidate_pipeline/CachedScoredTweetsCandidatePipelineConfig.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/candidate_pipeline/ScoredTweetsCrMixerCandidatePipelineConfig.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/candidate_pipeline/ScoredTweetsFrsCandidatePipelineConfig.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/candidate_pipeline/ScoredTweetsInNetworkCandidatePipelineConfig.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/candidate_pipeline/ScoredTweetsUtegCandidatePipelineConfig.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/candidate_source/BUILD.bazel create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/candidate_source/CachedScoredTweetsCandidateSource.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/feature_hydrator/BUILD.bazel create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/feature_hydrator/CachedScoredTweetsQueryFeatureHydrator.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/marshaller/BUILD.bazel create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/marshaller/ScoredTweetsResponseDomainMarshaller.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/marshaller/ScoredTweetsResponseTransportMarshaller.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/model/BUILD.bazel create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/model/ScoredTweetsQuery.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/model/ScoredTweetsResponse.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/param/BUILD.bazel create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/param/ScoredTweetsParam.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/param/ScoredTweetsParamConfig.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/query_feature_hydrator/BUILD.bazel create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/query_feature_hydrator/FrsSeedUsersQueryFeatureHydrator.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/query_transformer/BUILD.bazel create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/query_transformer/TimelineRankerFrsQueryTransformer.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/query_transformer/TimelineRankerInNetworkQueryTransformer.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/query_transformer/TimelineRankerQueryTransformer.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/query_transformer/TimelineRankerUtegQueryTransformer.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/response_transformer/BUILD.bazel create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/response_transformer/CachedScoredTweetsResponseFeatureTransformer.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/response_transformer/ScoredTweetsCrMixerResponseFeatureTransformer.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/response_transformer/ScoredTweetsFrsResponseFeatureTransformer.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/response_transformer/ScoredTweetsInNetworkResponseFeatureTransformer.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/response_transformer/ScoredTweetsUtegResponseFeatureTransformer.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/response_transformer/TimelineRankerResponseTransformer.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/scorer/BUILD.bazel create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/scorer/DiversityDiscountProvider.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/scorer/DiversityScorer.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/scorer/HomeNaviModelDataRecordScorer.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/scorer/WeightedScoresSumScorer.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/scoring_pipeline/BUILD.bazel create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/scoring_pipeline/ScoredTweetsDiversityScoringPipelineConfig.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/scoring_pipeline/ScoredTweetsRescoreOONScoringPipelineConfig.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/scoring_pipeline/ScoredTweetsRescoreVerifiedAuthorScoringPipelineConfig.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/scoring_pipeline/ScoredTweetsScoringPipelineConfig.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/scoring_pipeline/ScoredTweetsWeightedScoresSumScoringPipelineConfig.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/side_effect/BUILD.bazel create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/side_effect/CachedScoredTweetsSideEffect.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/side_effect/ScribeServedCommonFeaturesAndCandidateFeaturesSideEffect.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/service/BUILD.bazel create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/service/HomeMixerAccessPolicy.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/service/HomeMixerAlertConfig.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/service/ScoredTweetsService.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/store/BUILD.bazel create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/store/RealGraphInNetworkScoresStore.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/store/UserLanguagesStore.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/util/BUILD.bazel create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/util/CachedScoredTweetsHelper.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/util/CandidatesUtil.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/util/InjectionTransformer.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/util/LanguageUtil.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/util/MissingKeyException.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/util/ObservedKeyValueResultHandler.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/util/ReplyRetweetUtil.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/util/TensorFlowUtil.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/util/TweetImpressionsHelper.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/util/earlybird/BUILD.bazel create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/util/earlybird/EarlybirdRequestUtil.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/util/earlybird/EarlybirdResponseUtil.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/util/earlybird/RelevanceSearchUtil.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/util/tweetypie/BUILD.bazel create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/util/tweetypie/RequestFields.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/util/tweetypie/content/BUILD.bazel create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/util/tweetypie/content/FeatureExtractionHelper.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/util/tweetypie/content/TweetMediaFeaturesExtractor.scala create mode 100644 home-mixer/server/src/main/scala/com/twitter/home_mixer/util/tweetypie/content/TweetTextFeaturesExtractor.scala create mode 100644 navi/dr_transform/Cargo.toml create mode 100644 navi/dr_transform/src/all_config.rs create mode 100644 navi/dr_transform/src/converter.rs create mode 100644 navi/dr_transform/src/lib.rs create mode 100644 navi/dr_transform/src/util.rs create mode 100644 navi/navi/Cargo.toml create mode 100644 navi/navi/README.md create mode 100644 navi/navi/build.rs create mode 100644 navi/navi/proto/kfserving/grpc_predict_v2.proto create mode 100644 navi/navi/proto/tensorflow/core/example/example.proto create mode 100644 navi/navi/proto/tensorflow/core/example/feature.proto create mode 100644 navi/navi/proto/tensorflow/core/framework/allocation_description.proto create mode 100644 navi/navi/proto/tensorflow/core/framework/api_def.proto create mode 100644 navi/navi/proto/tensorflow/core/framework/attr_value.proto create mode 100644 navi/navi/proto/tensorflow/core/framework/cost_graph.proto create mode 100644 navi/navi/proto/tensorflow/core/framework/dataset_metadata.proto create mode 100644 navi/navi/proto/tensorflow/core/framework/dataset_options.proto create mode 100644 navi/navi/proto/tensorflow/core/framework/device_attributes.proto create mode 100644 navi/navi/proto/tensorflow/core/framework/full_type.proto create mode 100644 navi/navi/proto/tensorflow/core/framework/function.proto create mode 100644 navi/navi/proto/tensorflow/core/framework/graph.proto create mode 100644 navi/navi/proto/tensorflow/core/framework/graph_transfer_info.proto create mode 100644 navi/navi/proto/tensorflow/core/framework/kernel_def.proto create mode 100644 navi/navi/proto/tensorflow/core/framework/log_memory.proto create mode 100644 navi/navi/proto/tensorflow/core/framework/model.proto create mode 100644 navi/navi/proto/tensorflow/core/framework/node_def.proto create mode 100644 navi/navi/proto/tensorflow/core/framework/op_def.proto create mode 100644 navi/navi/proto/tensorflow/core/framework/reader_base.proto create mode 100644 navi/navi/proto/tensorflow/core/framework/resource_handle.proto create mode 100644 navi/navi/proto/tensorflow/core/framework/step_stats.proto create mode 100644 navi/navi/proto/tensorflow/core/framework/summary.proto create mode 100644 navi/navi/proto/tensorflow/core/framework/tensor.proto create mode 100644 navi/navi/proto/tensorflow/core/framework/tensor_description.proto create mode 100644 navi/navi/proto/tensorflow/core/framework/tensor_shape.proto create mode 100644 navi/navi/proto/tensorflow/core/framework/tensor_slice.proto create mode 100644 navi/navi/proto/tensorflow/core/framework/types.proto create mode 100644 navi/navi/proto/tensorflow/core/framework/variable.proto create mode 100644 navi/navi/proto/tensorflow/core/framework/versions.proto create mode 100644 navi/navi/proto/tensorflow/core/protobuf/autotuning.proto create mode 100644 navi/navi/proto/tensorflow/core/protobuf/bfc_memory_map.proto create mode 100644 navi/navi/proto/tensorflow/core/protobuf/cluster.proto create mode 100644 navi/navi/proto/tensorflow/core/protobuf/composite_tensor_variant.proto create mode 100644 navi/navi/proto/tensorflow/core/protobuf/config.proto create mode 100644 navi/navi/proto/tensorflow/core/protobuf/control_flow.proto create mode 100644 navi/navi/proto/tensorflow/core/protobuf/conv_autotuning.proto create mode 100644 navi/navi/proto/tensorflow/core/protobuf/coordination_config.proto create mode 100644 navi/navi/proto/tensorflow/core/protobuf/coordination_service.proto create mode 100644 navi/navi/proto/tensorflow/core/protobuf/critical_section.proto create mode 100644 navi/navi/proto/tensorflow/core/protobuf/data_service.proto create mode 100644 navi/navi/proto/tensorflow/core/protobuf/debug.proto create mode 100644 navi/navi/proto/tensorflow/core/protobuf/debug_event.proto create mode 100644 navi/navi/proto/tensorflow/core/protobuf/device_filters.proto create mode 100644 navi/navi/proto/tensorflow/core/protobuf/device_properties.proto create mode 100644 navi/navi/proto/tensorflow/core/protobuf/distributed_runtime_payloads.proto create mode 100644 navi/navi/proto/tensorflow/core/protobuf/eager_service.proto create mode 100644 navi/navi/proto/tensorflow/core/protobuf/error_codes.proto create mode 100644 navi/navi/proto/tensorflow/core/protobuf/graph_debug_info.proto create mode 100644 navi/navi/proto/tensorflow/core/protobuf/master.proto create mode 100644 navi/navi/proto/tensorflow/core/protobuf/master_service.proto create mode 100644 navi/navi/proto/tensorflow/core/protobuf/meta_graph.proto create mode 100644 navi/navi/proto/tensorflow/core/protobuf/named_tensor.proto create mode 100644 navi/navi/proto/tensorflow/core/protobuf/queue_runner.proto create mode 100644 navi/navi/proto/tensorflow/core/protobuf/remote_tensor_handle.proto create mode 100644 navi/navi/proto/tensorflow/core/protobuf/replay_log.proto create mode 100644 navi/navi/proto/tensorflow/core/protobuf/rewriter_config.proto create mode 100644 navi/navi/proto/tensorflow/core/protobuf/saved_model.proto create mode 100644 navi/navi/proto/tensorflow/core/protobuf/saved_object_graph.proto create mode 100644 navi/navi/proto/tensorflow/core/protobuf/saver.proto create mode 100644 navi/navi/proto/tensorflow/core/protobuf/service_config.proto create mode 100644 navi/navi/proto/tensorflow/core/protobuf/snapshot.proto create mode 100644 navi/navi/proto/tensorflow/core/protobuf/status.proto create mode 100644 navi/navi/proto/tensorflow/core/protobuf/struct.proto create mode 100644 navi/navi/proto/tensorflow/core/protobuf/tensor_bundle.proto create mode 100644 navi/navi/proto/tensorflow/core/protobuf/tensorflow_server.proto create mode 100644 navi/navi/proto/tensorflow/core/protobuf/trackable_object_graph.proto create mode 100644 navi/navi/proto/tensorflow/core/protobuf/transport_options.proto create mode 100644 navi/navi/proto/tensorflow/core/protobuf/verifier_config.proto create mode 100644 navi/navi/proto/tensorflow/core/protobuf/worker.proto create mode 100644 navi/navi/proto/tensorflow/core/protobuf/worker_service.proto create mode 100644 navi/navi/proto/tensorflow_serving/apis/classification.proto create mode 100644 navi/navi/proto/tensorflow_serving/apis/get_model_metadata.proto create mode 100644 navi/navi/proto/tensorflow_serving/apis/get_model_status.proto create mode 100644 navi/navi/proto/tensorflow_serving/apis/inference.proto create mode 100644 navi/navi/proto/tensorflow_serving/apis/input.proto create mode 100644 navi/navi/proto/tensorflow_serving/apis/logging.proto create mode 100644 navi/navi/proto/tensorflow_serving/apis/model.proto create mode 100644 navi/navi/proto/tensorflow_serving/apis/model_management.proto create mode 100644 navi/navi/proto/tensorflow_serving/apis/model_service.proto create mode 100644 navi/navi/proto/tensorflow_serving/apis/predict.proto create mode 100644 navi/navi/proto/tensorflow_serving/apis/prediction_log.proto create mode 100644 navi/navi/proto/tensorflow_serving/apis/prediction_service.proto create mode 100644 navi/navi/proto/tensorflow_serving/apis/regression.proto create mode 100644 navi/navi/proto/tensorflow_serving/apis/session_service.proto create mode 100644 navi/navi/proto/tensorflow_serving/apis/status.proto create mode 100644 navi/navi/proto/tensorflow_serving/config/file_system_storage_path_source.proto create mode 100644 navi/navi/proto/tensorflow_serving/config/log_collector_config.proto create mode 100644 navi/navi/proto/tensorflow_serving/config/logging_config.proto create mode 100644 navi/navi/proto/tensorflow_serving/config/model_server_config.proto create mode 100644 navi/navi/scripts/run_onnx.sh create mode 100644 navi/navi/scripts/run_tf2.sh create mode 100644 navi/navi/src/batch.rs create mode 100644 navi/navi/src/bin/navi.rs create mode 100644 navi/navi/src/bin/navi_onnx.rs create mode 100644 navi/navi/src/bin/navi_torch.rs create mode 100644 navi/navi/src/bootstrap.rs create mode 100644 navi/navi/src/cli_args.rs create mode 100644 navi/navi/src/cores/validator.rs create mode 100644 navi/navi/src/lib.rs create mode 100644 navi/navi/src/metrics.rs create mode 100644 navi/navi/src/onnx_model.rs create mode 100644 navi/navi/src/predict_service.rs create mode 100644 navi/navi/src/tf_model.rs create mode 100644 navi/navi/src/torch_model.rs create mode 100644 navi/segdense/Cargo.toml create mode 100644 navi/segdense/src/error.rs create mode 100644 navi/segdense/src/lib.rs create mode 100644 navi/segdense/src/main.rs create mode 100644 navi/segdense/src/mapper.rs create mode 100644 navi/segdense/src/segdense_transform_spec_home_recap_2022.rs create mode 100644 navi/segdense/src/util.rs create mode 100644 navi/thrift_bpr_adapter/thrift/Cargo.toml create mode 100644 navi/thrift_bpr_adapter/thrift/src/data.rs create mode 100644 navi/thrift_bpr_adapter/thrift/src/decoder.rs create mode 100644 navi/thrift_bpr_adapter/thrift/src/lib.rs create mode 100644 navi/thrift_bpr_adapter/thrift/src/main.rs create mode 100644 navi/thrift_bpr_adapter/thrift/src/prediction_service.rs create mode 100644 navi/thrift_bpr_adapter/thrift/src/tensor.rs create mode 100644 product-mixer/README.md create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/candidate_source/account_recommendations_mixer/AccountRecommendationsMixerCandidateSource.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/candidate_source/account_recommendations_mixer/BUILD create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/candidate_source/ads/AdsProdStratoCandidateSource.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/candidate_source/ads/AdsProdThriftCandidateSource.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/candidate_source/ads/AdsStagingCandidateSource.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/candidate_source/ads/BUILD create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/candidate_source/ann/AnnCandidateSource.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/candidate_source/ann/AnnIdQuery.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/candidate_source/ann/BUILD.bazel create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/candidate_source/audiospace/BUILD.bazel create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/candidate_source/audiospace/CreatedSpacesCandidateSource.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/candidate_source/business_profiles/BUILD create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/candidate_source/business_profiles/TeamMembersCandidateSource.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/candidate_source/cr_mixer/BUILD.bazel create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/candidate_source/cr_mixer/CrMixerFrsBasedTweetRecommendationsCandidateSource.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/candidate_source/cr_mixer/CrMixerTweetRecommendationsCandidateSource.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/candidate_source/earlybird/BUILD.bazel create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/candidate_source/earlybird/EarlybirdTweetCandidateSource.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/candidate_source/explore_ranker/BUILD.bazel create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/candidate_source/explore_ranker/ExploreRankerCandidateSource.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/candidate_source/flexible_injection_pipeline/BUILD create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/candidate_source/flexible_injection_pipeline/PromptCandidateSource.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/candidate_source/hermit/BUILD.bazel create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/candidate_source/hermit/UsersSimilarToMeCandidateSource.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/candidate_source/interest_discovery/BUILD create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/candidate_source/interest_discovery/RelatedTopicsCandidateSource.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/candidate_source/lists/BUILD.bazel create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/candidate_source/lists/OrganicPopGeoListsCandidateSource.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/candidate_source/people_discovery/BUILD create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/candidate_source/people_discovery/PeopleDiscoveryCandidateSource.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/candidate_source/recommendations/BUILD.bazel create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/candidate_source/recommendations/UserFollowRecommendationsCandidateSource.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/candidate_source/social_graph/BUILD.bazel create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/candidate_source/social_graph/SocialgraphCandidateSource.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/candidate_source/social_graph/SocialgraphCursorConstants.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/candidate_source/timeline_ranker/BUILD.bazel create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/candidate_source/timeline_ranker/TimelineRankerInNetworkCandidateSource.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/candidate_source/timeline_ranker/TimelineRankerRecapCandidateSource.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/candidate_source/timeline_ranker/TimelineRankerUtegCandidateSource.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/candidate_source/timeline_scorer/BUILD.bazel create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/candidate_source/timeline_scorer/TimelineScorerCandidateSource.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/candidate_source/timeline_service/BUILD create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/candidate_source/timeline_service/TimelineServiceTweetCandidateSource.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/candidate_source/timelines_impression_store/BUILD create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/candidate_source/timelines_impression_store/TimelinesImpressionStoreCandidateSourceV2.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/candidate_source/topics/BUILD create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/candidate_source/topics/FollowedTopicsCandidateSource.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/candidate_source/tweetconvosvc/BUILD.bazel create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/candidate_source/tweetconvosvc/ConversationServiceCandidateSource.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/candidate_source/tweetconvosvc/ConversationServiceResponseFeatureTransformer.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/candidate_source/tweetconvosvc/DropMaxConversationModuleItemCandidates.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/slice/BUILD create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/slice/SliceItemCandidateDecorator.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/slice/builder/BUILD create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/slice/builder/CursorCandidateSliceItemBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/BUILD create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/UrtConversationItemCandidateDecorator.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/UrtItemCandidateDecorator.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/UrtItemInModuleDecorator.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/UrtMultipleModulesDecorator.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/builder/contextual_ref/ContextualTweetRefBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/builder/conversations/ConversationModuleMetadataBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/builder/flexible_injection_pipeline/FlipPromptCandidateUrtItemBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/builder/flexible_injection_pipeline/FlipPromptModuleGrouping.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/builder/flexible_injection_pipeline/FlipPromptUrtModuleBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/builder/flexible_injection_pipeline/OnboardingInjectionConversions.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/builder/flexible_injection_pipeline/RelevancePromptConversions.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/builder/flexible_injection_pipeline/TilesCarouselConversions.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/builder/icon/HorizonIconBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/builder/item/ad/AdsCandidateUrtItemBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/builder/item/alert/DurationParamBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/builder/item/alert/ShowAlertCandidateUrtItemBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/builder/item/alert/StaticShowAlertColorConfigurationBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/builder/item/alert/StaticShowAlertDisplayLocationBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/builder/item/alert/StaticShowAlertIconDisplayInfoBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/builder/item/article/ArticleCandidateUrtItemBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/builder/item/audio_space/AudioSpaceCandidateUrtItemBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/builder/item/card/CardCandidateUtrItemBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/builder/item/commerce/CommerceProductCandidateUrtItemBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/builder/item/commerce/CommerceProductGroupCandidateUrtItemBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/builder/item/event_summary/EventCandidateUrtItemBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/builder/item/generic_summary/GenericSummaryActionBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/builder/item/generic_summary/GenericSummaryCandidateUrtItemBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/builder/item/generic_summary/GenericSummaryContextBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/builder/item/icon_label/IconLabelCandidateUrtItemBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/builder/item/message/CompactPromptCandidateUrtItemStringCenterBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/builder/item/message/InlinePromptCandidateUrtItemStringCenterBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/builder/item/message/MessageTextActionBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/builder/item/message/UserFacePileBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/builder/item/moment/MomentAnnotationCandidateUrtItemBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/builder/item/relevance_prompt/RelevancePromptCandidateUrtItemStringCenterBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/builder/item/suggestion/SpellingSuggestionCandidateUrtItemBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/builder/item/tile/TileCandidateUrtItemBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/builder/item/topic/ParamTopicDisplayTypeBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/builder/item/topic/ParamTopicFunctionalityTypeBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/builder/item/topic/StaticTopicDisplayTypeBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/builder/item/topic/StaticTopicFunctionalityTypeBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/builder/item/topic/TopicCandidateUrtItemBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/builder/item/topic/VerticalGridTopicCandidateUrtItemBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/builder/item/trend/TrendCandidateUrtItemBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/builder/item/trend/TrendMetaDescriptionBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/builder/item/trend/TrendPromotedMetadataBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/builder/item/tweet/TweetCandidateUrtItemBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/builder/item/twitter_list/TwitterListCandidateUrtItemBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/builder/item/unified_trend_event/UnifiedTrendEventCandidateUrtItemBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/builder/item/user/UserCandidateUrtItemBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/builder/metadata/ClientEventInfoBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/builder/metadata/ConversationTweetClientEventDetailsBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/builder/metadata/StaticUrlBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/builder/metadata/TopicClientEventDetailsBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/builder/metadata/TopicNotInterestedFeedbackActionInfoBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/builder/metadata/TopicTweetClientEventDetailsBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/builder/metadata/TopicsToFollowModuleMetadataBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/builder/metadata/WhoToFollowFeedbackActionInfoBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/builder/operation/CursorCandidateUrtOperationBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/builder/promoted/FeaturePromotedMetadataBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/builder/richtext/RichTextBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/builder/richtext/RichTextMarkupUtil.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/builder/richtext/RichTextReferenceObjectBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/builder/richtext/RichTextRtlOptionBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/builder/richtext/StaticRichTextBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/builder/richtext/twitter_text/TwitterTextEntityProcessor.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/builder/richtext/twitter_text/TwitterTextFormatProcessor.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/builder/richtext/twitter_text/TwitterTextRenderer.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/builder/richtext/twitter_text/TwitterTextRendererProcessor.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/builder/richtext/twitter_text/TwitterTextRichTextBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/builder/social_context/FeatureSocialContextBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/builder/social_context/GeneralModuleSocialContextBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/builder/social_context/GeneralSocialContextBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/builder/social_context/WhoToFollowSocialContextBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/builder/stringcenter/ModuleStr.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/builder/stringcenter/Str.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/builder/timeline_module/FeatureModuleDisplayTypeBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/builder/timeline_module/ModuleDynamicShowMoreBehaviorRevealByCountBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/builder/timeline_module/ModuleFooterBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/builder/timeline_module/ModuleHeaderBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/builder/timeline_module/ModuleHeaderDisplayTypeBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/builder/timeline_module/ModuleIdGeneration.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/builder/timeline_module/ModuleShowMoreBehaviorRevealByCountBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/builder/timeline_module/ParamGatedModuleFooterBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/builder/timeline_module/ParamGatedModuleHeaderBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/builder/timeline_module/ParamWhoToFollowModuleDisplayTypeBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/builder/timeline_module/StaticModuleDisplayTypeBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt/builder/timeline_module/TimelineModuleBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/experiments/metrics/BUILD create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/experiments/metrics/MetricDefinitions.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/experiments/metrics/MetricGroup.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/experiments/metrics/MetricTemplateCLIRunner.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/experiments/metrics/MetricTemplates.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/experiments/metrics/PlaceholderConfig.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/feature/featurestorev1/BUILD create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/feature/featurestorev1/FeatureStoreV1QueryUserIdFeature.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/feature/featurestorev1/FeatureStoreV1QueryUserIdTweetCandidateAuthorIdFeature.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/feature/featurestorev1/FeatureStoreV1QueryUserIdTweetCandidateTweetIdFeature.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/feature/featurestorev1/FeatureStoreV1TweetCandidateAuthorIdFeature.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/feature/featurestorev1/FeatureStoreV1TweetCandidateTweetIdFeature.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/feature/featurestorev1/FeatureStoreV1UserCandidateUserIdFeature.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/feature_hydrator/BUILD create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/feature_hydrator/candidate/ads/AdvertiserBrandSafetySettingsFeatureHydrator.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/feature_hydrator/candidate/ads/BUILD create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/feature_hydrator/candidate/decay/BUILD.bazel create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/feature_hydrator/candidate/decay/DecayCandidateFeatureHydrator.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/feature_hydrator/candidate/param_gated/BUILD create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/feature_hydrator/candidate/param_gated/ParamGatedBulkCandidateFeatureHydrator.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/feature_hydrator/candidate/param_gated/ParamGatedCandidateFeatureHydrator.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/feature_hydrator/candidate/param_gated/featurestorev1/BUILD create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/feature_hydrator/candidate/param_gated/featurestorev1/ParamGatedFeatureStoreV1CandidateFeatureHydrator.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/feature_hydrator/candidate/qualityfactor_gated/BUILD.bazel create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/feature_hydrator/candidate/qualityfactor_gated/QualityFactorGatedCandidateFeatureHydrator.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/feature_hydrator/candidate/tweet_is_nsfw/BUILD create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/feature_hydrator/candidate/tweet_is_nsfw/TweetIsNsfwCandidateFeatureHydrator.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/feature_hydrator/candidate/tweet_tlx/BUILD.bazel create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/feature_hydrator/candidate/tweet_tlx/TweetTLXScoreCandidateFeatureHydrator.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/feature_hydrator/candidate/tweet_tweetypie/BUILD create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/feature_hydrator/candidate/tweet_tweetypie/TweetTweetypieCandidateFeatureHydrator.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/feature_hydrator/candidate/tweet_visibility_reason/BUILD create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/feature_hydrator/candidate/tweet_visibility_reason/TweetVisibilityReasonBulkCandidateFeatureHydrator.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/feature_hydrator/query/async/AsyncQueryFeatureHydrator.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/feature_hydrator/query/async/BUILD create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/feature_hydrator/query/cr_ml_ranker/BUILD.bazel create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/feature_hydrator/query/cr_ml_ranker/CrMlRankerCommonQueryFeatureHydrator.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/feature_hydrator/query/cr_ml_ranker/CrMlRankerCommonQueryFeatureHydratorBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/feature_hydrator/query/cr_ml_ranker/RankingConfigBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/feature_hydrator/query/impressed_tweets/BUILD create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/feature_hydrator/query/impressed_tweets/ImpressedTweetsQueryFeatureHydrator.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/feature_hydrator/query/logged_in_only/BUILD create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/feature_hydrator/query/logged_in_only/LoggedInOnlyQueryFeatureHydrator.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/feature_hydrator/query/param_gated/AsyncParamGatedQueryFeatureHydrator.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/feature_hydrator/query/param_gated/BUILD create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/feature_hydrator/query/param_gated/ParamGatedQueryFeatureHydrator.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/feature_hydrator/query/param_gated/featurestorev1/AsyncParamGatedFeatureStoreV1QueryFeatureHydrator.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/feature_hydrator/query/param_gated/featurestorev1/BUILD create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/feature_hydrator/query/param_gated/featurestorev1/ParamGatedFeatureStoreV1QueryFeatureHydrator.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/feature_hydrator/query/qualityfactor_gated/BUILD.bazel create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/feature_hydrator/query/qualityfactor_gated/QualityFactorGatedQueryFeatureHydrator.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/filter/AdaptiveLongIntBloomFilterDedupFilter.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/filter/BUILD create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/filter/ExcludedIdsFilter.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/filter/FeatureFilter.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/filter/FeatureValueConditionalFilter.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/filter/HasAuthorIdFeatureFilter.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/filter/ParamGatedFilter.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/filter/PredicateFilter.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/filter/SnowflakeIdAgeFilter.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/filter/TweetAuthorCountryFilter.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/filter/TweetAuthorIsSelfFilter.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/filter/TweetIsNotReplyFilter.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/filter/TweetLanguageFilter.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/filter/TweetVisibilityFilter.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/filter/UrtUnorderedExcludeIdsCursorFilter.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/filter/list_visibility/BUILD.bazel create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/filter/list_visibility/ListVisibilityFilter.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/filter/tweet_impression/BUILD.bazel create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/filter/tweet_impression/TweetImpressionFilter.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/gate/BUILD create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/gate/DefinedCountryCodeGate.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/gate/FeatureGate.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/gate/FirstPageGate.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/gate/NoCandidatesGate.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/gate/NonEmptyAdsQueryStringGate.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/gate/NonEmptyCandidatesGate.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/gate/QualityFactorGate.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/gate/any_candidates_without_feature/AnyCandidatesWithoutFeatureGate.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/gate/any_candidates_without_feature/BUILD.bazel create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/candidate/ArticleCandidate.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/candidate/AudioSpaceCandidate.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/candidate/BUILD create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/candidate/CardCandidate.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/candidate/CommerceItemCandidate.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/candidate/CursorCandidate.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/candidate/DMConvoCandidate.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/candidate/DMEventCandidate.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/candidate/GenericSummaryCandidate.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/candidate/LabelCandidate.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/candidate/MomentAnnotationCandidate.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/candidate/PromptCandidate.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/candidate/ShowAlertCandidate.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/candidate/TopicCandidate.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/candidate/TweetCandidate.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/candidate/TwitterListCandidate.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/candidate/UserCandidate.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/candidate/ads/AdsCandidate.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/candidate/ads/BUILD create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/candidate/hubble/AdCreativeCandidate.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/candidate/hubble/AdGroupCandidate.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/candidate/hubble/AdUnitCandidate.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/candidate/hubble/BUILD create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/candidate/hubble/CampaignCandidate.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/candidate/hubble/FundingSourceCandidate.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/candidate/suggestion/BUILD create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/candidate/suggestion/QuerySuggestionCandidate.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/candidate/suggestion/SpellingSuggestionCandidate.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/candidate/trends_events/BUILD create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/candidate/trends_events/UnifiedTrendEventCandidate.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/cursor/BUILD create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/cursor/OrderedCursor.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/cursor/PassThroughCursor.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/cursor/UnorderedBloomFilterCursor.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/cursor/UnorderedExcludeIdsCursor.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/cursor/UrtPlaceholderCursor.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/feature/flexible_injection_pipeline/BUILD.bazel create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/presentation/slice/BUILD create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/presentation/slice/SliceItemPresentation.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/presentation/urt/BUILD create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/presentation/urt/ConversationModuleItem.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/presentation/urt/UrtItemPresentation.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/presentation/urt/UrtModulePresentation.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/presentation/urt/UrtOperationPresentation.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/query/ads/AdsQuery.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/query/ads/BUILD create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/module/AccountRecommendationsMixerModule.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/module/BUILD create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/module/ConversationServiceModule.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/module/CrMixerClientModule.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/module/DarkTrafficFilterModule.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/module/EarlybirdModule.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/module/ExploreRankerClientModule.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/module/FollowRecommenderServiceModule.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/module/GizmoduckClientModule.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/module/HomeScorerClientModule.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/module/InterestsDiscoveryServiceModule.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/module/OnboardingTaskServiceModule.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/module/PeopleDiscoveryServiceModule.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/module/SocialGraphServiceModule.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/module/TimelineMixerClientModule.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/module/TimelineRankerClientModule.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/module/TimelineScorerClientModule.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/module/TimelineServiceClientModule.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/module/TweetImpressionStoreModule.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/module/TweetyPieClientModule.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/module/UserSessionStoreModule.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/module/cr_ml_ranker/BUILD.bazel create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/module/cr_ml_ranker/CrMlRankerModule.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/module/http/BUILD create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/module/http/FinagleHttpClientModule.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/module/http/FinagleHttpClientWithCredentialProxyModule.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/module/http/FinagleHttpClientWithProxyModule.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/module/http/FinatraHttpClientModule.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/module/http/FinatraHttpClientWithCredentialProxyModule.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/module/http/FinatraHttpClientWithProxyModule.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/module/http/ProxyCredentialsModule.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/pipeline/candidate/ads/AdsCandidatePipelineConfig.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/pipeline/candidate/ads/AdsCandidatePipelineConfigBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/pipeline/candidate/ads/AdsCandidatePipelineQueryTransformer.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/pipeline/candidate/ads/AdsCandidatePipelineResultsTransformer.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/pipeline/candidate/ads/AdsDependentCandidatePipelineConfig.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/pipeline/candidate/ads/AdsDependentCandidatePipelineConfigBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/pipeline/candidate/ads/AdsDependentCandidatePipelineQueryTransformer.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/pipeline/candidate/ads/AdsDisplayLocationBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/pipeline/candidate/ads/BUILD create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/pipeline/candidate/ads/CountNumOrganicItems.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/pipeline/candidate/ads/GetOrganicItemIds.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/pipeline/candidate/ads/PromotedTweetsOnlyFilter.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/pipeline/candidate/ads/ValidAdImpressionIdFilter.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/pipeline/candidate/flexible_injection_pipeline/BUILD.bazel create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/pipeline/candidate/flexible_injection_pipeline/FlipPromptCandidatePipelineConfig.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/pipeline/candidate/flexible_injection_pipeline/FlipPromptCandidatePipelineConfigBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/pipeline/candidate/flexible_injection_pipeline/FlipPromptDependentCandidatePipelineConfig.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/pipeline/candidate/flexible_injection_pipeline/FlipPromptDependentCandidatePipelineConfigBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/pipeline/candidate/flexible_injection_pipeline/transformer/BUILD create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/pipeline/candidate/flexible_injection_pipeline/transformer/FlipCandidateFeatureTransformer.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/pipeline/candidate/flexible_injection_pipeline/transformer/FlipInjectionParams.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/pipeline/candidate/flexible_injection_pipeline/transformer/FlipQueryTransformer.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/pipeline/candidate/flexible_injection_pipeline/transformer/PromptResultsTransformer.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/pipeline/candidate/who_to_follow_module/BUILD.bazel create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/pipeline/candidate/who_to_follow_module/WhoToFollowArmCandidateDecorator.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/pipeline/candidate/who_to_follow_module/WhoToFollowArmCandidatePipelineConfig.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/pipeline/candidate/who_to_follow_module/WhoToFollowArmCandidatePipelineQueryTransformer.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/pipeline/candidate/who_to_follow_module/WhoToFollowArmDependentCandidatePipelineConfig.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/pipeline/candidate/who_to_follow_module/WhoToFollowArmDependentCandidatePipelineConfigBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/pipeline/candidate/who_to_follow_module/WhoToFollowArmResponseFeatureTransformer.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/pipeline/candidate/who_to_follow_module/WhoToFollowCandidateDecorator.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/pipeline/candidate/who_to_follow_module/WhoToFollowCandidatePipelineConfig.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/pipeline/candidate/who_to_follow_module/WhoToFollowCandidatePipelineConfigBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/pipeline/candidate/who_to_follow_module/WhoToFollowCandidatePipelineQueryTransformer.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/pipeline/candidate/who_to_follow_module/WhoToFollowClientEventDetailsBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/pipeline/candidate/who_to_follow_module/WhoToFollowDependentCandidatePipelineConfig.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/pipeline/candidate/who_to_follow_module/WhoToFollowDependentCandidatePipelineConfigBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/pipeline/candidate/who_to_follow_module/WhoToFollowResponseFeatureTransformer.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/premarshaller/cursor/BUILD create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/premarshaller/cursor/CursorSerializer.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/premarshaller/cursor/UrtCursorSerializer.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/premarshaller/slice/BUILD create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/premarshaller/slice/SliceDomainMarshaller.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/premarshaller/slice/builder/BUILD create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/premarshaller/slice/builder/OrderedNextCursorBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/premarshaller/slice/builder/OrderedNextCursorUpdater.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/premarshaller/slice/builder/OrderedPreviousCursorBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/premarshaller/slice/builder/OrderedPreviousCursorUpdater.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/premarshaller/slice/builder/ShouldInclude.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/premarshaller/slice/builder/SliceBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/premarshaller/slice/builder/SliceCursorBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/premarshaller/slice/builder/SliceCursorUpdater.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/premarshaller/urp/BUILD create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/premarshaller/urp/UrpDomainMarshaller.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/premarshaller/urp/builder/BUILD create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/premarshaller/urp/builder/PageBodyBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/premarshaller/urp/builder/PageHeaderBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/premarshaller/urp/builder/PageNavBarBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/premarshaller/urp/builder/StaticTimelineScribeConfigBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/premarshaller/urp/builder/TimelineScribeConfigBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/premarshaller/urt/BUILD create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/premarshaller/urt/UndecoratedUrtDomainMarshaller.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/premarshaller/urt/UrtDomainMarshaller.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/premarshaller/urt/builder/AddEntriesInstructionBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/premarshaller/urt/builder/AddEntriesWithAddToModuleInstructionBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/premarshaller/urt/builder/AddEntriesWithPinnedAndReplaceInstructionBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/premarshaller/urt/builder/AddEntriesWithReplaceAndShowAlertInstructionBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/premarshaller/urt/builder/AddEntriesWithReplaceInstructionBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/premarshaller/urt/builder/AddEntriesWithShowCoverInstructionBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/premarshaller/urt/builder/AddToModuleInstructionBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/premarshaller/urt/builder/BUILD create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/premarshaller/urt/builder/BaseUnorderedExcludeIdsBottomCursorBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/premarshaller/urt/builder/ClearCacheInstructionBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/premarshaller/urt/builder/FeaturePassThroughCursorBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/premarshaller/urt/builder/IncludeInstruction.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/premarshaller/urt/builder/MarkUnreadInstructionBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/premarshaller/urt/builder/OrderedBottomCursorBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/premarshaller/urt/builder/OrderedGapCursorBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/premarshaller/urt/builder/OrderedTopCursorBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/premarshaller/urt/builder/PinEntryInstructionBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/premarshaller/urt/builder/PlaceholderTopCursorBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/premarshaller/urt/builder/ReplaceEntryInstructionBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/premarshaller/urt/builder/ShowAlertInstructionBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/premarshaller/urt/builder/ShowCoverInstructionBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/premarshaller/urt/builder/StaticTimelineScribeConfigBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/premarshaller/urt/builder/TerminateInstructionBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/premarshaller/urt/builder/TimelineScribeConfigBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/premarshaller/urt/builder/UnorderedBloomFilterBottomCursorBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/premarshaller/urt/builder/UnorderedExcludeIdsBottomCursorBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/premarshaller/urt/builder/UnorderedExcludeIdsSeqBottomCursorBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/premarshaller/urt/builder/UrtBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/premarshaller/urt/builder/UrtCursorBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/premarshaller/urt/builder/UrtCursorUpdater.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/premarshaller/urt/builder/UrtInstructionBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/premarshaller/urt/builder/UrtMetadataBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/scorer/common/BUILD create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/scorer/common/MLModelInferenceClient.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/scorer/common/ManagedModelClient.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/scorer/common/ModelSelector.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/scorer/common/NaviModelClient.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/scorer/cortex/BUILD create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/scorer/cortex/CortexManagedInferenceServiceDataRecordScorer.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/scorer/cortex/CortexManagedInferenceServiceDataRecordScorerBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/scorer/cortex/CortexManagedInferenceServiceTensorScorer.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/scorer/cortex/CortexManagedInferenceServiceTensorScorerBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/scorer/cortex/ModelFeatureExtractor.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/scorer/cr_ml_ranker/BUILD.bazel create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/scorer/cr_ml_ranker/CrMlRankerScorer.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/scorer/cr_ml_ranker/CrMlRankerStitchClient.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/scorer/deepbird/BUILD create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/scorer/deepbird/BaseDeepbirdV2Scorer.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/scorer/deepbird/DeepbirdV2PredictionServerScorer.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/scorer/deepbird/LollyPredictionEngineScorer.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/scorer/deepbird/TensorflowPredictionEngineScorer.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/scorer/param_gated/BUILD create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/scorer/param_gated/ParamGatedScorer.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/scorer/qualityfactor_gated/BUILD create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/scorer/qualityfactor_gated/QualityFactorGatedScorer.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/scorer/tensorbuilder/BUILD create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/scorer/tensorbuilder/BooleanInferInputTensorBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/scorer/tensorbuilder/BytesInferInputTensorBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/scorer/tensorbuilder/CandidateInferInputTensorBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/scorer/tensorbuilder/Float32InferInputTensorBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/scorer/tensorbuilder/FloatTensorInferInputTensorBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/scorer/tensorbuilder/InferInputTensorBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/scorer/tensorbuilder/Int64InferInputTensorBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/scorer/tensorbuilder/ModelInferRequestBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/scorer/tensorbuilder/QueryInferInputTensorBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/scorer/tensorbuilder/SparseMapInferInputTensorBuilder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/scorer/tweet_tlx/BUILD.bazel create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/scorer/tweet_tlx/TweetTLXStratoScorer.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/scorer/tweet_tlx/TweetTLXThriftScorer.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/selector/BUILD create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/selector/Bucketer.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/selector/CandidateMergeStrategy.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/selector/CandidatePositionInResults.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/selector/DeduplicationKey.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/selector/DropAllCandidates.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/selector/DropDuplicateCandidates.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/selector/DropDuplicateModuleItemCandidates.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/selector/DropDuplicateResults.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/selector/DropFilteredCandidates.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/selector/DropFilteredModuleItemCandidates.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/selector/DropMaxCandidates.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/selector/DropMaxModuleItemCandidates.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/selector/DropMaxResults.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/selector/DropModuleTooFewModuleItemResults.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/selector/DropNonDuplicateCandidates.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/selector/DropOrthogonalCandidates.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/selector/DropRequestedMaxModuleItemCandidates.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/selector/DropRequestedMaxResults.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/selector/DropSelector.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/selector/DropTooFewResults.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/selector/DynamicPositionSelector.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/selector/InsertAppendIntoModuleCandidates.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/selector/InsertAppendPatternResults.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/selector/InsertAppendRatioResults.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/selector/InsertAppendResults.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/selector/InsertAppendWeaveResults.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/selector/InsertAppendWithoutFeatureResults.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/selector/InsertDynamicPositionResults.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/selector/InsertFixedPositionIntoModuleCandidates.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/selector/InsertFixedPositionResults.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/selector/InsertIntoModule.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/selector/InsertPerCandidateDynamicPositionResults.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/selector/InsertRandomPositionResults.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/selector/InsertRelativePositionResults.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/selector/InsertSelector.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/selector/SelectConditionally.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/selector/SelectFromSubpoolCandidates.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/selector/UpdateSortCandidates.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/selector/UpdateSortModuleItemCandidates.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/selector/UpdateSortResults.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/selector/ads/AdsInjector.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/selector/ads/BUILD.bazel create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/selector/ads/InsertAdResults.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/selector/sorter/BUILD create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/selector/sorter/FeatureValueSorter.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/selector/sorter/RandomShuffleSorter.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/selector/sorter/ReverseSorter.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/selector/sorter/SortOrder.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/selector/sorter/SorterFromOrdering.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/selector/sorter/SorterProvider.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/selector/sorter/featurestorev1/BUILD create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/selector/sorter/featurestorev1/FeatureStoreV1FeatureValueSorter.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/side_effect/BUILD create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/side_effect/KafkaPublishingSideEffect.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/side_effect/ParamGatedPipelineResultSideEffect.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/side_effect/ScribeClientEventSideEffect.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/side_effect/ScribeLogEventAsyncSideEffect.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/side_effect/ScribeLogEventSideEffect.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/side_effect/StratoInsertSideEffect.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/side_effect/UserSessionStoreUpdateSideEffect.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/side_effect/metrics/BUILD.bazel create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/side_effect/metrics/CandidateMetricFunction.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/side_effect/metrics/ScribeClientEventMetricsSideEffect.scala create mode 100644 product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/side_effect/metrics/ScribeClientEventMetricsSideEffectBuilder.scala create mode 100644 product-mixer/core/src/main/java/com/twitter/product_mixer/core/product/guice/scope/BUILD create mode 100644 product-mixer/core/src/main/java/com/twitter/product_mixer/core/product/guice/scope/ProductScoped.java create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/controllers/AlertConfig.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/controllers/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/controllers/DebugTwitterContext.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/controllers/GetComponentRegistryHandler.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/controllers/GetDebugConfigurationHandler.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/controllers/PredicateConfig.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/controllers/ProductMixerController.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/controllers/QualityFactorMonitoringConfig.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/feature/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/feature/Feature.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/feature/datarecord/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/feature/datarecord/DataRecordCompatible.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/feature/datarecord/DataRecordFeature.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/feature/featuremap/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/feature/featuremap/FeatureMap.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/feature/featuremap/FeatureMapBuilder.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/feature/featuremap/FeatureMapException.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/feature/featuremap/FeatureMapSerializer.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/feature/featuremap/asyncfeaturemap/AsyncFeatureMap.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/feature/featuremap/asyncfeaturemap/AsyncFeatureMapSerializer.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/feature/featuremap/asyncfeaturemap/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/feature/featuremap/datarecord/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/feature/featuremap/datarecord/DataRecordConverter.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/feature/featuremap/datarecord/DataRecordExtractor.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/feature/featuremap/datarecord/FeaturesScope.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/feature/featuremap/featurestorev1/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/feature/featuremap/featurestorev1/FeatureStoreV1FeatureMap.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/feature/featurestorev1/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/feature/featurestorev1/FeatureStoreV1Entity.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/feature/featurestorev1/FeatureStoreV1Feature.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/feature/featurestorev1/featurevalue/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/feature/featurestorev1/featurevalue/FeatureStoreV1Response.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/access_policy/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/candidate_source/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/candidate_source/CandidateSource.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/candidate_source/CandidatesWithSourceFeatures.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/candidate_source/PassthroughCandidateSource.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/candidate_source/StaticCandidateSource.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/candidate_source/product_pipeline/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/candidate_source/product_pipeline/ProductPipelineCandidateSource.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/candidate_source/strato/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/candidate_source/strato/StratoErrCategorizer.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/candidate_source/strato/StratoKeyFetcherSeqSource.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/candidate_source/strato/StratoKeyFetcherSource.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/candidate_source/strato/StratoKeyFetcherWithSourceFeaturesSource.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/candidate_source/strato/StratoKeyView.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/candidate_source/strato/StratoKeyViewFetcherSeqSource.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/candidate_source/strato/StratoKeyViewFetcherSource.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/candidate_source/strato/StratoKeyViewFetcherWithSourceFeaturesSource.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/common/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/common/CandidateScope.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/common/access_policy/AccessPolicy.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/common/access_policy/AccessPolicyEvaluator.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/common/access_policy/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/common/access_policy/WithDebugAccessPolicies.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/common/alert/Alert.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/common/alert/AlertType.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/common/alert/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/common/alert/EmptyResponseRateAlert.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/common/alert/GenericClientLatencyAlert.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/common/alert/GenericClientSuccessRateAlert.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/common/alert/GenericClientThroughputAlert.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/common/alert/IsObservableFromStrato.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/common/alert/LatencyAlert.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/common/alert/NotificationGroup.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/common/alert/Percentile.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/common/alert/ResponseSizeAlert.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/common/alert/Source.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/common/alert/StratoColumnAlert.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/common/alert/SuccessRateAlert.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/common/alert/ThroughputAlert.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/common/alert/predicate/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/common/alert/predicate/MetricGranularity.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/common/alert/predicate/Operator.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/common/alert/predicate/Predicate.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/common/alert/predicate/TriggerIfAbove.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/common/alert/predicate/TriggerIfBelow.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/common/alert/predicate/TriggerIfLatencyAbove.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/configapi/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/configapi/ConfigBuilder.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/configapi/ParamsBuilder.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/configapi/RequestContext.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/configapi/RequestContextBuilder.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/configapi/StaticParam.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/configapi/registry/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/configapi/registry/GlobalParamConfig.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/configapi/registry/GlobalParamRegistry.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/configapi/registry/ParamConfig.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/configapi/registry/ParamConfigBuilder.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/decorator/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/decorator/CandidateDecorator.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/decorator/Decoration.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/decorator/slice/builder/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/decorator/slice/builder/CandidateSliceItemBuilder.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/decorator/urt/builder/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/decorator/urt/builder/CandidateUrtEntryBuilder.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/decorator/urt/builder/icon/BaseHorizonIconBuilder.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/decorator/urt/builder/item/alert/BaseDurationBuilder.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/decorator/urt/builder/item/alert/BaseShowAlertColorConfigurationBuilder.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/decorator/urt/builder/item/alert/BaseShowAlertDisplayLocationBuilder.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/decorator/urt/builder/item/alert/BaseShowAlertIconDisplayInfoBuilder.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/decorator/urt/builder/item/alert/BaseShowAlertNavigationMetadataBuilder.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/decorator/urt/builder/item/alert/BaseShowAlertUserIdsBuilder.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/decorator/urt/builder/item/topic/BaseTopicDisplayTypeBuilder.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/decorator/urt/builder/item/topic/BaseTopicFunctionalityTypeBuilder.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/decorator/urt/builder/item/tweet/BaseEntryIdToReplaceBuilder.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/decorator/urt/builder/item/tweet/BaseTimelinesScoreInfoBuilder.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/decorator/urt/builder/item/tweet/BaseTweetHighlightsBuilder.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/decorator/urt/builder/item/user/BaseUserReactiveTriggersBuilder.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/decorator/urt/builder/metadata/BaseClientEventDetailsBuilder.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/decorator/urt/builder/metadata/BaseClientEventInfoBuilder.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/decorator/urt/builder/metadata/BaseFeedbackActionInfoBuilder.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/decorator/urt/builder/metadata/BaseModuleStr.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/decorator/urt/builder/metadata/BaseStr.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/decorator/urt/builder/metadata/BaseUrlBuilder.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/decorator/urt/builder/promoted/BasePromotedMetadataBuilder.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/decorator/urt/builder/richtext/BaseRichTextBuilder.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/decorator/urt/builder/social_context/BaseModuleSocialContextBuilder.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/decorator/urt/builder/social_context/BaseSocialContextBuilder.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/decorator/urt/builder/stringcenter/BaseModuleStringCenterPlaceholderBuilder.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/decorator/urt/builder/stringcenter/BaseStringCenterPlaceholderBuilder.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/decorator/urt/builder/timeline_module/BaseModuleDisplayTypeBuilder.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/decorator/urt/builder/timeline_module/BaseModuleFooterBuilder.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/decorator/urt/builder/timeline_module/BaseModuleHeaderBuilder.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/decorator/urt/builder/timeline_module/BaseModuleHeaderDisplayTypeBuilder.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/decorator/urt/builder/timeline_module/BaseModuleMetadataBuilder.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/decorator/urt/builder/timeline_module/BaseModuleShowMoreBehaviorBuilder.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/decorator/urt/builder/timeline_module/BaseTimelineModuleBuilder.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/feature_hydrator/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/feature_hydrator/CandidateFeatureHydrator.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/feature_hydrator/FeatureHydrator.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/feature_hydrator/HydratorCandidateResult.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/feature_hydrator/QueryFeatureHydrator.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/feature_hydrator/featurestorev1/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/feature_hydrator/featurestorev1/FeatureStoreDatasetErrorHandler.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/feature_hydrator/featurestorev1/FeatureStoreV1CandidateFeatureHydrator.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/feature_hydrator/featurestorev1/FeatureStoreV1DynamicClientBuilder.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/feature_hydrator/featurestorev1/FeatureStoreV1HydrationConfig.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/feature_hydrator/featurestorev1/FeatureStoreV1QueryFeatureHydrator.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/filter/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/filter/Filter.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/filter/FilterResult.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/gate/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/gate/Gate.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/gate/GateResult.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/gate/ShouldContinue.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/TransportMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/request/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/request/ClientContextMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/request/ClientContextUnmarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/request/FeatureValueUnmarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/graphql/contextual_ref/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/graphql/contextual_ref/ContextualTweetRefMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/graphql/contextual_ref/OuterTweetContextMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/graphql/contextual_ref/TweetHydrationContextMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/rtf/safety_level/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/rtf/safety_level/SafetyLevelMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/slice/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/slice/CursorTypeMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/slice/SliceItemMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/slice/SliceTransportMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urp/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urp/PageBodyMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urp/PageHeaderMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urp/PageNavBarMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urp/SegmentedTimelineMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urp/SegmentedTimelinesMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urp/TimelineKeyMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urp/TitleNavBarMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urp/TopicPageHeaderDisplayTypeMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urp/TopicPageHeaderFacepileMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urp/TopicPageHeaderMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urp/TopicPageNavBarMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urp/UrpTransportMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urp/UrpTransportMarshallerBuilder.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/AddEntriesInstructionMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/AddToModuleInstructionMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/CoverMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/MarkEntriesUnreadInstructionMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/ModuleItemMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/ModuleItemTreeDisplayMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/PinEntryInstructionMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/ReaderModeConfigMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/ReplaceEntryInstructionMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/ShowAlertInstructionMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/TerminateTimelineInstructionMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/TimelineEntryContentMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/TimelineEntryMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/TimelineInstructionMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/TimelineItemContentMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/TimelineItemMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/TimelineMetadataMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/TimelineModuleMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/TimelineOperationMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/TimelineScribeConfigMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/UrtTransportMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/UrtTransportMarshallerBuilder.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/alert/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/alert/ShowAlertColorConfigurationMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/alert/ShowAlertDisplayLocationMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/alert/ShowAlertIconDisplayInfoMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/alert/ShowAlertIconMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/alert/ShowAlertNavigationMetadataMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/alert/ShowAlertTypeMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/button/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/button/ButtonStyleMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/button/CtaButtonMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/button/IconCtaButtonMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/button/TextCtaButtonMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/color/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/color/ColorMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/color/ColorPaletteMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/color/RosettaColorMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/cover/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/cover/CoverContentMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/cover/CoverCtaBehaviorMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/cover/CoverCtaMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/cover/CoverImageMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/cover/FullCoverContentMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/cover/FullCoverDisplayTypeMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/cover/HalfCoverContentMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/cover/HalfCoverDisplayTypeMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/icon/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/icon/HorizonIconMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/item/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/item/article/ArticleDisplayTypeMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/item/article/ArticleItemMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/item/article/ArticleSeedTypeMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/item/audio_space/AudioSpaceItemMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/item/card/CardDisplayTypeMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/item/card/CardItemMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/item/commerce/CommerceProductGroupItemMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/item/commerce/CommerceProductItemMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/item/conversation_annotation/ConversationAnnotationMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/item/conversation_annotation/ConversationAnnotationTypeMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/item/event/EventSummaryDisplayTypeMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/item/event/EventSummaryItemMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/item/forward_pivot/ForwardPivotDisplayTypeMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/item/forward_pivot/ForwardPivotMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/item/forward_pivot/SoftInterventionDisplayTypeMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/item/generic_summary_item/GenericSummaryActionMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/item/generic_summary_item/GenericSummaryContextMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/item/generic_summary_item/GenericSummaryDisplayTypeMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/item/generic_summary_item/GenericSummaryItemMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/item/highlight/HighlightedSectionMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/item/icon_label/IconLabelItemMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/item/label/LabelDisplayTypeMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/item/label/LabelItemMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/item/message/CompactPromptMessageContentMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/item/message/HeaderImagePromptMessageContentMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/item/message/InlinePromptMessageContentMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/item/message/MessageActionMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/item/message/MessageActionTypeMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/item/message/MessageContentMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/item/message/MessageImageMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/item/message/MessagePromptItemMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/item/message/MessageTextActionMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/item/message/UserFacepileDisplayTypeMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/item/message/UserFacepileMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/item/moment/MomentAnnotationItemMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/item/prompt/PromptContentMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/item/prompt/PromptItemMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/item/prompt/RelevancePromptContentMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/item/prompt/RelevancePromptDisplayTypeMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/item/prompt/RelevancePromptFollowUpFeedbackTypeMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/item/prompt/RelevancePromptFollowUpTextInputMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/item/suggestion/SpellingActionTypeMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/item/suggestion/SpellingItemMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/item/suggestion/TextResultMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/item/thread/ThreadHeaderContentMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/item/thread/ThreadHeaderItemMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/item/tile/CallToActionTileContentMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/item/tile/StandardTileContentMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/item/tile/TileContentMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/item/tile/TileItemMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/item/tombstone/TombstoneDisplayTypeMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/item/tombstone/TombstoneInfoMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/item/tombstone/TombstoneItemMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/item/topic/TopicDisplayTypeMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/item/topic/TopicFollowPromptDisplayTypeMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/item/topic/TopicFollowPromptItemMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/item/topic/TopicFunctionalityTypeMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/item/topic/TopicItemMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/item/trend/TrendItemMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/item/tweet/TimelinesScoreInfoMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/item/tweet/TweetDisplayTypeMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/item/tweet/TweetHighlightsMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/item/tweet/TweetItemMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/item/tweet_composer/TweetComposerDisplayTypeMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/item/tweet_composer/TweetComposerItemMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/item/twitter_list/TwitterListDisplayTypeMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/item/twitter_list/TwitterListItemMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/item/user/UserDisplayTypeMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/item/user/UserItemMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/item/user/UserReactiveTriggersMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/item/vertical_grid_item/VerticalGridItemContentMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/item/vertical_grid_item/VerticalGridItemMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/item/vertical_grid_item/VerticalGridItemTileStyleMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/item/vertical_grid_item/VerticalGridItemTopicFunctionalityTypeMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/item/vertical_grid_item/VerticalGridItemTopicTileMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/media/AspectRatioMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/media/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/media/BroadcastIdMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/media/MediaEntityMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/media/MediaKeyMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/media/MediaMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/media/RectMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/media/TweetMediaMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/metadata/ArticleDetailsMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/metadata/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/metadata/BadgeMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/metadata/CallbackMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/metadata/ChildFeedbackActionMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/metadata/ClientEventDetailsMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/metadata/ClientEventInfoMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/metadata/CommerceDetailsMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/metadata/ConfirmationDisplayTypeMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/metadata/ConversationDetailsMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/metadata/ConversationSectionMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/metadata/DismissInfoMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/metadata/FeedbackActionMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/metadata/FeedbackDisplayContextMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/metadata/FeedbackInfoMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/metadata/FeedbackTypeMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/metadata/GeneralContextMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/metadata/GeneralContextTypeMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/metadata/ImageAnimationTypeMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/metadata/ImageDisplayTypeMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/metadata/ImageVariantMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/metadata/LiveEventDetailsMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/metadata/RichFeedbackBehaviorMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/metadata/SocialContextMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/metadata/TimelinesDetailsMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/metadata/TopicContextFunctionalityTypeMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/metadata/TopicContextMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/metadata/UrlMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/metadata/UrlTypeMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/metadata/UrtEndpointOptionsMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/operation/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/operation/CursorDisplayTreatmentMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/operation/CursorItemMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/operation/CursorOperationMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/operation/CursorTypeMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/promoted/AdMetadataContainerMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/promoted/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/promoted/CallToActionMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/promoted/ClickTrackingInfoMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/promoted/DisclaimerTypeMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/promoted/DisclosureTypeMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/promoted/DynamicPrerollTypeMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/promoted/MediaInfoMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/promoted/PrerollMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/promoted/PrerollMetadataMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/promoted/PromotedMetadataMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/promoted/SkAdNetworkDataMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/promoted/SponsorshipTypeMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/promoted/UrlOverrideTypeMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/promoted/VideoVariantsMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/reaction/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/reaction/TimelineReactionMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/richtext/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/richtext/ReferenceObjectMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/richtext/RichTextAlignmentMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/richtext/RichTextEntityMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/richtext/RichTextFormatMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/richtext/RichTextMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/timeline_module/AdsMetadataMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/timeline_module/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/timeline_module/GridCarouselMetadataMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/timeline_module/ModuleConversationMetadataMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/timeline_module/ModuleDisplayTypeMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/timeline_module/ModuleFooterMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/timeline_module/ModuleHeaderDisplayTypeMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/timeline_module/ModuleHeaderMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/timeline_module/ModuleMetadataMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/timeline_module/ModuleShowMoreBehaviorMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt/timeline_module/ModuleShowMoreBehaviorRevealByCountMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/premarshaller/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/premarshaller/DomainMarshaller.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/scorer/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/scorer/ScoredCandidateResult.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/scorer/Scorer.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/selector/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/selector/Selector.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/selector/SelectorResult.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/side_effect/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/side_effect/ExecuteSynchronously.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/side_effect/PipelineResultSideEffect.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/side_effect/SideEffect.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/transformer/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/transformer/CandidateFeatureTransformer.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/transformer/CandidatePipelineQueryTransformer.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/transformer/CandidatePipelineResultsTransformer.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/transformer/FeatureTransformer.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/transformer/Transformer.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/gate/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/gate/DenyLoggedOutUsersGate.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/gate/ParamGate.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/gate/ParamNotGate.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/common/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/common/CandidateWithFeatures.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/common/Component.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/common/Conditionally.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/common/UniversalNoun.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/common/identifier/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/common/identifier/CandidatePipelineIdentifier.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/common/identifier/CandidateSourceIdentifier.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/common/identifier/ComponentIdentifier.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/common/identifier/ComponentIdentifierSerializer.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/common/identifier/ComponentIdentifierStack.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/common/identifier/ComponentIdentifierStackSerializer.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/common/identifier/DecoratorIdentifier.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/common/identifier/DomainMarshallerIdentifier.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/common/identifier/FeatureHydratorIdentifier.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/common/identifier/FilterIdentifier.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/common/identifier/GateIdentifier.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/common/identifier/MixerPipelineIdentifier.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/common/identifier/PipelineStepIdentifier.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/common/identifier/PlatformIdentifier.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/common/identifier/ProductIdentifier.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/common/identifier/ProductPipelineIdentifier.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/common/identifier/RecommendationPipelineIdentifier.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/common/identifier/RootIdentifier.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/common/identifier/ScorerIdentifier.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/common/identifier/ScoringPipelineIdentifier.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/common/identifier/SelectorIdentifier.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/common/identifier/SideEffectIdentifier.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/common/identifier/TransformerIdentifier.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/common/identifier/TransportMarshallerIdentifier.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/common/presentation/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/common/presentation/CandidateFeatures.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/common/presentation/CandidateWithDetails.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/common/presentation/ItemPresentation.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/common/presentation/ModulePresentation.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/common/presentation/UniversalPresentation.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/common/presentation/slice/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/common/presentation/slice/BaseSliceItemPresentation.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/common/presentation/urt/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/common/presentation/urt/BaseUrtItemPresentation.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/common/presentation/urt/BaseUrtModulePresentation.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/common/presentation/urt/BaseUrtOperationPresentation.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/common/presentation/urt/IsDispensable.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/common/presentation/urt/WithItemTreeDisplay.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/HasMarshalling.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/request/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/request/ClientContext.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/request/DebugOptions.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/request/DebugParams.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/request/HasExcludedIds.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/request/HasSerializedRequestCursor.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/request/Product.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/request/ProductContext.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/request/Request.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/rtf/safety_level/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/rtf/safety_level/SafetyLevel.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/slice/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/slice/SliceItem.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urp/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urp/Page.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urp/PageBody.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urp/PageHeader.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urp/PageNavBar.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urp/SegmentedTimeline.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urp/TimelineKey.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urp/TopicPageHeaderDisplayType.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urp/TopicPageHeaderFacepile.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/Cover.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/EntryNamespace.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/HasEntryIdentifier.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/HasExpirationTime.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/HasSortIndex.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/ModuleItemTreeDisplay.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/ReaderModeConfig.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/ShowAlert.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/Timeline.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/TimelineEntry.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/TimelineInstruction.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/TimelineMetadata.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/TimelineScribeConfig.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/alert/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/alert/ShowAlertColorConfiguration.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/alert/ShowAlertDisplayLocation.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/alert/ShowAlertIcon.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/alert/ShowAlertIconDisplayInfo.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/alert/ShowAlertNavigationMetadata.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/alert/ShowAlertType.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/button/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/button/ButtonStyle.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/button/CtaButton.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/color/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/color/Color.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/color/ColorPalette.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/color/RosettaColor.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/contextual_ref/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/contextual_ref/ContextualTweetRef.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/contextual_ref/OuterTweetContext.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/contextual_ref/TweetHydrationContext.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/cover/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/cover/CoverContent.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/cover/CoverCta.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/cover/CoverCtaBehavior.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/cover/CoverImage.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/cover/FullCoverDisplayType.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/cover/HalfCoverDisplayType.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/cover/ShowCover.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/icon/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/icon/HorizonIcon.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/item/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/item/article/ArticleDisplayType.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/item/article/ArticleItem.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/item/article/ArticleSeedType.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/item/audio_space/AudioSpaceItem.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/item/card/CardDisplayType.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/item/card/CardItem.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/item/commerce/CommerceProductGroupItem.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/item/commerce/CommerceProductItem.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/item/conversation_annotation/ConversationAnnotation.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/item/conversation_annotation/ConversationAnnotationType.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/item/event/EventSummaryDisplayType.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/item/event/EventSummaryItem.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/item/forward_pivot/ForwardPivot.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/item/forward_pivot/ForwardPivotDisplayType.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/item/forward_pivot/SoftInterventionDisplayType.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/item/generic_summary/GenericSummaryAction.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/item/generic_summary/GenericSummaryContext.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/item/generic_summary/GenericSummaryDisplayType.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/item/generic_summary/GenericSummaryItem.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/item/highlight/HighlightedSection.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/item/icon_label/IconLabelItem.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/item/label/LabelDisplayType.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/item/label/LabelItem.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/item/message/MessageAction.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/item/message/MessageActionType.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/item/message/MessageContent.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/item/message/MessageImage.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/item/message/MessagePromptItem.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/item/message/MessageTextAction.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/item/message/UserFacepile.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/item/message/UserFacepileDisplayType.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/item/moment/MomentAnnotationItem.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/item/prompt/PromptContent.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/item/prompt/PromptItem.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/item/prompt/RelevancePromptDisplayType.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/item/prompt/RelevancePromptFollowUpFeedbackType.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/item/suggestion/SpellingActionType.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/item/suggestion/SpellingItem.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/item/suggestion/TextResult.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/item/thread/ThreadHeaderContent.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/item/thread/ThreadHeaderItem.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/item/tile/TileContent.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/item/tile/TileItem.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/item/tombstone/TombstoneDisplayType.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/item/tombstone/TombstoneInfo.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/item/tombstone/TombstoneItem.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/item/topic/TopicDisplayType.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/item/topic/TopicFollowPromptDisplayType.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/item/topic/TopicFollowPromptItem.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/item/topic/TopicFunctionalityType.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/item/topic/TopicItem.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/item/trend/TrendItem.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/item/tweet/TimelinesScoreInfo.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/item/tweet/TweetDisplayType.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/item/tweet/TweetHighlights.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/item/tweet/TweetItem.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/item/tweet_composer/TweetComposerDisplayType.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/item/tweet_composer/TweetComposerItem.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/item/twitter_list/TwitterListDisplayType.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/item/twitter_list/TwitterListItem.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/item/user/UserDisplayType.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/item/user/UserItem.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/item/user/UserReactiveTriggers.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/item/vertical_grid_item/VerticalGridItem.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/item/vertical_grid_item/VerticalGridItemTileStyle.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/item/vertical_grid_item/VerticalGridItemTopicFunctionalityType.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/media/AspectRatio.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/media/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/media/Media.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/media/MediaEntity.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/media/MediaKey.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/media/Rect.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/metadata/ArticleDetails.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/metadata/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/metadata/Badge.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/metadata/Callback.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/metadata/ClientEventInfo.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/metadata/CommerceDetails.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/metadata/ConfirmationDisplayType.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/metadata/ConversationDetails.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/metadata/ConversationSection.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/metadata/DismissInfo.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/metadata/FeedbackAction.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/metadata/FeedbackActionInfo.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/metadata/FeedbackInfo.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/metadata/FeedbackType.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/metadata/ImageAnimationType.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/metadata/ImageDisplayType.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/metadata/ImageVariant.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/metadata/LiveEventDetails.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/metadata/MarkUnreadableEntry.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/metadata/PinnableEntry.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/metadata/ReplaceableEntry.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/metadata/ReplyPinState.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/metadata/RichFeedbackBehavior.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/metadata/SocialContext.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/metadata/TimelinesDetails.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/metadata/Url.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/operation/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/operation/CursorDisplayTreatment.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/operation/CursorItem.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/operation/CursorOperation.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/operation/CursorType.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/promoted/AdMetadataContainer.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/promoted/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/promoted/CallToAction.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/promoted/ClickTrackingInfo.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/promoted/DisclaimerType.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/promoted/DisclosureType.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/promoted/DynamicPrerollType.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/promoted/MediaInfo.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/promoted/Preroll.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/promoted/PrerollMetadata.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/promoted/PromotedMetadata.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/promoted/SkAdNetworkData.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/promoted/SponsorshipType.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/promoted/UrlOverrideType.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/promoted/VideoVariant.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/reaction/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/reaction/TimelineReaction.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/reaction/TimelineReactionExecution.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/richtext/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/richtext/ReferenceObject.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/richtext/RichText.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/richtext/RichTextAlignment.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/richtext/RichTextEntity.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/richtext/RichTextFormat.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/timeline_module/AdsMetadata.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/timeline_module/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/timeline_module/GridCarouselMetadata.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/timeline_module/ModuleConversationMetadata.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/timeline_module/ModuleDisplayType.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/timeline_module/ModuleFooter.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/timeline_module/ModuleHeader.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/timeline_module/ModuleHeaderDisplayType.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/timeline_module/ModuleMetadata.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/timeline_module/ModuleShowMoreBehavior.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/module/ABDeciderModule.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/module/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/module/ConfigApiModule.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/module/FeatureSwitchesModule.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/module/LoggingThrowableExceptionMapper.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/module/PipelineExecutionLoggerModule.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/module/ProductMixerModule.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/module/StratoClientModule.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/module/product_mixer_flags/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/module/product_mixer_flags/ProductMixerFlagModule.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/module/stringcenter/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/module/stringcenter/ProductScopeStringCenterModule.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/CandidatePipelineFeatures.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/FailOpenPolicy.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/InvalidStepStateException.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/NewPipelineArrowBuilder.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/NewPipelineBuilder.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/NewPipelineResult.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/NewStepData.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/Pipeline.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/PipelineBuilder.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/PipelineConfig.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/PipelineCursor.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/PipelineCursorSerializer.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/PipelineQuery.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/PipelineResult.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/candidate/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/candidate/CandidatePipeline.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/candidate/CandidatePipelineBuilder.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/candidate/CandidatePipelineBuilderFactory.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/candidate/CandidatePipelineConfig.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/candidate/CandidatePipelineResult.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/candidate/PassthroughCandidatePipelineConfig.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/candidate/StaticCandidatePipelineConfig.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/mixer/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/mixer/MixerPipeline.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/mixer/MixerPipelineBuilder.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/mixer/MixerPipelineBuilderFactory.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/mixer/MixerPipelineConfig.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/mixer/MixerPipelineResult.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/pipeline_failure/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/pipeline_failure/PipelineFailure.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/pipeline_failure/PipelineFailureCategory.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/pipeline_failure/PipelineFailureClassifier.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/pipeline_failure/PipelineFailureSerializer.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/product/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/product/ProductPipeline.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/product/ProductPipelineBuilder.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/product/ProductPipelineBuilderFactory.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/product/ProductPipelineConfig.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/product/ProductPipelineRequest.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/product/ProductPipelineResult.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/recommendation/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/recommendation/RecommendationPipeline.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/recommendation/RecommendationPipelineBuilder.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/recommendation/RecommendationPipelineBuilderFactory.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/recommendation/RecommendationPipelineConfig.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/recommendation/RecommendationPipelineResult.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/scoring/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/scoring/NewScoringPipelineBuilder.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/scoring/ScoringPipeline.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/scoring/ScoringPipelineBuilder.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/scoring/ScoringPipelineBuilderFactory.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/scoring/ScoringPipelineConfig.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/scoring/ScoringPipelineResult.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/state/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/state/HasAsyncFeatureMap.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/state/HasCandidates.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/state/HasCandidatesWithDetails.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/state/HasCandidatesWithFeatures.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/state/HasExecutorResults.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/state/HasParams.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/state/HasQuery.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/state/HasRequest.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/state/HasResult.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/step/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/step/Step.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/step/async_feature_map/AsyncFeatureMapStep.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/step/async_feature_map/BUILD.bazel create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/step/candidate_feature_hydrator/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/step/candidate_feature_hydrator/CandidateFeatureHydratorStep.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/step/candidate_source/BUILD.bazel create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/step/candidate_source/CandidateSourceStep.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/step/decorator/BUILD.bazel create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/step/decorator/DecoratorStep.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/step/domain_marshaller/BUILD.bazel create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/step/domain_marshaller/DomainMarshallerStep.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/step/filter/BUILD.bazel create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/step/filter/FilterStep.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/step/gate/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/step/gate/GateStep.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/step/group_results/BUILD.bazel create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/step/group_results/GroupResultsStep.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/step/pipeline_executor/BUILD.bazel create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/step/pipeline_executor/PipelineExecutorStep.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/step/pipeline_selector/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/step/pipeline_selector/PipelineSelectorStep.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/step/quality_factor/BUILD.bazel create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/step/quality_factor/QualityFactorStep.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/step/query_feature_hydrator/BUILD.bazel create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/step/query_feature_hydrator/QueryFeatureHydratorStep.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/step/query_transformer/BUILD.bazel create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/step/query_transformer/QueryTransformerStep.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/step/scorer/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/step/scorer/ScorerStep.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/step/selector/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/step/selector/SelectorStep.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/step/side_effect/BUILD.bazel create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/step/side_effect/SideEffectStep.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/step/transport_marshaller/BUILD.bazel create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/step/transport_marshaller/TransportMarshallerStep.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/product/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/product/ProductParamConfig.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/product/ProductParamConfigBuilder.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/product/guice/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/product/guice/ProductScope.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/product/guice/ProductScopeModule.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/product/guice/SimpleScope.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/product/registry/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/product/registry/ProductParamRegistry.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/product/registry/ProductPipelineRegistry.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/product/registry/ProductPipelineRegistryConfig.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/quality_factor/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/quality_factor/Bounds.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/quality_factor/LinearLatencyQualityFactor.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/quality_factor/LinearLatencyQualityFactorObserver.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/quality_factor/QualityFactor.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/quality_factor/QualityFactorConfig.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/quality_factor/QualityFactorObserver.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/quality_factor/QualityFactorStatus.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/quality_factor/QueriesPerSecondBasedQualityFactor.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/quality_factor/QueriesPerSecondBasedQualityFactorObserver.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/quality_factor/QueryRateCounter.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/Executor.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/ExecutorObserver.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/ExecutorResult.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/async_feature_map_executor/AsyncFeatureMapExecutor.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/async_feature_map_executor/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/candidate_decorator_executor/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/candidate_decorator_executor/CandidateDecoratorExecutor.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/candidate_decorator_executor/CandidateDecoratorExecutorResult.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/candidate_feature_hydrator_executor/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/candidate_feature_hydrator_executor/CandidateFeatureHydratorExecutor.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/candidate_feature_hydrator_executor/CandidateFeatureHydratorExecutorResult.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/candidate_feature_transformer_executor/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/candidate_feature_transformer_executor/CandidateFeatureTransformerExecutor.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/candidate_feature_transformer_executor/CandidateFeatureTransformerExecutorResult.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/candidate_pipeline_executor/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/candidate_pipeline_executor/CandidatePipelineExecutor.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/candidate_pipeline_executor/CandidatePipelineExecutorResult.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/candidate_source_executor/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/candidate_source_executor/CandidateSourceExecutor.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/candidate_source_executor/CandidateSourceExecutorResult.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/component_registry/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/component_registry/ComponentRegistry.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/component_registry/RegisteredComponent.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/debug_query/AuthorizationService.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/debug_query/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/debug_query/DebugQueryNotSupportedService.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/debug_query/DebugQueryService.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/debug_query/ParamsSerializerModule.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/domain_marshaller_executor/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/domain_marshaller_executor/DomainMarshallerExecutor.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/feature_hydrator_observer/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/feature_hydrator_observer/FeatureHydratorObserver.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/filter_executor/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/filter_executor/FilterExecutor.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/filter_executor/FilterExecutorResult.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/gate_executor/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/gate_executor/ExecutedGateResult.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/gate_executor/GateExecutor.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/gate_executor/GateExecutorResult.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/gate_executor/StoppedGateException.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/group_results_executor/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/group_results_executor/GroupResultsExecutor.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/pipeline_execution_logger/AllowListedPipelineExecutionLogger.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/pipeline_execution_logger/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/pipeline_execution_logger/PipelineExecutionLogger.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/pipeline_executor/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/pipeline_executor/PipelineExecutor.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/pipeline_executor/PipelineExecutorResult.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/pipeline_result_side_effect_executor/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/pipeline_result_side_effect_executor/PipelineResultSideEffectExecutor.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/pipeline_selector_executor/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/pipeline_selector_executor/PipelineSelectorExecutor.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/pipeline_selector_executor/PipelineSelectorExecutorResult.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/quality_factor_executor/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/quality_factor_executor/QualityFactorExecutorResult.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/query_feature_hydrator_executor/AsyncIndividualFeatureHydratorResultSerializer.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/query_feature_hydrator_executor/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/query_feature_hydrator_executor/QueryFeatureHydratorExecutor.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/scoring_pipeline_executor/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/scoring_pipeline_executor/ScoringPipelineExecutor.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/scoring_pipeline_executor/ScoringPipelineExecutorResult.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/selector_executor/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/selector_executor/SelectorExecutor.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/selector_executor/SelectorExecutorResult.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/slice/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/slice/SliceService.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/transformer_executor/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/transformer_executor/PerCandidateTransformerExecutor.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/transformer_executor/TransformerExecutor.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/transport_marshaller_executor/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/transport_marshaller_executor/TransportMarshallerExecutor.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/urp/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/urp/UrpService.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/urt/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/urt/UrtService.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/util/BUILD create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/util/FuturePools.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/util/OffloadFuturePools.scala create mode 100644 product-mixer/core/src/main/scala/com/twitter/product_mixer/core/util/SortIndexBuilder.scala create mode 100644 product-mixer/shared-library/src/main/scala/com/twitter/product_mixer/shared_library/http_client/BUILD create mode 100644 product-mixer/shared-library/src/main/scala/com/twitter/product_mixer/shared_library/http_client/FinagleHttpClientBuilder.scala create mode 100644 product-mixer/shared-library/src/main/scala/com/twitter/product_mixer/shared_library/http_client/FinagleHttpClientWithProxyBuilder.scala create mode 100644 product-mixer/shared-library/src/main/scala/com/twitter/product_mixer/shared_library/http_client/HttpHostPort.scala create mode 100644 product-mixer/shared-library/src/main/scala/com/twitter/product_mixer/shared_library/manhattan_client/BUILD create mode 100644 product-mixer/shared-library/src/main/scala/com/twitter/product_mixer/shared_library/manhattan_client/ManhattanClientBuilder.scala create mode 100644 product-mixer/shared-library/src/main/scala/com/twitter/product_mixer/shared_library/memcached_client/BUILD create mode 100644 product-mixer/shared-library/src/main/scala/com/twitter/product_mixer/shared_library/memcached_client/MemcachedClientBuilder.scala create mode 100644 product-mixer/shared-library/src/main/scala/com/twitter/product_mixer/shared_library/observer/BUILD create mode 100644 product-mixer/shared-library/src/main/scala/com/twitter/product_mixer/shared_library/observer/Observer.scala create mode 100644 product-mixer/shared-library/src/main/scala/com/twitter/product_mixer/shared_library/observer/ResultsObserver.scala create mode 100644 product-mixer/shared-library/src/main/scala/com/twitter/product_mixer/shared_library/observer/ResultsStatsObserver.scala create mode 100644 product-mixer/shared-library/src/main/scala/com/twitter/product_mixer/shared_library/thrift_client/BUILD create mode 100644 product-mixer/shared-library/src/main/scala/com/twitter/product_mixer/shared_library/thrift_client/FinagleThriftClientBuilder.scala create mode 100644 recos-injector/BUILD.bazel create mode 100644 recos-injector/CONFIG.ini create mode 100644 recos-injector/README.md create mode 100644 recos-injector/server/BUILD create mode 100644 recos-injector/server/config/BUILD create mode 100755 recos-injector/server/config/change_log_config.ini create mode 100644 recos-injector/server/config/decider.yml create mode 100644 recos-injector/server/src/main/scala/com/twitter/recosinjector/BUILD create mode 100644 recos-injector/server/src/main/scala/com/twitter/recosinjector/Main.scala create mode 100644 recos-injector/server/src/main/scala/com/twitter/recosinjector/clients/BUILD create mode 100644 recos-injector/server/src/main/scala/com/twitter/recosinjector/clients/Gizmoduck.scala create mode 100644 recos-injector/server/src/main/scala/com/twitter/recosinjector/clients/RecosHoseEntitiesCache.scala create mode 100644 recos-injector/server/src/main/scala/com/twitter/recosinjector/clients/SocialGraph.scala create mode 100644 recos-injector/server/src/main/scala/com/twitter/recosinjector/clients/Tweetypie.scala create mode 100644 recos-injector/server/src/main/scala/com/twitter/recosinjector/clients/UrlResolver.scala create mode 100644 recos-injector/server/src/main/scala/com/twitter/recosinjector/config/BUILD create mode 100644 recos-injector/server/src/main/scala/com/twitter/recosinjector/config/CacheConfig.scala create mode 100644 recos-injector/server/src/main/scala/com/twitter/recosinjector/config/Config.scala create mode 100644 recos-injector/server/src/main/scala/com/twitter/recosinjector/config/DeployConfig.scala create mode 100644 recos-injector/server/src/main/scala/com/twitter/recosinjector/config/ProdConfig.scala create mode 100644 recos-injector/server/src/main/scala/com/twitter/recosinjector/config/StagingConfig.scala create mode 100644 recos-injector/server/src/main/scala/com/twitter/recosinjector/decider/BUILD create mode 100644 recos-injector/server/src/main/scala/com/twitter/recosinjector/decider/RecosInjectorDecider.scala create mode 100644 recos-injector/server/src/main/scala/com/twitter/recosinjector/edges/BUILD create mode 100644 recos-injector/server/src/main/scala/com/twitter/recosinjector/edges/Edges.scala create mode 100644 recos-injector/server/src/main/scala/com/twitter/recosinjector/edges/EventToMessageBuilder.scala create mode 100644 recos-injector/server/src/main/scala/com/twitter/recosinjector/edges/SocialWriteEventToUserUserGraphBuilder.scala create mode 100644 recos-injector/server/src/main/scala/com/twitter/recosinjector/edges/TimelineEventToUserTweetEntityGraphBuilder.scala create mode 100644 recos-injector/server/src/main/scala/com/twitter/recosinjector/edges/TimelineEventToUserTweetGraphBuilder.scala create mode 100644 recos-injector/server/src/main/scala/com/twitter/recosinjector/edges/TweetEventToUserTweetEntityGraphBuilder.scala create mode 100644 recos-injector/server/src/main/scala/com/twitter/recosinjector/edges/TweetEventToUserTweetGraphBuilder.scala create mode 100644 recos-injector/server/src/main/scala/com/twitter/recosinjector/edges/TweetEventToUserUserGraphBuilder.scala create mode 100644 recos-injector/server/src/main/scala/com/twitter/recosinjector/edges/UnifiedUserActionToUserAdGraphBuilder.scala create mode 100644 recos-injector/server/src/main/scala/com/twitter/recosinjector/edges/UnifiedUserActionToUserTweetGraphPlusBuilder.scala create mode 100644 recos-injector/server/src/main/scala/com/twitter/recosinjector/edges/UnifiedUserActionToUserVideoGraphBuilder.scala create mode 100644 recos-injector/server/src/main/scala/com/twitter/recosinjector/edges/UserTweetEntityEdgeBuilder.scala create mode 100644 recos-injector/server/src/main/scala/com/twitter/recosinjector/event_processors/BUILD create mode 100644 recos-injector/server/src/main/scala/com/twitter/recosinjector/event_processors/EventBusProcessor.scala create mode 100644 recos-injector/server/src/main/scala/com/twitter/recosinjector/event_processors/SocialWriteEventProcessor.scala create mode 100644 recos-injector/server/src/main/scala/com/twitter/recosinjector/event_processors/TimelineEventProcessor.scala create mode 100644 recos-injector/server/src/main/scala/com/twitter/recosinjector/event_processors/TweetEventProcessor.scala create mode 100644 recos-injector/server/src/main/scala/com/twitter/recosinjector/filters/BUILD create mode 100644 recos-injector/server/src/main/scala/com/twitter/recosinjector/filters/NullCastTweetFilter.scala create mode 100644 recos-injector/server/src/main/scala/com/twitter/recosinjector/filters/TweetFilter.scala create mode 100644 recos-injector/server/src/main/scala/com/twitter/recosinjector/filters/UserFilter.scala create mode 100644 recos-injector/server/src/main/scala/com/twitter/recosinjector/publishers/BUILD create mode 100644 recos-injector/server/src/main/scala/com/twitter/recosinjector/publishers/KafkaEventPublisher.scala create mode 100644 recos-injector/server/src/main/scala/com/twitter/recosinjector/util/BUILD create mode 100644 recos-injector/server/src/main/scala/com/twitter/recosinjector/util/EventDetails.scala create mode 100644 recos-injector/server/src/main/scala/com/twitter/recosinjector/uua_processors/BUILD create mode 100644 recos-injector/server/src/main/scala/com/twitter/recosinjector/uua_processors/UnifiedUserActionProcessor.scala create mode 100644 recos-injector/server/src/main/scala/com/twitter/recosinjector/uua_processors/UnifiedUserActionsConsumer.scala create mode 100644 science/search/ingester/config/README.md create mode 100644 science/search/ingester/config/pipeline-indexer.userupdates.xml create mode 100644 science/search/ingester/config/pipeline-ingester.protected.xml create mode 100644 science/search/ingester/config/pipeline-ingester.realtime.xml create mode 100644 science/search/ingester/config/pipeline-ingester.realtime_cg.xml create mode 100644 simclusters-ann/BUILD.bazel create mode 100644 simclusters-ann/README.md create mode 100644 simclusters-ann/server/BUILD create mode 100644 simclusters-ann/server/src/main/resources/BUILD create mode 100644 simclusters-ann/server/src/main/resources/config/decider.yml create mode 100644 simclusters-ann/server/src/main/resources/logback.xml create mode 100644 simclusters-ann/server/src/main/scala/com/twitter/simclustersann/BUILD create mode 100644 simclusters-ann/server/src/main/scala/com/twitter/simclustersann/SimclustersAnnServer.scala create mode 100644 simclusters-ann/server/src/main/scala/com/twitter/simclustersann/SimclustersAnnWarmupHandler.scala create mode 100644 simclusters-ann/server/src/main/scala/com/twitter/simclustersann/candidate_source/ApproximateCosineSimilarity.scala create mode 100644 simclusters-ann/server/src/main/scala/com/twitter/simclustersann/candidate_source/BUILD create mode 100644 simclusters-ann/server/src/main/scala/com/twitter/simclustersann/candidate_source/ExperimentalApproximateCosineSimilarity.scala create mode 100644 simclusters-ann/server/src/main/scala/com/twitter/simclustersann/candidate_source/OptimizedApproximateCosineSimilarity.scala create mode 100644 simclusters-ann/server/src/main/scala/com/twitter/simclustersann/candidate_source/SimClustersANNCandidateSource.scala create mode 100644 simclusters-ann/server/src/main/scala/com/twitter/simclustersann/common/BUILD create mode 100644 simclusters-ann/server/src/main/scala/com/twitter/simclustersann/common/FlagNames.scala create mode 100644 simclusters-ann/server/src/main/scala/com/twitter/simclustersann/controllers/BUILD create mode 100644 simclusters-ann/server/src/main/scala/com/twitter/simclustersann/controllers/SimClustersANNController.scala create mode 100644 simclusters-ann/server/src/main/scala/com/twitter/simclustersann/exceptions/BUILD create mode 100644 simclusters-ann/server/src/main/scala/com/twitter/simclustersann/exceptions/InvalidRequestForSimClustersAnnVariantException.scala create mode 100644 simclusters-ann/server/src/main/scala/com/twitter/simclustersann/exceptions/InvalidRequestForSimClustersAnnVariantExceptionMapper.scala create mode 100644 simclusters-ann/server/src/main/scala/com/twitter/simclustersann/exceptions/MissingClusterConfigForSimClustersAnnVariantException.scala create mode 100644 simclusters-ann/server/src/main/scala/com/twitter/simclustersann/filters/BUILD create mode 100644 simclusters-ann/server/src/main/scala/com/twitter/simclustersann/filters/GetTweetCandidatesResponseStatsFilter.scala create mode 100644 simclusters-ann/server/src/main/scala/com/twitter/simclustersann/filters/SimClustersAnnVariantFilter.scala create mode 100644 simclusters-ann/server/src/main/scala/com/twitter/simclustersann/modules/BUILD create mode 100644 simclusters-ann/server/src/main/scala/com/twitter/simclustersann/modules/CacheModule.scala create mode 100644 simclusters-ann/server/src/main/scala/com/twitter/simclustersann/modules/ClusterConfigMapperModule.scala create mode 100644 simclusters-ann/server/src/main/scala/com/twitter/simclustersann/modules/ClusterConfigModule.scala create mode 100644 simclusters-ann/server/src/main/scala/com/twitter/simclustersann/modules/ClusterTweetIndexProviderModule.scala create mode 100644 simclusters-ann/server/src/main/scala/com/twitter/simclustersann/modules/CustomMtlsThriftWebFormsModule.scala create mode 100644 simclusters-ann/server/src/main/scala/com/twitter/simclustersann/modules/EmbeddingStoreModule.scala create mode 100644 simclusters-ann/server/src/main/scala/com/twitter/simclustersann/modules/FlagsModule.scala create mode 100644 simclusters-ann/server/src/main/scala/com/twitter/simclustersann/modules/FuturePoolProvider.scala create mode 100644 simclusters-ann/server/src/main/scala/com/twitter/simclustersann/modules/RateLimiterModule.scala create mode 100644 simclusters-ann/server/src/main/scala/com/twitter/simclustersann/modules/ServiceNameMapperModule.scala create mode 100644 simclusters-ann/server/src/main/scala/com/twitter/simclustersann/modules/SimClustersANNCandidateSourceModule.scala create mode 100644 simclusters-ann/server/src/main/scala/com/twitter/simclustersann/modules/StratoClientProviderModule.scala create mode 100644 simclusters-ann/thrift/src/main/thrift/BUILD create mode 100644 simclusters-ann/thrift/src/main/thrift/simClustersAnn.thrift create mode 100644 src/java/com/twitter/search/README.md create mode 100644 src/java/com/twitter/search/common/README.md create mode 100644 src/java/com/twitter/search/common/converter/earlybird/BUILD create mode 100644 src/java/com/twitter/search/common/converter/earlybird/BasicIndexingConverter.java create mode 100644 src/java/com/twitter/search/common/converter/earlybird/CombinedIndexingConverter.java create mode 100644 src/java/com/twitter/search/common/converter/earlybird/DelayedIndexingConverter.java create mode 100644 src/java/com/twitter/search/common/converter/earlybird/EncodedFeatureBuilder.java create mode 100644 src/java/com/twitter/search/common/encoding/docvalues/BUILD create mode 100644 src/java/com/twitter/search/common/encoding/docvalues/CSFTypeUtil.java create mode 100644 src/java/com/twitter/search/common/encoding/features/BUILD create mode 100644 src/java/com/twitter/search/common/encoding/features/BinByteNormalizer.java create mode 100644 src/java/com/twitter/search/common/encoding/features/ByteNormalizer.java create mode 100644 src/java/com/twitter/search/common/encoding/features/ClampByteNormalizer.java create mode 100644 src/java/com/twitter/search/common/encoding/features/EncodedFeatures.java create mode 100644 src/java/com/twitter/search/common/encoding/features/IntNormalizer.java create mode 100644 src/java/com/twitter/search/common/encoding/features/IntegerEncodedFeatures.java create mode 100644 src/java/com/twitter/search/common/encoding/features/LogByteNormalizer.java create mode 100644 src/java/com/twitter/search/common/encoding/features/PredictionScoreNormalizer.java create mode 100644 src/java/com/twitter/search/common/encoding/features/SingleBytePositiveFloatNormalizer.java create mode 100644 src/java/com/twitter/search/common/encoding/features/SingleBytePositiveFloatUtil.java create mode 100644 src/java/com/twitter/search/common/encoding/features/SmartIntegerNormalizer.java create mode 100644 src/java/com/twitter/search/common/query/BUILD create mode 100644 src/java/com/twitter/search/common/query/BoostUtils.java create mode 100644 src/java/com/twitter/search/common/query/CollectAnnotationsVisitor.java create mode 100644 src/java/com/twitter/search/common/query/CollectQueryTypeVisitor.java create mode 100644 src/java/com/twitter/search/common/query/CollectVariantVisitor.java create mode 100644 src/java/com/twitter/search/common/query/DefaultFilterWeight.java create mode 100644 src/java/com/twitter/search/common/query/DocIdFilter.java create mode 100644 src/java/com/twitter/search/common/query/FieldRankHitInfo.java create mode 100644 src/java/com/twitter/search/common/query/FieldWeightUtil.java create mode 100644 src/java/com/twitter/search/common/query/FilteredQuery.java create mode 100644 src/java/com/twitter/search/common/query/FilteredScorer.java create mode 100644 src/java/com/twitter/search/common/query/HitAttributeCollector.java create mode 100644 src/java/com/twitter/search/common/query/HitAttributeHelper.java create mode 100644 src/java/com/twitter/search/common/query/HitAttributeProvider.java create mode 100644 src/java/com/twitter/search/common/query/IDDisjunctionQuery.java create mode 100644 src/java/com/twitter/search/common/query/IdentifiableQuery.java create mode 100644 src/java/com/twitter/search/common/query/IdentifiableQueryScorer.java create mode 100644 src/java/com/twitter/search/common/query/IdentifiableQueryWeight.java create mode 100644 src/java/com/twitter/search/common/query/MappableField.java create mode 100644 src/java/com/twitter/search/common/query/MultiTermDisjunctionQuery.java create mode 100644 src/java/com/twitter/search/common/query/QueryCommonFieldHitsVisitor.java create mode 100644 src/java/com/twitter/search/common/query/QueryHitAttributeHelper.java create mode 100644 src/java/com/twitter/search/common/query/QueryRankVisitor.java create mode 100644 src/java/com/twitter/search/common/query/SingleDocDocIdSetIterator.java create mode 100644 src/java/com/twitter/search/common/query/StaticHitAttributeProvider.java create mode 100644 src/java/com/twitter/search/common/relevance/BUILD create mode 100644 src/java/com/twitter/search/common/relevance/NGramCache.java create mode 100644 src/java/com/twitter/search/common/relevance/TrendsThriftDataServiceManager.java create mode 100644 src/java/com/twitter/search/common/relevance/classifiers/TweetClassifier.java create mode 100644 src/java/com/twitter/search/common/relevance/classifiers/TweetEvaluator.java create mode 100644 src/java/com/twitter/search/common/relevance/classifiers/TweetOffensiveEvaluator.java create mode 100644 src/java/com/twitter/search/common/relevance/classifiers/TweetQualityFeatureExtractor.java create mode 100644 src/java/com/twitter/search/common/relevance/classifiers/TweetTextClassifier.java create mode 100644 src/java/com/twitter/search/common/relevance/classifiers/TweetTextEvaluator.java create mode 100644 src/java/com/twitter/search/common/relevance/classifiers/TweetTrendsExtractor.java create mode 100644 src/java/com/twitter/search/common/relevance/config/TweetProcessingConfig.java create mode 100644 src/java/com/twitter/search/common/relevance/entities/GeoObject.java create mode 100644 src/java/com/twitter/search/common/relevance/entities/PotentialLocationObject.java create mode 100644 src/java/com/twitter/search/common/relevance/entities/TwitterMessage.java create mode 100644 src/java/com/twitter/search/common/relevance/entities/TwitterMessageUser.java create mode 100644 src/java/com/twitter/search/common/relevance/entities/TwitterMessageUtil.java create mode 100644 src/java/com/twitter/search/common/relevance/entities/TwitterQuotedMessage.java create mode 100644 src/java/com/twitter/search/common/relevance/entities/TwitterRetweetMessage.java create mode 100644 src/java/com/twitter/search/common/relevance/features/AgeDecay.java create mode 100644 src/java/com/twitter/search/common/relevance/features/BUILD create mode 100644 src/java/com/twitter/search/common/relevance/features/EarlybirdDocumentFeatures.java create mode 100644 src/java/com/twitter/search/common/relevance/features/FeatureSink.java create mode 100644 src/java/com/twitter/search/common/relevance/features/IntNormalizers.java create mode 100644 src/java/com/twitter/search/common/relevance/features/MutableFeatureNormalizers.java create mode 100644 src/java/com/twitter/search/common/relevance/features/QueryFeatureType.java create mode 100644 src/java/com/twitter/search/common/relevance/features/RelevanceSignalConstants.java create mode 100644 src/java/com/twitter/search/common/relevance/features/ScoringUtils.java create mode 100644 src/java/com/twitter/search/common/relevance/features/TermVector.java create mode 100644 src/java/com/twitter/search/common/relevance/features/TweetEngagementFeatures.java create mode 100644 src/java/com/twitter/search/common/relevance/features/TweetFeatureType.java create mode 100644 src/java/com/twitter/search/common/relevance/features/TweetFeatures.java create mode 100644 src/java/com/twitter/search/common/relevance/features/TweetIntegerShingleSignature.java create mode 100644 src/java/com/twitter/search/common/relevance/features/TweetSignatureUtil.java create mode 100644 src/java/com/twitter/search/common/relevance/features/TweetTextFeatures.java create mode 100644 src/java/com/twitter/search/common/relevance/features/TweetTextQuality.java create mode 100644 src/java/com/twitter/search/common/relevance/features/TweetUserFeatures.java create mode 100644 src/java/com/twitter/search/common/relevance/scorers/TweetScorer.java create mode 100644 src/java/com/twitter/search/common/relevance/scorers/TweetTextScorer.java create mode 100644 src/java/com/twitter/search/common/relevance/text/LocationUtils.java create mode 100644 src/java/com/twitter/search/common/relevance/text/TweetParser.java create mode 100644 src/java/com/twitter/search/common/relevance/text/VisibleTokenRatioNormalizer.java create mode 100644 src/java/com/twitter/search/common/schema/AnalyzerFactory.java create mode 100644 src/java/com/twitter/search/common/schema/BUILD create mode 100644 src/java/com/twitter/search/common/schema/DynamicSchema.java create mode 100644 src/java/com/twitter/search/common/schema/ImmutableSchema.java create mode 100644 src/java/com/twitter/search/common/schema/NumericField.java create mode 100644 src/java/com/twitter/search/common/schema/SchemaBuilder.java create mode 100644 src/java/com/twitter/search/common/schema/SchemaDocumentFactory.java create mode 100644 src/java/com/twitter/search/common/schema/SchemaUtil.java create mode 100644 src/java/com/twitter/search/common/schema/SearchWhitespaceAnalyzer.java create mode 100644 src/java/com/twitter/search/common/schema/ThriftDocumentBuilder.java create mode 100644 src/java/com/twitter/search/common/schema/base/BUILD create mode 100644 src/java/com/twitter/search/common/schema/base/EarlybirdFieldType.java create mode 100644 src/java/com/twitter/search/common/schema/base/FeatureConfiguration.java create mode 100644 src/java/com/twitter/search/common/schema/base/FieldNameToIdMapping.java create mode 100644 src/java/com/twitter/search/common/schema/base/FieldWeightDefault.java create mode 100644 src/java/com/twitter/search/common/schema/base/ImmutableSchemaInterface.java create mode 100644 src/java/com/twitter/search/common/schema/base/IndexedNumericFieldSettings.java create mode 100644 src/java/com/twitter/search/common/schema/base/Schema.java create mode 100644 src/java/com/twitter/search/common/schema/base/ThriftDocumentUtil.java create mode 100644 src/java/com/twitter/search/common/schema/earlybird/BUILD create mode 100644 src/java/com/twitter/search/common/schema/earlybird/EarlybirdCluster.java create mode 100644 src/java/com/twitter/search/common/schema/earlybird/EarlybirdEncodedFeatures.java create mode 100644 src/java/com/twitter/search/common/schema/earlybird/EarlybirdEncodedFeaturesUtil.java create mode 100644 src/java/com/twitter/search/common/schema/earlybird/EarlybirdFieldConstants.java create mode 100644 src/java/com/twitter/search/common/schema/earlybird/EarlybirdSchemaBuilder.java create mode 100644 src/java/com/twitter/search/common/schema/earlybird/EarlybirdSchemaCreateTool.java create mode 100644 src/java/com/twitter/search/common/schema/earlybird/EarlybirdThriftDocumentBuilder.java create mode 100644 src/java/com/twitter/search/common/schema/earlybird/EarlybirdThriftDocumentUtil.java create mode 100644 src/java/com/twitter/search/common/schema/earlybird/FlushVersion.java create mode 100644 src/java/com/twitter/search/common/search/AndNotDocIdSetIterator.java create mode 100644 src/java/com/twitter/search/common/search/BUILD create mode 100644 src/java/com/twitter/search/common/search/DelegatingEarlyTerminationCollector.java create mode 100644 src/java/com/twitter/search/common/search/DocIdTracker.java create mode 100644 src/java/com/twitter/search/common/search/EarlyTerminationState.java create mode 100644 src/java/com/twitter/search/common/search/GeoQuadTreeQueryBuilderUtil.java create mode 100644 src/java/com/twitter/search/common/search/IntArrayDocIdSetIterator.java create mode 100644 src/java/com/twitter/search/common/search/PairDocIdSetIterator.java create mode 100644 src/java/com/twitter/search/common/search/QueryCostProvider.java create mode 100644 src/java/com/twitter/search/common/search/TerminationTracker.java create mode 100644 src/java/com/twitter/search/common/search/TwitterCollector.java create mode 100644 src/java/com/twitter/search/common/search/TwitterEarlyTerminationCollector.java create mode 100644 src/java/com/twitter/search/common/search/TwitterIndexSearcher.java create mode 100644 src/java/com/twitter/search/common/search/termination/BUILD create mode 100644 src/java/com/twitter/search/common/search/termination/QueryTimeout.java create mode 100644 src/java/com/twitter/search/common/search/termination/QueryTimeoutFactory.java create mode 100644 src/java/com/twitter/search/common/search/termination/QueryTimeoutImpl.java create mode 100644 src/java/com/twitter/search/common/search/termination/TerminationQuery.java create mode 100644 src/java/com/twitter/search/common/search/termination/TerminationQueryScorer.java create mode 100644 src/java/com/twitter/search/common/search/termination/TerminationQueryWeight.java create mode 100644 src/java/com/twitter/search/common/util/earlybird/BUILD create mode 100644 src/java/com/twitter/search/common/util/earlybird/EarlybirdResponseMergeUtil.java create mode 100644 src/java/com/twitter/search/common/util/earlybird/EarlybirdResponseUtil.java create mode 100644 src/java/com/twitter/search/common/util/earlybird/FacetsResultsUtils.java create mode 100644 src/java/com/twitter/search/common/util/earlybird/ResponseMergerUtils.java create mode 100644 src/java/com/twitter/search/common/util/earlybird/ResultsUtil.java create mode 100644 src/java/com/twitter/search/common/util/earlybird/TermStatisticsUtil.java create mode 100644 src/java/com/twitter/search/common/util/earlybird/ThriftSearchQueryUtil.java create mode 100644 src/java/com/twitter/search/common/util/earlybird/ThriftSearchResultUtil.java create mode 100644 src/java/com/twitter/search/common/util/earlybird/ThriftSearchResultsRelevanceStatsUtil.java create mode 100644 src/java/com/twitter/search/common/util/lang/BUILD create mode 100644 src/java/com/twitter/search/common/util/lang/ThriftLanguageUtil.java create mode 100644 src/java/com/twitter/search/common/util/ml/BUILD create mode 100644 src/java/com/twitter/search/common/util/ml/EnumBasedLinearModel.java create mode 100644 src/java/com/twitter/search/common/util/ml/FeatureUtils.java create mode 100644 src/java/com/twitter/search/common/util/ml/MapBasedLinearModel.java create mode 100644 src/java/com/twitter/search/common/util/ml/StringMapBasedLinearModel.java create mode 100644 src/java/com/twitter/search/common/util/ml/models_manager/BUILD create mode 100644 src/java/com/twitter/search/common/util/ml/models_manager/BaseModelsManager.java create mode 100644 src/java/com/twitter/search/common/util/ml/prediction_engine/BUILD create mode 100644 src/java/com/twitter/search/common/util/ml/prediction_engine/BaseLegacyScoreAccumulator.java create mode 100644 src/java/com/twitter/search/common/util/ml/prediction_engine/BaseModelBuilder.java create mode 100644 src/java/com/twitter/search/common/util/ml/prediction_engine/BaseScoreAccumulator.java create mode 100644 src/java/com/twitter/search/common/util/ml/prediction_engine/CompositeFeatureContext.java create mode 100644 src/java/com/twitter/search/common/util/ml/prediction_engine/DecisionForestModelsManager.java create mode 100644 src/java/com/twitter/search/common/util/ml/prediction_engine/DiscretizedFeature.java create mode 100644 src/java/com/twitter/search/common/util/ml/prediction_engine/DiscretizedFeatureRange.java create mode 100644 src/java/com/twitter/search/common/util/ml/prediction_engine/LegacyModelBuilder.java create mode 100644 src/java/com/twitter/search/common/util/ml/prediction_engine/LightweightLinearModel.java create mode 100644 src/java/com/twitter/search/common/util/ml/prediction_engine/ModelBuilder.java create mode 100644 src/java/com/twitter/search/common/util/ml/prediction_engine/ModelLoader.java create mode 100644 src/java/com/twitter/search/common/util/ml/prediction_engine/PredictionEngineModelsManager.java create mode 100644 src/java/com/twitter/search/common/util/ml/prediction_engine/SchemaBasedModelBuilder.java create mode 100644 src/java/com/twitter/search/common/util/ml/prediction_engine/SchemaBasedScoreAccumulator.java create mode 100644 src/java/com/twitter/search/common/util/ml/tensorflow_engine/BUILD create mode 100644 src/java/com/twitter/search/common/util/ml/tensorflow_engine/TensorflowModelsManager.java create mode 100644 src/java/com/twitter/search/core/earlybird/BUILD create mode 100644 src/java/com/twitter/search/core/earlybird/README.md create mode 100644 src/java/com/twitter/search/core/earlybird/facets/AbstractFacetCountingArray.java create mode 100644 src/java/com/twitter/search/core/earlybird/facets/CSFFacetCountIterator.java create mode 100644 src/java/com/twitter/search/core/earlybird/facets/CompositeFacetCountIterator.java create mode 100644 src/java/com/twitter/search/core/earlybird/facets/DummyFacetAccumulator.java create mode 100644 src/java/com/twitter/search/core/earlybird/facets/EarlybirdFacetDocValueSet.java create mode 100644 src/java/com/twitter/search/core/earlybird/facets/EarlybirdFacets.java create mode 100644 src/java/com/twitter/search/core/earlybird/facets/EarlybirdFacetsFactory.java create mode 100644 src/java/com/twitter/search/core/earlybird/facets/FacetAccumulator.java create mode 100644 src/java/com/twitter/search/core/earlybird/facets/FacetCountAggregator.java create mode 100644 src/java/com/twitter/search/core/earlybird/facets/FacetCountIterator.java create mode 100644 src/java/com/twitter/search/core/earlybird/facets/FacetCountIteratorFactory.java create mode 100644 src/java/com/twitter/search/core/earlybird/facets/FacetCountState.java create mode 100644 src/java/com/twitter/search/core/earlybird/facets/FacetCountingArray.java create mode 100644 src/java/com/twitter/search/core/earlybird/facets/FacetCountingArrayWriter.java create mode 100644 src/java/com/twitter/search/core/earlybird/facets/FacetIDMap.java create mode 100644 src/java/com/twitter/search/core/earlybird/facets/FacetLabelProvider.java create mode 100644 src/java/com/twitter/search/core/earlybird/facets/FacetResponseRewriter.java create mode 100644 src/java/com/twitter/search/core/earlybird/facets/FacetTermCollector.java create mode 100644 src/java/com/twitter/search/core/earlybird/facets/FacetUtil.java create mode 100644 src/java/com/twitter/search/core/earlybird/facets/LanguageHistogram.java create mode 100644 src/java/com/twitter/search/core/earlybird/facets/OptimizedFacetCountingArray.java create mode 100644 src/java/com/twitter/search/core/earlybird/facets/PerfieldFacetCountAggregator.java create mode 100644 src/java/com/twitter/search/core/earlybird/facets/SortedSetDocValuesFacetsFactory.java create mode 100644 src/java/com/twitter/search/core/earlybird/facets/SortedSetDocValuesReaderStateHelper.java create mode 100644 src/java/com/twitter/search/core/earlybird/index/DocIDToTweetIDMapper.java create mode 100644 src/java/com/twitter/search/core/earlybird/index/EarlybirdIndexSegmentAtomicReader.java create mode 100644 src/java/com/twitter/search/core/earlybird/index/EarlybirdIndexSegmentData.java create mode 100644 src/java/com/twitter/search/core/earlybird/index/EarlybirdIndexSegmentWriter.java create mode 100644 src/java/com/twitter/search/core/earlybird/index/EarlybirdIndexableField.java create mode 100644 src/java/com/twitter/search/core/earlybird/index/EarlybirdLuceneIndexSegmentAtomicReader.java create mode 100644 src/java/com/twitter/search/core/earlybird/index/EarlybirdLuceneIndexSegmentData.java create mode 100644 src/java/com/twitter/search/core/earlybird/index/EarlybirdLuceneIndexSegmentWriter.java create mode 100644 src/java/com/twitter/search/core/earlybird/index/EarlybirdRealtimeIndexSegmentAtomicReader.java create mode 100644 src/java/com/twitter/search/core/earlybird/index/EarlybirdRealtimeIndexSegmentData.java create mode 100644 src/java/com/twitter/search/core/earlybird/index/EarlybirdRealtimeIndexSegmentWriter.java create mode 100644 src/java/com/twitter/search/core/earlybird/index/QueryCacheResultForSegment.java create mode 100644 src/java/com/twitter/search/core/earlybird/index/SequentialDocIDMapper.java create mode 100644 src/java/com/twitter/search/core/earlybird/index/TimeMapper.java create mode 100644 src/java/com/twitter/search/core/earlybird/index/column/AbstractColumnStrideMultiIntIndex.java create mode 100644 src/java/com/twitter/search/core/earlybird/index/column/ColumnStrideByteIndex.java create mode 100644 src/java/com/twitter/search/core/earlybird/index/column/ColumnStrideFieldDocValues.java create mode 100644 src/java/com/twitter/search/core/earlybird/index/column/ColumnStrideFieldIndex.java create mode 100644 src/java/com/twitter/search/core/earlybird/index/column/ColumnStrideIntIndex.java create mode 100644 src/java/com/twitter/search/core/earlybird/index/column/ColumnStrideIntViewIndex.java create mode 100644 src/java/com/twitter/search/core/earlybird/index/column/ColumnStrideLongIndex.java create mode 100644 src/java/com/twitter/search/core/earlybird/index/column/ColumnStrideMultiIntIndex.java create mode 100644 src/java/com/twitter/search/core/earlybird/index/column/ConstantColumnStrideFieldIndex.java create mode 100644 src/java/com/twitter/search/core/earlybird/index/column/DocValuesManager.java create mode 100644 src/java/com/twitter/search/core/earlybird/index/column/DocValuesUpdate.java create mode 100644 src/java/com/twitter/search/core/earlybird/index/column/OptimizedColumnStrideByteIndex.java create mode 100644 src/java/com/twitter/search/core/earlybird/index/column/OptimizedColumnStrideIntIndex.java create mode 100644 src/java/com/twitter/search/core/earlybird/index/column/OptimizedColumnStrideLongIndex.java create mode 100644 src/java/com/twitter/search/core/earlybird/index/column/OptimizedColumnStrideMultiIntIndex.java create mode 100644 src/java/com/twitter/search/core/earlybird/index/column/OptimizedDocValuesManager.java create mode 100644 src/java/com/twitter/search/core/earlybird/index/column/UnoptimizedDocValuesManager.java create mode 100644 src/java/com/twitter/search/core/earlybird/index/extensions/EarlybirdIndexExtensionsData.java create mode 100644 src/java/com/twitter/search/core/earlybird/index/extensions/EarlybirdIndexExtensionsFactory.java create mode 100644 src/java/com/twitter/search/core/earlybird/index/extensions/EarlybirdRealtimeIndexExtensionsData.java create mode 100644 src/java/com/twitter/search/core/earlybird/index/inverted/BaseByteBlockPool.java create mode 100644 src/java/com/twitter/search/core/earlybird/index/inverted/ByteBlockPool.java create mode 100644 src/java/com/twitter/search/core/earlybird/index/inverted/ByteTermUtils.java create mode 100644 src/java/com/twitter/search/core/earlybird/index/inverted/DeletedDocs.java create mode 100644 src/java/com/twitter/search/core/earlybird/index/inverted/EarlybirdCSFDocValuesProcessor.java create mode 100644 src/java/com/twitter/search/core/earlybird/index/inverted/EarlybirdOptimizedPostingsEnum.java create mode 100644 src/java/com/twitter/search/core/earlybird/index/inverted/EarlybirdPostingsEnum.java create mode 100644 src/java/com/twitter/search/core/earlybird/index/inverted/FSTTermDictionary.java create mode 100644 src/java/com/twitter/search/core/earlybird/index/inverted/HighDFPackedIntsDocsAndPositionsEnum.java create mode 100644 src/java/com/twitter/search/core/earlybird/index/inverted/HighDFPackedIntsDocsEnum.java create mode 100644 src/java/com/twitter/search/core/earlybird/index/inverted/HighDFPackedIntsPostingLists.java create mode 100644 src/java/com/twitter/search/core/earlybird/index/inverted/HighDFPackedIntsSkipListReader.java create mode 100644 src/java/com/twitter/search/core/earlybird/index/inverted/InMemoryFields.java create mode 100644 src/java/com/twitter/search/core/earlybird/index/inverted/IndexOptimizer.java create mode 100644 src/java/com/twitter/search/core/earlybird/index/inverted/IntBlockPool.java create mode 100644 src/java/com/twitter/search/core/earlybird/index/inverted/IntBlockPoolPackedLongsReader.java create mode 100644 src/java/com/twitter/search/core/earlybird/index/inverted/IntBlockPoolPackedLongsWriter.java create mode 100644 src/java/com/twitter/search/core/earlybird/index/inverted/InvertedIndex.java create mode 100644 src/java/com/twitter/search/core/earlybird/index/inverted/InvertedRealtimeIndex.java create mode 100644 src/java/com/twitter/search/core/earlybird/index/inverted/InvertedRealtimeIndexWriter.java create mode 100644 src/java/com/twitter/search/core/earlybird/index/inverted/LowDFPackedIntsPostingLists.java create mode 100644 src/java/com/twitter/search/core/earlybird/index/inverted/LowDFPackedIntsPostingsEnum.java create mode 100644 src/java/com/twitter/search/core/earlybird/index/inverted/MPHTermDictionary.java create mode 100644 src/java/com/twitter/search/core/earlybird/index/inverted/MultiPostingLists.java create mode 100644 src/java/com/twitter/search/core/earlybird/index/inverted/MultiSegmentTermDictionary.java create mode 100644 src/java/com/twitter/search/core/earlybird/index/inverted/MultiSegmentTermDictionaryWithFastutil.java create mode 100644 src/java/com/twitter/search/core/earlybird/index/inverted/MultiSegmentTermDictionaryWithMap.java create mode 100644 src/java/com/twitter/search/core/earlybird/index/inverted/OptimizedIndexTerms.java create mode 100644 src/java/com/twitter/search/core/earlybird/index/inverted/OptimizedMemoryIndex.java create mode 100644 src/java/com/twitter/search/core/earlybird/index/inverted/OptimizedPostingLists.java create mode 100644 src/java/com/twitter/search/core/earlybird/index/inverted/OptimizingPostingsEnumWrapper.java create mode 100644 src/java/com/twitter/search/core/earlybird/index/inverted/PackedLongsReaderPreComputedValues.java create mode 100644 src/java/com/twitter/search/core/earlybird/index/inverted/PayloadUtil.java create mode 100644 src/java/com/twitter/search/core/earlybird/index/inverted/PostingsBufferQueue.java create mode 100644 src/java/com/twitter/search/core/earlybird/index/inverted/QueryCostTracker.java create mode 100644 src/java/com/twitter/search/core/earlybird/index/inverted/RealtimeIndexTerms.java create mode 100644 src/java/com/twitter/search/core/earlybird/index/inverted/SkipListComparator.java create mode 100644 src/java/com/twitter/search/core/earlybird/index/inverted/SkipListContainer.java create mode 100644 src/java/com/twitter/search/core/earlybird/index/inverted/SkipListIntegerComparator.java create mode 100644 src/java/com/twitter/search/core/earlybird/index/inverted/SkipListPostingList.java create mode 100644 src/java/com/twitter/search/core/earlybird/index/inverted/SkipListPostingsEnum.java create mode 100644 src/java/com/twitter/search/core/earlybird/index/inverted/SkipListSearchFinger.java create mode 100644 src/java/com/twitter/search/core/earlybird/index/inverted/TermDictionary.java create mode 100644 src/java/com/twitter/search/core/earlybird/index/inverted/TermPointerEncoding.java create mode 100644 src/java/com/twitter/search/core/earlybird/index/inverted/TermsArray.java create mode 100644 src/java/com/twitter/search/core/earlybird/index/util/AllDocsIterator.java create mode 100644 src/java/com/twitter/search/core/earlybird/index/util/RangeDISI.java create mode 100644 src/java/com/twitter/search/core/earlybird/index/util/RangeFilterDISI.java create mode 100644 src/java/com/twitter/search/core/earlybird/index/util/SearchSortUtils.java create mode 100644 src/java/com/twitter/search/earlybird/BUILD create mode 100644 src/java/com/twitter/search/earlybird/CONFIG.ini create mode 100644 src/java/com/twitter/search/earlybird/Earlybird.java create mode 100644 src/java/com/twitter/search/earlybird/EarlybirdCPUQualityFactor.java create mode 100644 src/java/com/twitter/search/earlybird/EarlybirdDarkProxy.java create mode 100644 src/java/com/twitter/search/earlybird/EarlybirdFinagleServerManager.java create mode 100644 src/java/com/twitter/search/earlybird/EarlybirdFuturePoolManager.java create mode 100644 src/java/com/twitter/search/earlybird/EarlybirdIndexConfig.java create mode 100644 src/java/com/twitter/search/earlybird/EarlybirdMain.java create mode 100644 src/java/com/twitter/search/earlybird/EarlybirdProductionFinagleServerManager.java create mode 100644 src/java/com/twitter/search/earlybird/EarlybirdSearcher.java create mode 100644 src/java/com/twitter/search/earlybird/EarlybirdServer.java create mode 100644 src/java/com/twitter/search/earlybird/EarlybirdServerSetManager.java create mode 100644 src/java/com/twitter/search/earlybird/EarlybirdStatus.java create mode 100644 src/java/com/twitter/search/earlybird/EarlybirdWarmUpManager.java create mode 100644 src/java/com/twitter/search/earlybird/QualityFactor.java create mode 100644 src/java/com/twitter/search/earlybird/README.md create mode 100644 src/java/com/twitter/search/earlybird/RealtimeEarlybirdIndexConfig.java create mode 100644 src/java/com/twitter/search/earlybird/RecentTweetRestriction.java create mode 100644 src/java/com/twitter/search/earlybird/ServerSetMember.java create mode 100644 src/java/com/twitter/search/earlybird/UpdateableEarlybirdStateManager.java create mode 100644 src/java/com/twitter/search/earlybird/archive/ArchiveEarlybirdIndexConfig.java create mode 100644 src/java/com/twitter/search/earlybird/archive/ArchiveHDFSUtils.java create mode 100644 src/java/com/twitter/search/earlybird/archive/ArchiveOnDiskEarlybirdIndexConfig.java create mode 100644 src/java/com/twitter/search/earlybird/archive/ArchiveSearchPartitionManager.java create mode 100644 src/java/com/twitter/search/earlybird/archive/ArchiveSegment.java create mode 100644 src/java/com/twitter/search/earlybird/archive/ArchiveSegmentDataProvider.java create mode 100644 src/java/com/twitter/search/earlybird/archive/ArchiveSegmentUpdater.java create mode 100644 src/java/com/twitter/search/earlybird/archive/ArchiveSegmentVerifier.java create mode 100644 src/java/com/twitter/search/earlybird/archive/ArchiveTimeSlicer.java create mode 100644 src/java/com/twitter/search/earlybird/archive/DailyStatusBatch.java create mode 100644 src/java/com/twitter/search/earlybird/archive/DailyStatusBatches.java create mode 100644 src/java/com/twitter/search/earlybird/archive/PartitionedBatch.java create mode 100644 src/java/com/twitter/search/earlybird/archive/segmentbuilder/BUILD.bazel create mode 100644 src/java/com/twitter/search/earlybird/archive/segmentbuilder/BuiltAndFinalizedSegment.java create mode 100644 src/java/com/twitter/search/earlybird/archive/segmentbuilder/NotYetBuiltSegment.java create mode 100644 src/java/com/twitter/search/earlybird/archive/segmentbuilder/RateLimitingSegmentHandler.java create mode 100644 src/java/com/twitter/search/earlybird/archive/segmentbuilder/SegmentBuilder.java create mode 100644 src/java/com/twitter/search/earlybird/archive/segmentbuilder/SegmentBuilderApp.java create mode 100644 src/java/com/twitter/search/earlybird/archive/segmentbuilder/SegmentBuilderCoordinator.java create mode 100644 src/java/com/twitter/search/earlybird/archive/segmentbuilder/SegmentBuilderMain.java create mode 100644 src/java/com/twitter/search/earlybird/archive/segmentbuilder/SegmentBuilderModule.java create mode 100644 src/java/com/twitter/search/earlybird/archive/segmentbuilder/SegmentBuilderSegment.java create mode 100644 src/java/com/twitter/search/earlybird/archive/segmentbuilder/SegmentConfig.java create mode 100644 src/java/com/twitter/search/earlybird/archive/segmentbuilder/SegmentInfoConstructionException.java create mode 100644 src/java/com/twitter/search/earlybird/archive/segmentbuilder/SegmentUpdaterException.java create mode 100644 src/java/com/twitter/search/earlybird/archive/segmentbuilder/SomeoneElseIsBuildingSegment.java create mode 100644 src/java/com/twitter/search/earlybird/common/BUILD create mode 100644 src/java/com/twitter/search/earlybird/common/Base64RequestResponseForLogging.java create mode 100644 src/java/com/twitter/search/earlybird/common/CaughtUpMonitor.java create mode 100644 src/java/com/twitter/search/earlybird/common/ClientIdUtil.java create mode 100644 src/java/com/twitter/search/earlybird/common/EarlybirdRequestLogger.java create mode 100644 src/java/com/twitter/search/earlybird/common/EarlybirdRequestPostLogger.java create mode 100644 src/java/com/twitter/search/earlybird/common/EarlybirdRequestPreLogger.java create mode 100644 src/java/com/twitter/search/earlybird/common/EarlybirdRequestUtil.java create mode 100644 src/java/com/twitter/search/earlybird/common/EarlybirdThriftBackend.java create mode 100644 src/java/com/twitter/search/earlybird/common/NonPagingAssert.java create mode 100644 src/java/com/twitter/search/earlybird/common/RequestResponseForLogging.java create mode 100644 src/java/com/twitter/search/earlybird/common/RequestResponsePair.java create mode 100644 src/java/com/twitter/search/earlybird/common/UnknownClientRequestForLogging.java create mode 100644 src/java/com/twitter/search/earlybird/common/config/BUILD create mode 100644 src/java/com/twitter/search/earlybird/common/config/EarlybirdConfig.java create mode 100644 src/java/com/twitter/search/earlybird/common/config/EarlybirdProperty.java create mode 100644 src/java/com/twitter/search/earlybird/common/userupdates/BUILD create mode 100644 src/java/com/twitter/search/earlybird/common/userupdates/UserScrubGeoMap.java create mode 100644 src/java/com/twitter/search/earlybird/common/userupdates/UserTable.java create mode 100644 src/java/com/twitter/search/earlybird/common/userupdates/UserTableBuilderFromSnapshot.java create mode 100644 src/java/com/twitter/search/earlybird/common/userupdates/UserUpdate.java create mode 100644 src/java/com/twitter/search/earlybird/common/userupdates/UserUpdatesChecker.java create mode 100644 src/java/com/twitter/search/earlybird/config/BUILD create mode 100644 src/java/com/twitter/search/earlybird/config/ServingRange.java create mode 100644 src/java/com/twitter/search/earlybird/config/TierConfig.java create mode 100644 src/java/com/twitter/search/earlybird/config/TierInfo.java create mode 100644 src/java/com/twitter/search/earlybird/config/TierInfoSource.java create mode 100644 src/java/com/twitter/search/earlybird/config/TierInfoUtil.java create mode 100644 src/java/com/twitter/search/earlybird/config/TierInfoWrapper.java create mode 100644 src/java/com/twitter/search/earlybird/config/TierServingBoundaryEndPoint.java create mode 100644 src/java/com/twitter/search/earlybird/document/DeletedStatus.java create mode 100644 src/java/com/twitter/search/earlybird/document/DocumentFactory.java create mode 100644 src/java/com/twitter/search/earlybird/document/ThriftDocumentPreprocessor.java create mode 100644 src/java/com/twitter/search/earlybird/document/ThriftIndexingEventDocumentFactory.java create mode 100644 src/java/com/twitter/search/earlybird/document/ThriftIndexingEventUpdateFactory.java create mode 100644 src/java/com/twitter/search/earlybird/document/TimeSlicedThriftIndexingEvent.java create mode 100644 src/java/com/twitter/search/earlybird/document/TruncationTokenStreamWriter.java create mode 100644 src/java/com/twitter/search/earlybird/document/TweetDocument.java create mode 100644 src/java/com/twitter/search/earlybird/exception/AlreadyInServerSetUpdateException.java create mode 100644 src/java/com/twitter/search/earlybird/exception/BadRequestException.java create mode 100644 src/java/com/twitter/search/earlybird/exception/ClientException.java create mode 100644 src/java/com/twitter/search/earlybird/exception/CriticalExceptionHandler.java create mode 100644 src/java/com/twitter/search/earlybird/exception/EarlybirdException.java create mode 100644 src/java/com/twitter/search/earlybird/exception/EarlybirdFinagleServerMonitor.java create mode 100644 src/java/com/twitter/search/earlybird/exception/EarlybirdRuntimeException.java create mode 100644 src/java/com/twitter/search/earlybird/exception/EarlybirdStartupException.java create mode 100644 src/java/com/twitter/search/earlybird/exception/FlushVersionMismatchException.java create mode 100644 src/java/com/twitter/search/earlybird/exception/MissingKafkaTopicException.java create mode 100644 src/java/com/twitter/search/earlybird/exception/MissingUserException.java create mode 100644 src/java/com/twitter/search/earlybird/exception/NotInServerSetUpdateException.java create mode 100644 src/java/com/twitter/search/earlybird/exception/TransientException.java create mode 100644 src/java/com/twitter/search/earlybird/exception/UncaughtExceptionHandler.java create mode 100644 src/java/com/twitter/search/earlybird/exception/WrappedKafkaApiException.java create mode 100644 src/java/com/twitter/search/earlybird/factory/EarlybirdIndexConfigUtil.java create mode 100644 src/java/com/twitter/search/earlybird/factory/EarlybirdKafkaConsumersFactory.java create mode 100644 src/java/com/twitter/search/earlybird/factory/EarlybirdServerFactory.java create mode 100644 src/java/com/twitter/search/earlybird/factory/EarlybirdWireModule.java create mode 100644 src/java/com/twitter/search/earlybird/factory/PartitionConfigUtil.java create mode 100644 src/java/com/twitter/search/earlybird/factory/ProductionEarlybirdKafkaConsumersFactory.java create mode 100644 src/java/com/twitter/search/earlybird/factory/QueryCacheUpdaterScheduledExecutorService.java create mode 100644 src/java/com/twitter/search/earlybird/index/AbstractInMemoryTimeMapper.java create mode 100644 src/java/com/twitter/search/earlybird/index/DocValuesBasedTimeMapper.java create mode 100644 src/java/com/twitter/search/earlybird/index/DocValuesBasedTweetIDMapper.java create mode 100644 src/java/com/twitter/search/earlybird/index/DocValuesHelper.java create mode 100644 src/java/com/twitter/search/earlybird/index/EarlybirdSegment.java create mode 100644 src/java/com/twitter/search/earlybird/index/EarlybirdSegmentFactory.java create mode 100644 src/java/com/twitter/search/earlybird/index/EarlybirdSingleSegmentSearcher.java create mode 100644 src/java/com/twitter/search/earlybird/index/OptimizedTimeMapper.java create mode 100644 src/java/com/twitter/search/earlybird/index/OptimizedTweetIDMapper.java create mode 100644 src/java/com/twitter/search/earlybird/index/OutOfOrderRealtimeTweetIDMapper.java create mode 100644 src/java/com/twitter/search/earlybird/index/RealtimeTimeMapper.java create mode 100644 src/java/com/twitter/search/earlybird/index/TimeMappingWriter.java create mode 100644 src/java/com/twitter/search/earlybird/index/TweetIDMapper.java create mode 100644 src/java/com/twitter/search/earlybird/index/TweetIDQuery.java create mode 100644 src/java/com/twitter/search/earlybird/index/TweetIDToInternalIDMap.java create mode 100644 src/java/com/twitter/search/earlybird/index/TweetSearchIndexExtensionsFactory.java create mode 100644 src/java/com/twitter/search/earlybird/index/TweetSearchLuceneIndexExtensionsData.java create mode 100644 src/java/com/twitter/search/earlybird/index/TweetSearchRealtimeIndexExtensionsData.java create mode 100644 src/java/com/twitter/search/earlybird/index/facets/BUILD create mode 100644 src/java/com/twitter/search/earlybird/index/facets/FacetSkipList.java create mode 100644 src/java/com/twitter/search/earlybird/ml/ScoringModelsManager.java create mode 100644 src/java/com/twitter/search/earlybird/partition/AudioSpaceEventsStreamIndexer.java create mode 100644 src/java/com/twitter/search/earlybird/partition/AudioSpaceTable.java create mode 100644 src/java/com/twitter/search/earlybird/partition/BalancingKafkaConsumer.java create mode 100644 src/java/com/twitter/search/earlybird/partition/CompleteSegmentManager.java create mode 100644 src/java/com/twitter/search/earlybird/partition/DynamicPartitionConfig.java create mode 100644 src/java/com/twitter/search/earlybird/partition/EarlybirdIndex.java create mode 100644 src/java/com/twitter/search/earlybird/partition/EarlybirdIndexFlusher.java create mode 100644 src/java/com/twitter/search/earlybird/partition/EarlybirdIndexLoader.java create mode 100644 src/java/com/twitter/search/earlybird/partition/EarlybirdKafkaConsumer.java create mode 100644 src/java/com/twitter/search/earlybird/partition/EarlybirdStartup.java create mode 100644 src/java/com/twitter/search/earlybird/partition/FlowControlException.java create mode 100644 src/java/com/twitter/search/earlybird/partition/HdfsUtil.java create mode 100644 src/java/com/twitter/search/earlybird/partition/ISegmentWriter.java create mode 100644 src/java/com/twitter/search/earlybird/partition/IndexingResultCounts.java create mode 100644 src/java/com/twitter/search/earlybird/partition/InstrumentedQueue.java create mode 100644 src/java/com/twitter/search/earlybird/partition/KafkaStartup.java create mode 100644 src/java/com/twitter/search/earlybird/partition/MultiSegmentTermDictionaryManager.java create mode 100644 src/java/com/twitter/search/earlybird/partition/OptimizationAndFlushingCoordinationLock.java create mode 100644 src/java/com/twitter/search/earlybird/partition/OptimizingSegmentWriter.java create mode 100644 src/java/com/twitter/search/earlybird/partition/PartitionConfig.java create mode 100644 src/java/com/twitter/search/earlybird/partition/PartitionConfigLoader.java create mode 100644 src/java/com/twitter/search/earlybird/partition/PartitionConfigLoadingException.java create mode 100644 src/java/com/twitter/search/earlybird/partition/PartitionManager.java create mode 100644 src/java/com/twitter/search/earlybird/partition/PartitionManagerStartup.java create mode 100644 src/java/com/twitter/search/earlybird/partition/PartitionWriter.java create mode 100644 src/java/com/twitter/search/earlybird/partition/SearchIndexingMetricSet.java create mode 100644 src/java/com/twitter/search/earlybird/partition/SegmentHdfsFlusher.java create mode 100644 src/java/com/twitter/search/earlybird/partition/SegmentIndexStats.java create mode 100644 src/java/com/twitter/search/earlybird/partition/SegmentIndexStatsExporter.java create mode 100644 src/java/com/twitter/search/earlybird/partition/SegmentInfo.java create mode 100644 src/java/com/twitter/search/earlybird/partition/SegmentLoader.java create mode 100644 src/java/com/twitter/search/earlybird/partition/SegmentManager.java create mode 100644 src/java/com/twitter/search/earlybird/partition/SegmentOptimizer.java create mode 100644 src/java/com/twitter/search/earlybird/partition/SegmentSyncConfig.java create mode 100644 src/java/com/twitter/search/earlybird/partition/SegmentSyncInfo.java create mode 100644 src/java/com/twitter/search/earlybird/partition/SegmentVulture.java create mode 100644 src/java/com/twitter/search/earlybird/partition/SegmentWarmer.java create mode 100644 src/java/com/twitter/search/earlybird/partition/SegmentWriter.java create mode 100644 src/java/com/twitter/search/earlybird/partition/SimpleSegmentIndexer.java create mode 100644 src/java/com/twitter/search/earlybird/partition/SimpleStreamIndexer.java create mode 100644 src/java/com/twitter/search/earlybird/partition/SimpleUpdateIndexer.java create mode 100644 src/java/com/twitter/search/earlybird/partition/StartupUserEventIndexer.java create mode 100644 src/java/com/twitter/search/earlybird/partition/StatusBatchFlushVersion.java create mode 100644 src/java/com/twitter/search/earlybird/partition/TimeLimitedHadoopExistsCall.java create mode 100644 src/java/com/twitter/search/earlybird/partition/TweetCreateHandler.java create mode 100644 src/java/com/twitter/search/earlybird/partition/TweetUpdateHandler.java create mode 100644 src/java/com/twitter/search/earlybird/partition/UserPartitionUtil.java create mode 100644 src/java/com/twitter/search/earlybird/partition/UserScrubGeoEventStreamIndexer.java create mode 100644 src/java/com/twitter/search/earlybird/partition/UserUpdatesStreamIndexer.java create mode 100644 src/java/com/twitter/search/earlybird/partition/freshstartup/FreshStartupHandler.java create mode 100644 src/java/com/twitter/search/earlybird/partition/freshstartup/KafkaOffsetPair.java create mode 100644 src/java/com/twitter/search/earlybird/partition/freshstartup/PostOptimizationUpdatesIndexer.java create mode 100644 src/java/com/twitter/search/earlybird/partition/freshstartup/PreOptimizationSegmentIndexer.java create mode 100644 src/java/com/twitter/search/earlybird/partition/freshstartup/SegmentBuildInfo.java create mode 100644 src/java/com/twitter/search/earlybird/partition/freshstartup/SegmentTweetsIndexingResult.java create mode 100644 src/java/com/twitter/search/earlybird/partition/freshstartup/SkippedPickedCounter.java create mode 100644 src/java/com/twitter/search/earlybird/querycache/CachedFilterQuery.java create mode 100644 src/java/com/twitter/search/earlybird/querycache/CachedResultDocIdSetIterator.java create mode 100644 src/java/com/twitter/search/earlybird/querycache/QueryCacheConfig.java create mode 100644 src/java/com/twitter/search/earlybird/querycache/QueryCacheConversionRules.java create mode 100644 src/java/com/twitter/search/earlybird/querycache/QueryCacheFilter.java create mode 100644 src/java/com/twitter/search/earlybird/querycache/QueryCacheManager.java create mode 100644 src/java/com/twitter/search/earlybird/querycache/QueryCacheResultCollector.java create mode 100644 src/java/com/twitter/search/earlybird/querycache/QueryCacheUpdateTask.java create mode 100644 src/java/com/twitter/search/earlybird/querycache/QueryCacheUpdater.java create mode 100644 src/java/com/twitter/search/earlybird/queryparser/DetectAntisocialVisitor.java create mode 100644 src/java/com/twitter/search/earlybird/queryparser/DetectFieldAnnotationVisitor.java create mode 100644 src/java/com/twitter/search/earlybird/queryparser/EarlybirdLuceneQueryVisitor.java create mode 100644 src/java/com/twitter/search/earlybird/queryparser/EarlybirdQueryHelper.java create mode 100644 src/java/com/twitter/search/earlybird/queryparser/HighFrequencyTermPairExtractor.java create mode 100644 src/java/com/twitter/search/earlybird/queryparser/HighFrequencyTermPairRewriteVisitor.java create mode 100644 src/java/com/twitter/search/earlybird/queryparser/HighFrequencyTermQueryGroup.java create mode 100644 src/java/com/twitter/search/earlybird/queryparser/LuceneRelevanceQueryVisitor.java create mode 100644 src/java/com/twitter/search/earlybird/queryparser/ProtectedOperatorQueryRewriter.java create mode 100644 src/java/com/twitter/search/earlybird/search/AbstractResultsCollector.java create mode 100644 src/java/com/twitter/search/earlybird/search/AntiGamingFilter.java create mode 100644 src/java/com/twitter/search/earlybird/search/EarlybirdLuceneSearcher.java create mode 100644 src/java/com/twitter/search/earlybird/search/EarlybirdMultiSegmentSearcher.java create mode 100644 src/java/com/twitter/search/earlybird/search/GeoQuadTreeQueryBuilder.java create mode 100644 src/java/com/twitter/search/earlybird/search/Hit.java create mode 100644 src/java/com/twitter/search/earlybird/search/SearchRequestInfo.java create mode 100644 src/java/com/twitter/search/earlybird/search/SearchResultsCollector.java create mode 100644 src/java/com/twitter/search/earlybird/search/SearchResultsInfo.java create mode 100644 src/java/com/twitter/search/earlybird/search/SimpleSearchResults.java create mode 100644 src/java/com/twitter/search/earlybird/search/SocialFilter.java create mode 100644 src/java/com/twitter/search/earlybird/search/SocialSearchResultsCollector.java create mode 100644 src/java/com/twitter/search/earlybird/search/facets/AbstractFacetTermCollector.java create mode 100644 src/java/com/twitter/search/earlybird/search/facets/DefaultFacetScorer.java create mode 100644 src/java/com/twitter/search/earlybird/search/facets/EntityAnnotationCollector.java create mode 100644 src/java/com/twitter/search/earlybird/search/facets/ExpandedUrlCollector.java create mode 100644 src/java/com/twitter/search/earlybird/search/facets/ExplainFacetResultsCollector.java create mode 100644 src/java/com/twitter/search/earlybird/search/facets/FacetLabelCollector.java create mode 100644 src/java/com/twitter/search/earlybird/search/facets/FacetRankingModule.java create mode 100644 src/java/com/twitter/search/earlybird/search/facets/FacetResultsCollector.java create mode 100644 src/java/com/twitter/search/earlybird/search/facets/FacetScorer.java create mode 100644 src/java/com/twitter/search/earlybird/search/facets/FacetSearchRequestInfo.java create mode 100644 src/java/com/twitter/search/earlybird/search/facets/HashingAndPruningFacetAccumulator.java create mode 100644 src/java/com/twitter/search/earlybird/search/facets/NamedEntityCollector.java create mode 100644 src/java/com/twitter/search/earlybird/search/facets/RetweetFacetCountIterator.java create mode 100644 src/java/com/twitter/search/earlybird/search/facets/SimpleCountRankingModule.java create mode 100644 src/java/com/twitter/search/earlybird/search/facets/SpaceFacetCollector.java create mode 100644 src/java/com/twitter/search/earlybird/search/facets/TermStatisticsCollector.java create mode 100644 src/java/com/twitter/search/earlybird/search/facets/TermStatisticsRequestInfo.java create mode 100644 src/java/com/twitter/search/earlybird/search/facets/TweetSearchFacetCountIteratorFactory.java create mode 100644 src/java/com/twitter/search/earlybird/search/queries/BadUserRepFilter.java create mode 100644 src/java/com/twitter/search/earlybird/search/queries/CSFDisjunctionFilter.java create mode 100644 src/java/com/twitter/search/earlybird/search/queries/DocValRangeFilter.java create mode 100644 src/java/com/twitter/search/earlybird/search/queries/FeatureValueInAcceptListOrUnsetFilter.java create mode 100644 src/java/com/twitter/search/earlybird/search/queries/GeoTwoPhaseQuery.java create mode 100644 src/java/com/twitter/search/earlybird/search/queries/MatchAllDocIdSet.java create mode 100644 src/java/com/twitter/search/earlybird/search/queries/MatchAllDocsQuery.java create mode 100644 src/java/com/twitter/search/earlybird/search/queries/RequiredStatusIDsFilter.java create mode 100644 src/java/com/twitter/search/earlybird/search/queries/SimpleTermQuery.java create mode 100644 src/java/com/twitter/search/earlybird/search/queries/SinceMaxIDFilter.java create mode 100644 src/java/com/twitter/search/earlybird/search/queries/SinceUntilFilter.java create mode 100644 src/java/com/twitter/search/earlybird/search/queries/TermQueryWithSafeToString.java create mode 100644 src/java/com/twitter/search/earlybird/search/queries/TimedDocIdSetIterator.java create mode 100644 src/java/com/twitter/search/earlybird/search/queries/UserFlagsExcludeFilter.java create mode 100644 src/java/com/twitter/search/earlybird/search/queries/UserIdMultiSegmentQuery.java create mode 100644 src/java/com/twitter/search/earlybird/search/queries/UserScrubGeoFilter.java create mode 100644 src/java/com/twitter/search/earlybird/search/relevance/LinearScoringData.java create mode 100644 src/java/com/twitter/search/earlybird/search/relevance/LinearScoringParams.java create mode 100644 src/java/com/twitter/search/earlybird/search/relevance/MinFeatureValueFilter.java create mode 100644 src/java/com/twitter/search/earlybird/search/relevance/RelevanceHit.java create mode 100644 src/java/com/twitter/search/earlybird/search/relevance/RelevanceSearchRequestInfo.java create mode 100644 src/java/com/twitter/search/earlybird/search/relevance/RelevanceSearchResults.java create mode 100644 src/java/com/twitter/search/earlybird/search/relevance/ScoreFilterQuery.java create mode 100644 src/java/com/twitter/search/earlybird/search/relevance/collectors/AbstractRelevanceCollector.java create mode 100644 src/java/com/twitter/search/earlybird/search/relevance/collectors/BatchRelevanceTopCollector.java create mode 100644 src/java/com/twitter/search/earlybird/search/relevance/collectors/RelevanceAllCollector.java create mode 100644 src/java/com/twitter/search/earlybird/search/relevance/collectors/RelevanceTopCollector.java create mode 100644 src/java/com/twitter/search/earlybird/search/relevance/scoring/BatchHit.java create mode 100644 src/java/com/twitter/search/earlybird/search/relevance/scoring/DefaultScoringFunction.java create mode 100644 src/java/com/twitter/search/earlybird/search/relevance/scoring/FeatureBasedScoringFunction.java create mode 100644 src/java/com/twitter/search/earlybird/search/relevance/scoring/LegacyScoreAccumulator.java create mode 100644 src/java/com/twitter/search/earlybird/search/relevance/scoring/LinearScoringFunction.java create mode 100644 src/java/com/twitter/search/earlybird/search/relevance/scoring/ModelBasedScoringFunction.java create mode 100644 src/java/com/twitter/search/earlybird/search/relevance/scoring/RelevanceQuery.java create mode 100644 src/java/com/twitter/search/earlybird/search/relevance/scoring/RetweetBasedTopTweetsScoringFunction.java create mode 100644 src/java/com/twitter/search/earlybird/search/relevance/scoring/ScoringFunction.java create mode 100644 src/java/com/twitter/search/earlybird/search/relevance/scoring/ScoringFunctionProvider.java create mode 100644 src/java/com/twitter/search/earlybird/search/relevance/scoring/SpamVectorScoringFunction.java create mode 100644 src/java/com/twitter/search/earlybird/search/relevance/scoring/SparseTensor.java create mode 100644 src/java/com/twitter/search/earlybird/search/relevance/scoring/TensorflowBasedScoringFunction.java create mode 100644 src/java/com/twitter/search/earlybird/search/relevance/scoring/TestScoringFunction.java create mode 100644 src/java/com/twitter/search/earlybird/segment/DLSegmentDataProvider.java create mode 100644 src/java/com/twitter/search/earlybird/segment/DLSegmentDataReaderSet.java create mode 100644 src/java/com/twitter/search/earlybird/segment/EmptySegmentDataReaderSet.java create mode 100644 src/java/com/twitter/search/earlybird/segment/SegmentDataProvider.java create mode 100644 src/java/com/twitter/search/earlybird/segment/SegmentDataReaderSet.java create mode 100644 src/java/com/twitter/search/earlybird/segment/SegmentProvider.java create mode 100644 src/java/com/twitter/search/earlybird/stats/EarlybirdRPCStats.java create mode 100644 src/java/com/twitter/search/earlybird/stats/EarlybirdSearcherStats.java create mode 100644 src/java/com/twitter/search/earlybird/stats/SegmentSyncStats.java create mode 100644 src/java/com/twitter/search/earlybird/tools/EarlybirdThriftRequestDeserializerUtil.java create mode 100644 src/java/com/twitter/search/earlybird/util/ActionLogger.java create mode 100644 src/java/com/twitter/search/earlybird/util/CoordinatedEarlybirdAction.java create mode 100644 src/java/com/twitter/search/earlybird/util/CoordinatedEarlybirdActionInterface.java create mode 100644 src/java/com/twitter/search/earlybird/util/CoordinatedEarlybirdActionLockFailed.java create mode 100644 src/java/com/twitter/search/earlybird/util/EarlybirdDecider.java create mode 100644 src/java/com/twitter/search/earlybird/util/EarlybirdSearchResultUtil.java create mode 100644 src/java/com/twitter/search/earlybird/util/FieldTermCounter.java create mode 100644 src/java/com/twitter/search/earlybird/util/Histogram.java create mode 100644 src/java/com/twitter/search/earlybird/util/IndexViewer.java create mode 100644 src/java/com/twitter/search/earlybird/util/JsonViewerWriter.java create mode 100644 src/java/com/twitter/search/earlybird/util/OneTaskScheduledExecutorManager.java create mode 100644 src/java/com/twitter/search/earlybird/util/ParallelUtil.java create mode 100644 src/java/com/twitter/search/earlybird/util/PeriodicActionParams.java create mode 100644 src/java/com/twitter/search/earlybird/util/ScheduledExecutorManager.java create mode 100644 src/java/com/twitter/search/earlybird/util/ScheduledExecutorTask.java create mode 100644 src/java/com/twitter/search/earlybird/util/ScrubGenUtil.java create mode 100644 src/java/com/twitter/search/earlybird/util/ShutdownWaitTimeParams.java create mode 100644 src/java/com/twitter/search/earlybird/util/TermCountMonitor.java create mode 100644 src/java/com/twitter/search/earlybird/util/TweetCountMonitor.java create mode 100644 src/java/com/twitter/search/earlybird/util/ViewerWriter.java create mode 100644 src/java/com/twitter/search/earlybird_root/BUILD create mode 100644 src/java/com/twitter/search/earlybird_root/ClientBackupFilter.java create mode 100644 src/java/com/twitter/search/earlybird_root/ClientLatencyFilter.java create mode 100644 src/java/com/twitter/search/earlybird_root/EarlybirdCacheCommonModule.java create mode 100644 src/java/com/twitter/search/earlybird_root/EarlybirdChainedScatterGatherService.java create mode 100644 src/java/com/twitter/search/earlybird_root/EarlybirdCommonModule.java create mode 100644 src/java/com/twitter/search/earlybird_root/EarlybirdFullArchiveScatterGatherSupport.java create mode 100644 src/java/com/twitter/search/earlybird_root/EarlybirdProtectedScatterGatherSupport.java create mode 100644 src/java/com/twitter/search/earlybird_root/EarlybirdProtectedValidationBehavior.java create mode 100644 src/java/com/twitter/search/earlybird_root/EarlybirdProtectedWarmup.java create mode 100644 src/java/com/twitter/search/earlybird_root/EarlybirdQueryRewriteFilter.java create mode 100644 src/java/com/twitter/search/earlybird_root/EarlybirdRealtimeCgScatterGatherSupport.java create mode 100644 src/java/com/twitter/search/earlybird_root/EarlybirdRealtimeScatterGatherSupport.java create mode 100644 src/java/com/twitter/search/earlybird_root/EarlybirdRootQueryUtils.java create mode 100644 src/java/com/twitter/search/earlybird_root/EarlybirdServiceChainBuilder.java create mode 100644 src/java/com/twitter/search/earlybird_root/EarlybirdServiceLoggingSupport.java create mode 100644 src/java/com/twitter/search/earlybird_root/EarlybirdServicePartitionLoggingSupport.java create mode 100644 src/java/com/twitter/search/earlybird_root/EarlybirdServiceScatterGatherSupport.java create mode 100644 src/java/com/twitter/search/earlybird_root/EarlybirdServiceValidationBehavior.java create mode 100644 src/java/com/twitter/search/earlybird_root/EarlybirdTierThrottleDeciders.java create mode 100644 src/java/com/twitter/search/earlybird_root/EarlybirdWarmup.java create mode 100644 src/java/com/twitter/search/earlybird_root/ExceptionHandler.java create mode 100644 src/java/com/twitter/search/earlybird_root/FullArchiveRootAppMain.java create mode 100644 src/java/com/twitter/search/earlybird_root/FullArchiveRootModule.java create mode 100644 src/java/com/twitter/search/earlybird_root/FullArchiveRootServer.java create mode 100644 src/java/com/twitter/search/earlybird_root/FullArchiveRootService.java create mode 100644 src/java/com/twitter/search/earlybird_root/InitializeFilter.java create mode 100644 src/java/com/twitter/search/earlybird_root/MultiTierResultsMergeFilter.java create mode 100644 src/java/com/twitter/search/earlybird_root/PartitionAccessController.java create mode 100644 src/java/com/twitter/search/earlybird_root/ProtectedRootAppMain.java create mode 100644 src/java/com/twitter/search/earlybird_root/ProtectedRootAppModule.java create mode 100644 src/java/com/twitter/search/earlybird_root/ProtectedRootServer.java create mode 100644 src/java/com/twitter/search/earlybird_root/ProtectedRootService.java create mode 100644 src/java/com/twitter/search/earlybird_root/ProtectedScatterGatherModule.java create mode 100644 src/java/com/twitter/search/earlybird_root/QuotaModule.java create mode 100644 src/java/com/twitter/search/earlybird_root/README.md create mode 100644 src/java/com/twitter/search/earlybird_root/RealtimeCgRootAppMain.java create mode 100644 src/java/com/twitter/search/earlybird_root/RealtimeCgRootAppModule.java create mode 100644 src/java/com/twitter/search/earlybird_root/RealtimeCgRootServer.java create mode 100644 src/java/com/twitter/search/earlybird_root/RealtimeCgRootService.java create mode 100644 src/java/com/twitter/search/earlybird_root/RealtimeCgScatterGatherModule.java create mode 100644 src/java/com/twitter/search/earlybird_root/RealtimeRootAppMain.java create mode 100644 src/java/com/twitter/search/earlybird_root/RealtimeRootAppModule.java create mode 100644 src/java/com/twitter/search/earlybird_root/RealtimeRootServer.java create mode 100644 src/java/com/twitter/search/earlybird_root/RealtimeRootService.java create mode 100644 src/java/com/twitter/search/earlybird_root/RealtimeScatterGatherModule.java create mode 100644 src/java/com/twitter/search/earlybird_root/RootResponseClassifier.java create mode 100644 src/java/com/twitter/search/earlybird_root/ScatterGatherModule.java create mode 100644 src/java/com/twitter/search/earlybird_root/SkipPartitionFilter.java create mode 100644 src/java/com/twitter/search/earlybird_root/SuperRootAppMain.java create mode 100644 src/java/com/twitter/search/earlybird_root/SuperRootAppModule.java create mode 100644 src/java/com/twitter/search/earlybird_root/SuperRootRequestTypeRouter.java create mode 100644 src/java/com/twitter/search/earlybird_root/SuperRootServer.java create mode 100644 src/java/com/twitter/search/earlybird_root/SuperRootService.java create mode 100644 src/java/com/twitter/search/earlybird_root/caching/BUILD create mode 100644 src/java/com/twitter/search/earlybird_root/caching/CacheCommonUtil.java create mode 100644 src/java/com/twitter/search/earlybird_root/caching/CacheStats.java create mode 100644 src/java/com/twitter/search/earlybird_root/caching/DefaultForcedCacheMissDecider.java create mode 100644 src/java/com/twitter/search/earlybird_root/caching/EarlybirdCachePostProcessor.java create mode 100644 src/java/com/twitter/search/earlybird_root/caching/EarlybirdRequestPerClientCacheStats.java create mode 100644 src/java/com/twitter/search/earlybird_root/caching/FacetsCache.java create mode 100644 src/java/com/twitter/search/earlybird_root/caching/FacetsCacheFilter.java create mode 100644 src/java/com/twitter/search/earlybird_root/caching/FacetsCacheRequestNormalizer.java create mode 100644 src/java/com/twitter/search/earlybird_root/caching/FacetsQueryCachePredicate.java create mode 100644 src/java/com/twitter/search/earlybird_root/caching/FacetsServicePostProcessor.java create mode 100644 src/java/com/twitter/search/earlybird_root/caching/RecencyAndRelevanceCachePostProcessor.java create mode 100644 src/java/com/twitter/search/earlybird_root/caching/RecencyCache.java create mode 100644 src/java/com/twitter/search/earlybird_root/caching/RecencyCacheFilter.java create mode 100644 src/java/com/twitter/search/earlybird_root/caching/RecencyCacheRequestNormalizer.java create mode 100644 src/java/com/twitter/search/earlybird_root/caching/RecencyQueryCachePredicate.java create mode 100644 src/java/com/twitter/search/earlybird_root/caching/RecencyServicePostProcessor.java create mode 100644 src/java/com/twitter/search/earlybird_root/caching/RelevanceCache.java create mode 100644 src/java/com/twitter/search/earlybird_root/caching/RelevanceCacheFilter.java create mode 100644 src/java/com/twitter/search/earlybird_root/caching/RelevanceCacheRequestNormalizer.java create mode 100644 src/java/com/twitter/search/earlybird_root/caching/RelevanceQueryCachePredicate.java create mode 100644 src/java/com/twitter/search/earlybird_root/caching/RelevanceServicePostProcessor.java create mode 100644 src/java/com/twitter/search/earlybird_root/caching/RelevanceZeroResultsCacheFilter.java create mode 100644 src/java/com/twitter/search/earlybird_root/caching/RelevanceZeroResultsCachePostProcessor.java create mode 100644 src/java/com/twitter/search/earlybird_root/caching/RelevanceZeroResultsCacheRequestNormalizer.java create mode 100644 src/java/com/twitter/search/earlybird_root/caching/RelevanceZeroResultsQueryCachePredicate.java create mode 100644 src/java/com/twitter/search/earlybird_root/caching/RelevanceZeroResultsServicePostProcessor.java create mode 100644 src/java/com/twitter/search/earlybird_root/caching/StrictRecencyCache.java create mode 100644 src/java/com/twitter/search/earlybird_root/caching/StrictRecencyCacheFilter.java create mode 100644 src/java/com/twitter/search/earlybird_root/caching/StrictRecencyQueryCachePredicate.java create mode 100644 src/java/com/twitter/search/earlybird_root/caching/TermStatsCache.java create mode 100644 src/java/com/twitter/search/earlybird_root/caching/TermStatsCacheFilter.java create mode 100644 src/java/com/twitter/search/earlybird_root/caching/TermStatsCacheRequestNormalizer.java create mode 100644 src/java/com/twitter/search/earlybird_root/caching/TermStatsQueryCachePredicate.java create mode 100644 src/java/com/twitter/search/earlybird_root/caching/TermStatsServicePostProcessor.java create mode 100644 src/java/com/twitter/search/earlybird_root/caching/TopTweetsCache.java create mode 100644 src/java/com/twitter/search/earlybird_root/caching/TopTweetsCacheFilter.java create mode 100644 src/java/com/twitter/search/earlybird_root/caching/TopTweetsCacheRequestNormalizer.java create mode 100644 src/java/com/twitter/search/earlybird_root/caching/TopTweetsQueryCachePredicate.java create mode 100644 src/java/com/twitter/search/earlybird_root/caching/TopTweetsServicePostProcessor.java create mode 100644 src/java/com/twitter/search/earlybird_root/collectors/BUILD create mode 100644 src/java/com/twitter/search/earlybird_root/collectors/MultiwayMergeCollector.java create mode 100644 src/java/com/twitter/search/earlybird_root/collectors/RecencyMergeCollector.java create mode 100644 src/java/com/twitter/search/earlybird_root/collectors/RelevanceMergeCollector.java create mode 100644 src/java/com/twitter/search/earlybird_root/common/BUILD create mode 100644 src/java/com/twitter/search/earlybird_root/common/ClientErrorException.java create mode 100644 src/java/com/twitter/search/earlybird_root/common/EarlybirdFeatureSchemaMerger.java create mode 100644 src/java/com/twitter/search/earlybird_root/common/EarlybirdRequestContext.java create mode 100644 src/java/com/twitter/search/earlybird_root/common/EarlybirdRequestType.java create mode 100644 src/java/com/twitter/search/earlybird_root/common/EarlybirdRequestUtil.java create mode 100644 src/java/com/twitter/search/earlybird_root/common/EarlybirdServiceResponse.java create mode 100644 src/java/com/twitter/search/earlybird_root/common/InjectionNames.java create mode 100644 src/java/com/twitter/search/earlybird_root/common/QueryParsingUtils.java create mode 100644 src/java/com/twitter/search/earlybird_root/common/TwitterContextProvider.java create mode 100644 src/java/com/twitter/search/earlybird_root/config/BUILD.bazel create mode 100644 src/java/com/twitter/search/earlybird_root/config/RootClusterBoundaryInfo.java create mode 100644 src/java/com/twitter/search/earlybird_root/filters/BUILD create mode 100644 src/java/com/twitter/search/earlybird_root/filters/ClientIdArchiveAccessFilter.java create mode 100644 src/java/com/twitter/search/earlybird_root/filters/ClientIdQueryOperatorStatsFilter.java create mode 100644 src/java/com/twitter/search/earlybird_root/filters/ClientIdQuotaFilter.java create mode 100644 src/java/com/twitter/search/earlybird_root/filters/ClientIdTrackingFilter.java create mode 100644 src/java/com/twitter/search/earlybird_root/filters/ClientRequestTimeFilter.java create mode 100644 src/java/com/twitter/search/earlybird_root/filters/DeadlineTimeoutStatsFilter.java create mode 100644 src/java/com/twitter/search/earlybird_root/filters/DisableClientByTierFilter.java create mode 100644 src/java/com/twitter/search/earlybird_root/filters/DropAllProtectedOperatorFilter.java create mode 100644 src/java/com/twitter/search/earlybird_root/filters/EarlybirdClusterAvailableFilter.java create mode 100644 src/java/com/twitter/search/earlybird_root/filters/EarlybirdFeatureSchemaAnnotateFilter.java create mode 100644 src/java/com/twitter/search/earlybird_root/filters/EarlybirdResponseExceptionHandler.java create mode 100644 src/java/com/twitter/search/earlybird_root/filters/EarlybirdSuccessfulResponseHandler.java create mode 100644 src/java/com/twitter/search/earlybird_root/filters/EarlybirdTimeFilterQueryRewriter.java create mode 100644 src/java/com/twitter/search/earlybird_root/filters/EarlybirdTimeRangeFilter.java create mode 100644 src/java/com/twitter/search/earlybird_root/filters/FullArchiveProtectedOperatorFilter.java create mode 100644 src/java/com/twitter/search/earlybird_root/filters/FullArchiveServingRangeProvider.java create mode 100644 src/java/com/twitter/search/earlybird_root/filters/InitializeRequestContextFilter.java create mode 100644 src/java/com/twitter/search/earlybird_root/filters/IsUserProtectedMetadataTrackingFilter.java create mode 100644 src/java/com/twitter/search/earlybird_root/filters/MarkTweetSourceFilter.java create mode 100644 src/java/com/twitter/search/earlybird_root/filters/MetadataTrackingFilter.java create mode 100644 src/java/com/twitter/search/earlybird_root/filters/NamedMultiTermDisjunctionStatsFilter.java create mode 100644 src/java/com/twitter/search/earlybird_root/filters/NullcastTrackingFilter.java create mode 100644 src/java/com/twitter/search/earlybird_root/filters/PostCacheRequestTypeCountFilter.java create mode 100644 src/java/com/twitter/search/earlybird_root/filters/PreCacheRequestTypeCountFilter.java create mode 100644 src/java/com/twitter/search/earlybird_root/filters/QueryLangStatFilter.java create mode 100644 src/java/com/twitter/search/earlybird_root/filters/QueryOperatorStatFilter.java create mode 100644 src/java/com/twitter/search/earlybird_root/filters/QueryTokenizerFilter.java create mode 100644 src/java/com/twitter/search/earlybird_root/filters/RealtimeServingRangeProvider.java create mode 100644 src/java/com/twitter/search/earlybird_root/filters/RejectRequestsByQuerySourceFilter.java create mode 100644 src/java/com/twitter/search/earlybird_root/filters/RequestContextToEarlybirdRequestFilter.java create mode 100644 src/java/com/twitter/search/earlybird_root/filters/RequestResultStatsFilter.java create mode 100644 src/java/com/twitter/search/earlybird_root/filters/RequestSuccessStatsFilter.java create mode 100644 src/java/com/twitter/search/earlybird_root/filters/RequestTypeCountFilter.java create mode 100644 src/java/com/twitter/search/earlybird_root/filters/ResponseCodeStatFilter.java create mode 100644 src/java/com/twitter/search/earlybird_root/filters/ResultTierCountFilter.java create mode 100644 src/java/com/twitter/search/earlybird_root/filters/ScatterGatherWithExperimentRedirectsService.java create mode 100644 src/java/com/twitter/search/earlybird_root/filters/SearchPayloadSizeLocalContextFilter.java create mode 100644 src/java/com/twitter/search/earlybird_root/filters/SensitiveResultsTrackingFilter.java create mode 100644 src/java/com/twitter/search/earlybird_root/filters/ServiceExceptionHandlingFilter.java create mode 100644 src/java/com/twitter/search/earlybird_root/filters/ServiceResponseValidationFilter.java create mode 100644 src/java/com/twitter/search/earlybird_root/filters/ServingRangeProvider.java create mode 100644 src/java/com/twitter/search/earlybird_root/filters/StratoAttributionClientIdFilter.java create mode 100644 src/java/com/twitter/search/earlybird_root/filters/TopLevelExceptionHandlingFilter.java create mode 100644 src/java/com/twitter/search/earlybird_root/filters/UnsetSuperRootFieldsFilter.java create mode 100644 src/java/com/twitter/search/earlybird_root/filters/VeryRecentTweetsFilter.java create mode 100644 src/java/com/twitter/search/earlybird_root/img/serving.png create mode 100644 src/java/com/twitter/search/earlybird_root/mergers/AccumulatedResponses.java create mode 100644 src/java/com/twitter/search/earlybird_root/mergers/BUILD create mode 100644 src/java/com/twitter/search/earlybird_root/mergers/EarlyTerminateTierMergePredicate.java create mode 100644 src/java/com/twitter/search/earlybird_root/mergers/EarlybirdResponseDebugMessageBuilder.java create mode 100644 src/java/com/twitter/search/earlybird_root/mergers/EarlybirdResponseMerger.java create mode 100644 src/java/com/twitter/search/earlybird_root/mergers/FacetResponseMerger.java create mode 100644 src/java/com/twitter/search/earlybird_root/mergers/PartitionResponseAccumulator.java create mode 100644 src/java/com/twitter/search/earlybird_root/mergers/RecencyResponseMerger.java create mode 100644 src/java/com/twitter/search/earlybird_root/mergers/RelevanceResponseMerger.java create mode 100644 src/java/com/twitter/search/earlybird_root/mergers/ResponseAccumulator.java create mode 100644 src/java/com/twitter/search/earlybird_root/mergers/StrictRecencyResponseMerger.java create mode 100644 src/java/com/twitter/search/earlybird_root/mergers/SuperRootResponseMerger.java create mode 100644 src/java/com/twitter/search/earlybird_root/mergers/TermStatisticsResponseMerger.java create mode 100644 src/java/com/twitter/search/earlybird_root/mergers/ThriftTermResultsMerger.java create mode 100644 src/java/com/twitter/search/earlybird_root/mergers/TierResponseAccumulator.java create mode 100644 src/java/com/twitter/search/earlybird_root/mergers/TopTweetsResponseMerger.java create mode 100644 src/java/com/twitter/search/earlybird_root/mergers/TrimStats.java create mode 100644 src/java/com/twitter/search/earlybird_root/quota/BUILD create mode 100644 src/java/com/twitter/search/earlybird_root/quota/ClientIdQuotaManager.java create mode 100644 src/java/com/twitter/search/earlybird_root/quota/ConfigBasedQuotaConfig.java create mode 100644 src/java/com/twitter/search/earlybird_root/quota/ConfigRepoBasedQuotaManager.java create mode 100644 src/java/com/twitter/search/earlybird_root/quota/QuotaInfo.java create mode 100644 src/java/com/twitter/search/earlybird_root/routers/AbstractRecencyAndRelevanceRequestRouter.java create mode 100644 src/java/com/twitter/search/earlybird_root/routers/BUILD create mode 100644 src/java/com/twitter/search/earlybird_root/routers/FacetsRequestRouter.java create mode 100644 src/java/com/twitter/search/earlybird_root/routers/FacetsRequestRouterModule.java create mode 100644 src/java/com/twitter/search/earlybird_root/routers/RecencyRequestRouter.java create mode 100644 src/java/com/twitter/search/earlybird_root/routers/RecencyRequestRouterModule.java create mode 100644 src/java/com/twitter/search/earlybird_root/routers/RelevanceRequestRouter.java create mode 100644 src/java/com/twitter/search/earlybird_root/routers/RelevanceRequestRouterModule.java create mode 100644 src/java/com/twitter/search/earlybird_root/routers/RequestRouter.java create mode 100644 src/java/com/twitter/search/earlybird_root/routers/RequestRouterUtil.java create mode 100644 src/java/com/twitter/search/earlybird_root/routers/TermStatsRequestRouter.java create mode 100644 src/java/com/twitter/search/earlybird_root/routers/TermStatsRequestRouterModule.java create mode 100644 src/java/com/twitter/search/earlybird_root/routers/TopTweetsRequestRouter.java create mode 100644 src/java/com/twitter/search/earlybird_root/routers/TopTweetsRequestRouterModule.java create mode 100644 src/java/com/twitter/search/earlybird_root/validators/BUILD create mode 100644 src/java/com/twitter/search/earlybird_root/validators/FacetsResponseValidator.java create mode 100644 src/java/com/twitter/search/earlybird_root/validators/PassThroughResponseValidator.java create mode 100644 src/java/com/twitter/search/earlybird_root/validators/SearchResultsValidator.java create mode 100644 src/java/com/twitter/search/earlybird_root/validators/ServiceResponseValidator.java create mode 100644 src/java/com/twitter/search/earlybird_root/validators/TermStatsResultsValidator.java create mode 100644 src/java/com/twitter/search/earlybird_root/validators/TopTweetsResultsValidator.java create mode 100644 src/java/com/twitter/search/earlybird_root/visitors/BUILD create mode 100644 src/java/com/twitter/search/earlybird_root/visitors/MultiTermDisjunctionPerPartitionVisitor.java create mode 100644 src/java/com/twitter/search/feature_update_service/BUILD create mode 100644 src/java/com/twitter/search/feature_update_service/FeatureUpdateController.java create mode 100644 src/java/com/twitter/search/feature_update_service/FeatureUpdateResponseClassifier.java create mode 100644 src/java/com/twitter/search/feature_update_service/FeatureUpdateServiceThriftServer.java create mode 100644 src/java/com/twitter/search/feature_update_service/FeatureUpdateServiceThriftServerMain.java create mode 100644 src/java/com/twitter/search/feature_update_service/README.md create mode 100644 src/java/com/twitter/search/feature_update_service/filters/BUILD create mode 100644 src/java/com/twitter/search/feature_update_service/filters/ClientIdWhitelistFilter.java create mode 100644 src/java/com/twitter/search/feature_update_service/modules/BUILD create mode 100644 src/java/com/twitter/search/feature_update_service/modules/ClientIdWhitelistModule.java create mode 100644 src/java/com/twitter/search/feature_update_service/modules/EarlybirdUtilModule.java create mode 100644 src/java/com/twitter/search/feature_update_service/modules/FeatureUpdateServiceDiffyModule.java create mode 100644 src/java/com/twitter/search/feature_update_service/modules/FinagleKafkaProducerModule.java create mode 100644 src/java/com/twitter/search/feature_update_service/modules/FuturePoolModule.java create mode 100644 src/java/com/twitter/search/feature_update_service/modules/TweetypieModule.java create mode 100644 src/java/com/twitter/search/feature_update_service/stats/BUILD create mode 100644 src/java/com/twitter/search/feature_update_service/stats/FeatureUpdateStats.java create mode 100644 src/java/com/twitter/search/feature_update_service/util/BUILD create mode 100644 src/java/com/twitter/search/feature_update_service/util/FeatureUpdateValidator.java create mode 100644 src/java/com/twitter/search/feature_update_service/whitelist/BUILD create mode 100644 src/java/com/twitter/search/feature_update_service/whitelist/ClientIdWhitelist.java create mode 100644 src/java/com/twitter/search/img/foryou.png create mode 100644 src/java/com/twitter/search/img/in-network.png create mode 100644 src/java/com/twitter/search/img/indexing.png create mode 100644 src/java/com/twitter/search/img/serving.png create mode 100644 src/java/com/twitter/search/img/top-search.png create mode 100644 src/java/com/twitter/search/ingester/BUILD create mode 100644 src/java/com/twitter/search/ingester/README.md create mode 100644 src/java/com/twitter/search/ingester/model/BUILD create mode 100644 src/java/com/twitter/search/ingester/model/IndexerStatus.java create mode 100644 src/java/com/twitter/search/ingester/model/IngesterThriftVersionedEvents.java create mode 100644 src/java/com/twitter/search/ingester/model/IngesterTweetEvent.java create mode 100644 src/java/com/twitter/search/ingester/model/IngesterTwitterMessage.java create mode 100644 src/java/com/twitter/search/ingester/model/KafkaRawRecord.java create mode 100644 src/java/com/twitter/search/ingester/model/PromiseContainer.java create mode 100644 src/java/com/twitter/search/ingester/model/VisibleTokenRatioUtil.java create mode 100644 src/java/com/twitter/search/ingester/pipeline/app/BUILD create mode 100644 src/java/com/twitter/search/ingester/pipeline/app/IngesterPipelineApplication.java create mode 100644 src/java/com/twitter/search/ingester/pipeline/app/PipelineExceptionImpl.java create mode 100644 src/java/com/twitter/search/ingester/pipeline/app/PipelineExceptionImplV2.java create mode 100644 src/java/com/twitter/search/ingester/pipeline/app/RealtimeIngesterPipelineV2.java create mode 100644 src/java/com/twitter/search/ingester/pipeline/strato_fetchers/AudioSpaceCoreFetcher.java create mode 100644 src/java/com/twitter/search/ingester/pipeline/strato_fetchers/AudioSpaceParticipantsFetcher.java create mode 100644 src/java/com/twitter/search/ingester/pipeline/strato_fetchers/BUILD create mode 100644 src/java/com/twitter/search/ingester/pipeline/strato_fetchers/NamedEntityFetcher.java create mode 100644 src/java/com/twitter/search/ingester/pipeline/twitter/AsyncPinkUrlsResolver.java create mode 100644 src/java/com/twitter/search/ingester/pipeline/twitter/BUILD create mode 100644 src/java/com/twitter/search/ingester/pipeline/twitter/CollectComparableObjectsStage.java create mode 100644 src/java/com/twitter/search/ingester/pipeline/twitter/ComputeTweetSignatureStage.java create mode 100644 src/java/com/twitter/search/ingester/pipeline/twitter/ConvertDelayedMessageToThriftStage.java create mode 100644 src/java/com/twitter/search/ingester/pipeline/twitter/ConvertMessageToThriftStage.java create mode 100644 src/java/com/twitter/search/ingester/pipeline/twitter/ConvertToThriftVersionedEventsStage.java create mode 100644 src/java/com/twitter/search/ingester/pipeline/twitter/EventBusReaderStage.java create mode 100644 src/java/com/twitter/search/ingester/pipeline/twitter/FieldStatExporter.java create mode 100644 src/java/com/twitter/search/ingester/pipeline/twitter/FilterEventsBySafetyTypeStage.java create mode 100644 src/java/com/twitter/search/ingester/pipeline/twitter/FilterRetweetsAndRepliesStage.java create mode 100644 src/java/com/twitter/search/ingester/pipeline/twitter/FilterTwitterMessageStage.java create mode 100644 src/java/com/twitter/search/ingester/pipeline/twitter/LookupUserPropertiesBatchedStage.java create mode 100644 src/java/com/twitter/search/ingester/pipeline/twitter/NamedEntityHandler.java create mode 100644 src/java/com/twitter/search/ingester/pipeline/twitter/PopulateCodedLocationsBatchedStage.java create mode 100644 src/java/com/twitter/search/ingester/pipeline/twitter/ResolveCompressedUrlsBatchedStage.java create mode 100644 src/java/com/twitter/search/ingester/pipeline/twitter/ResolveCompressedUrlsPink.java create mode 100644 src/java/com/twitter/search/ingester/pipeline/twitter/ResolveCompressedUrlsUtils.java create mode 100644 src/java/com/twitter/search/ingester/pipeline/twitter/RetrieveCardBatchedStage.java create mode 100644 src/java/com/twitter/search/ingester/pipeline/twitter/RetrieveNamedEntitiesSingleTweetStage.java create mode 100644 src/java/com/twitter/search/ingester/pipeline/twitter/RetrieveSpaceAdminsAndTitleStage.java create mode 100644 src/java/com/twitter/search/ingester/pipeline/twitter/RetrieveSpaceIdsStage.java create mode 100644 src/java/com/twitter/search/ingester/pipeline/twitter/SingleTweetExtractAndGeocodeLatLonStage.java create mode 100644 src/java/com/twitter/search/ingester/pipeline/twitter/TextFeatureExtractionWorkersStage.java create mode 100644 src/java/com/twitter/search/ingester/pipeline/twitter/TextQualityEvaluationWorkerStage.java create mode 100644 src/java/com/twitter/search/ingester/pipeline/twitter/TextUrlsFeatureExtractionStage.java create mode 100644 src/java/com/twitter/search/ingester/pipeline/twitter/ThriftTweetParserStage.java create mode 100644 src/java/com/twitter/search/ingester/pipeline/twitter/ThriftVersionedEventsConverter.java create mode 100644 src/java/com/twitter/search/ingester/pipeline/twitter/TweetEventDeserializerStage.java create mode 100644 src/java/com/twitter/search/ingester/pipeline/twitter/TwitterBaseStage.java create mode 100644 src/java/com/twitter/search/ingester/pipeline/twitter/TwitterBatchedBaseStage.java create mode 100644 src/java/com/twitter/search/ingester/pipeline/twitter/filters/BUILD create mode 100644 src/java/com/twitter/search/ingester/pipeline/twitter/filters/IngesterValidMessageFilter.java create mode 100644 src/java/com/twitter/search/ingester/pipeline/twitter/kafka/BUILD create mode 100644 src/java/com/twitter/search/ingester/pipeline/twitter/kafka/DeleteUpdateEventsKafkaProducerStage.java create mode 100644 src/java/com/twitter/search/ingester/pipeline/twitter/kafka/KafkaConsumerStage.java create mode 100644 src/java/com/twitter/search/ingester/pipeline/twitter/kafka/KafkaProducerStage.java create mode 100644 src/java/com/twitter/search/ingester/pipeline/twitter/kafka/KafkaRawRecordConsumerStage.java create mode 100644 src/java/com/twitter/search/ingester/pipeline/twitter/kafka/RetweetAndReplyUpdateEventsKafkaProducerStage.java create mode 100644 src/java/com/twitter/search/ingester/pipeline/twitter/kafka/TweetThriftVersionedEventsKafkaProducerStage.java create mode 100644 src/java/com/twitter/search/ingester/pipeline/twitter/thriftparse/BUILD create mode 100644 src/java/com/twitter/search/ingester/pipeline/twitter/thriftparse/ThriftTweetParsingException.java create mode 100644 src/java/com/twitter/search/ingester/pipeline/twitter/thriftparse/TweetEventParseHelper.java create mode 100644 src/java/com/twitter/search/ingester/pipeline/twitter/userupdates/BUILD create mode 100644 src/java/com/twitter/search/ingester/pipeline/twitter/userupdates/UserUpdateIngester.java create mode 100644 src/java/com/twitter/search/ingester/pipeline/twitter/userupdates/UserUpdatesPipeline.java create mode 100644 src/java/com/twitter/search/ingester/pipeline/twitter/userupdates/UserUpdatesPipelineStage.java create mode 100644 src/java/com/twitter/search/ingester/pipeline/util/BUILD create mode 100644 src/java/com/twitter/search/ingester/pipeline/util/BatchedElement.java create mode 100644 src/java/com/twitter/search/ingester/pipeline/util/BatchingClient.java create mode 100644 src/java/com/twitter/search/ingester/pipeline/util/CardFieldUtil.java create mode 100644 src/java/com/twitter/search/ingester/pipeline/util/IngesterStageTimer.java create mode 100644 src/java/com/twitter/search/ingester/pipeline/util/ManhattanCodedLocationProvider.java create mode 100644 src/java/com/twitter/search/ingester/pipeline/util/PenguinVersionsUtil.java create mode 100644 src/java/com/twitter/search/ingester/pipeline/util/PipelineExceptionHandler.java create mode 100644 src/java/com/twitter/search/ingester/pipeline/util/PipelineStageException.java create mode 100644 src/java/com/twitter/search/ingester/pipeline/util/PipelineStageRuntimeException.java create mode 100644 src/java/com/twitter/search/ingester/pipeline/util/PipelineUtil.java create mode 100644 src/java/com/twitter/search/ingester/pipeline/util/PipelineV2CreationException.java create mode 100644 src/java/com/twitter/search/ingester/pipeline/util/ResponseNotReturnedException.java create mode 100644 src/java/com/twitter/search/ingester/pipeline/util/UserPropertiesManager.java create mode 100644 src/java/com/twitter/search/ingester/pipeline/wire/BUILD create mode 100644 src/java/com/twitter/search/ingester/pipeline/wire/IngesterPartitioner.java create mode 100644 src/java/com/twitter/search/ingester/pipeline/wire/ProductionWireModule.java create mode 100644 src/java/com/twitter/search/ingester/pipeline/wire/StratoMetaStoreWireModule.java create mode 100644 src/java/com/twitter/search/ingester/pipeline/wire/TweetyPieWireModule.java create mode 100644 src/java/com/twitter/search/ingester/pipeline/wire/WireModule.java create mode 100644 src/java/com/twitter/search/ingester/util/jndi/BUILD create mode 100644 src/java/com/twitter/search/ingester/util/jndi/JndiUtil.java create mode 100644 src/python/twitter/deepbird/projects/timelines/configs/recap_earlybird/feature_config.py create mode 100644 src/python/twitter/deepbird/projects/timelines/configs/rectweet_earlybird/feature_config.py create mode 100644 src/python/twitter/deepbird/projects/timelines/scripts/models/earlybird/BUILD create mode 100644 src/python/twitter/deepbird/projects/timelines/scripts/models/earlybird/README.md create mode 100644 src/python/twitter/deepbird/projects/timelines/scripts/models/earlybird/__init__.py create mode 100644 src/python/twitter/deepbird/projects/timelines/scripts/models/earlybird/constants.py create mode 100644 src/python/twitter/deepbird/projects/timelines/scripts/models/earlybird/earlybird_features.png create mode 100644 src/python/twitter/deepbird/projects/timelines/scripts/models/earlybird/example_weights.py create mode 100644 src/python/twitter/deepbird/projects/timelines/scripts/models/earlybird/lolly/BUILD create mode 100644 src/python/twitter/deepbird/projects/timelines/scripts/models/earlybird/lolly/__init__.py create mode 100644 src/python/twitter/deepbird/projects/timelines/scripts/models/earlybird/lolly/data_helpers.py create mode 100644 src/python/twitter/deepbird/projects/timelines/scripts/models/earlybird/lolly/parsers.py create mode 100644 src/python/twitter/deepbird/projects/timelines/scripts/models/earlybird/lolly/reader.py create mode 100644 src/python/twitter/deepbird/projects/timelines/scripts/models/earlybird/lolly/score.py create mode 100644 src/python/twitter/deepbird/projects/timelines/scripts/models/earlybird/lolly/scorer.py create mode 100644 src/python/twitter/deepbird/projects/timelines/scripts/models/earlybird/lolly/tf_model_initializer_builder.py create mode 100644 src/python/twitter/deepbird/projects/timelines/scripts/models/earlybird/metrics.py create mode 100644 src/python/twitter/deepbird/projects/timelines/scripts/models/earlybird/tf_model/BUILD create mode 100644 src/python/twitter/deepbird/projects/timelines/scripts/models/earlybird/tf_model/__init__.py create mode 100644 src/python/twitter/deepbird/projects/timelines/scripts/models/earlybird/tf_model/discretizer_builder.py create mode 100644 src/python/twitter/deepbird/projects/timelines/scripts/models/earlybird/tf_model/hashing_utils.py create mode 100644 src/python/twitter/deepbird/projects/timelines/scripts/models/earlybird/tf_model/weights_initializer_builder.py create mode 100644 src/python/twitter/deepbird/projects/timelines/scripts/models/earlybird/train.py create mode 100644 src/scala/com/twitter/graph/batch/BUILD.bazel create mode 100644 src/scala/com/twitter/graph/batch/job/tweepcred/ExtractTweepcred.scala create mode 100644 src/scala/com/twitter/graph/batch/job/tweepcred/PreparePageRankData.scala create mode 100644 src/scala/com/twitter/graph/batch/job/tweepcred/README create mode 100644 src/scala/com/twitter/graph/batch/job/tweepcred/Reputation.scala create mode 100644 src/scala/com/twitter/graph/batch/job/tweepcred/TweepcredBatchJob.scala create mode 100644 src/scala/com/twitter/graph/batch/job/tweepcred/UserMass.scala create mode 100644 src/scala/com/twitter/graph/batch/job/tweepcred/WeightedPageRank.scala create mode 100644 src/scala/com/twitter/interaction_graph/README.md create mode 100644 src/scala/com/twitter/interaction_graph/bqe/scoring/README.md create mode 100644 src/scala/com/twitter/interaction_graph/bqe/scoring/candidates.sql create mode 100644 src/scala/com/twitter/interaction_graph/bqe/scoring/check_models.sql create mode 100644 src/scala/com/twitter/interaction_graph/bqe/scoring/follow_graph_features.sql create mode 100644 src/scala/com/twitter/interaction_graph/bqe/scoring/scoring.sql create mode 100644 src/scala/com/twitter/interaction_graph/bqe/training/README.md create mode 100644 src/scala/com/twitter/interaction_graph/bqe/training/candidates.sql create mode 100644 src/scala/com/twitter/interaction_graph/bqe/training/check_candidates_exist.sql create mode 100644 src/scala/com/twitter/interaction_graph/bqe/training/check_labels_exist.sql create mode 100644 src/scala/com/twitter/interaction_graph/bqe/training/labeled_candidates.sql create mode 100644 src/scala/com/twitter/interaction_graph/bqe/training/train_model.sql create mode 100644 src/scala/com/twitter/interaction_graph/injection/BUILD create mode 100644 src/scala/com/twitter/interaction_graph/injection/EdgeListInjection.scala create mode 100644 src/scala/com/twitter/interaction_graph/injection/UserSessionInjection.scala create mode 100644 src/scala/com/twitter/interaction_graph/scio/README.md create mode 100644 src/scala/com/twitter/interaction_graph/scio/agg_address_book/BUILD create mode 100644 src/scala/com/twitter/interaction_graph/scio/agg_address_book/InteractionGraphAddressBookCounters.scala create mode 100644 src/scala/com/twitter/interaction_graph/scio/agg_address_book/InteractionGraphAddressBookJob.scala create mode 100644 src/scala/com/twitter/interaction_graph/scio/agg_address_book/InteractionGraphAddressBookOption.scala create mode 100644 src/scala/com/twitter/interaction_graph/scio/agg_address_book/InteractionGraphAddressBookSource.scala create mode 100644 src/scala/com/twitter/interaction_graph/scio/agg_address_book/InteractionGraphAddressBookUtil.scala create mode 100644 src/scala/com/twitter/interaction_graph/scio/agg_address_book/README.md create mode 100644 src/scala/com/twitter/interaction_graph/scio/agg_all/BUILD create mode 100644 src/scala/com/twitter/interaction_graph/scio/agg_all/InteractionGraphAggregationConfig.scala create mode 100644 src/scala/com/twitter/interaction_graph/scio/agg_all/InteractionGraphAggregationJob.scala create mode 100644 src/scala/com/twitter/interaction_graph/scio/agg_all/InteractionGraphAggregationOption.scala create mode 100644 src/scala/com/twitter/interaction_graph/scio/agg_all/InteractionGraphAggregationSource.scala create mode 100644 src/scala/com/twitter/interaction_graph/scio/agg_all/InteractionGraphAggregationTransform.scala create mode 100644 src/scala/com/twitter/interaction_graph/scio/agg_all/README.md create mode 100644 src/scala/com/twitter/interaction_graph/scio/agg_client_event_logs/BUILD create mode 100644 src/scala/com/twitter/interaction_graph/scio/agg_client_event_logs/InteractionGraphClientEventLogsCounters.scala create mode 100644 src/scala/com/twitter/interaction_graph/scio/agg_client_event_logs/InteractionGraphClientEventLogsJob.scala create mode 100644 src/scala/com/twitter/interaction_graph/scio/agg_client_event_logs/InteractionGraphClientEventLogsOption.scala create mode 100644 src/scala/com/twitter/interaction_graph/scio/agg_client_event_logs/InteractionGraphClientEventLogsSource.scala create mode 100644 src/scala/com/twitter/interaction_graph/scio/agg_client_event_logs/InteractionGraphClientEventLogsUtil.scala create mode 100644 src/scala/com/twitter/interaction_graph/scio/agg_client_event_logs/README.md create mode 100644 src/scala/com/twitter/interaction_graph/scio/agg_direct_interactions/BUILD create mode 100644 src/scala/com/twitter/interaction_graph/scio/agg_direct_interactions/InteractionGraphAggDirectInteractionsJob.scala create mode 100644 src/scala/com/twitter/interaction_graph/scio/agg_direct_interactions/InteractionGraphAggDirectInteractionsOption.scala create mode 100644 src/scala/com/twitter/interaction_graph/scio/agg_direct_interactions/InteractionGraphAggDirectInteractionsSource.scala create mode 100644 src/scala/com/twitter/interaction_graph/scio/agg_direct_interactions/InteractionGraphAggDirectInteractionsUtil.scala create mode 100644 src/scala/com/twitter/interaction_graph/scio/agg_direct_interactions/README.md create mode 100644 src/scala/com/twitter/interaction_graph/scio/agg_flock/BUILD create mode 100644 src/scala/com/twitter/interaction_graph/scio/agg_flock/InteractionGraphAggFlockJob.scala create mode 100644 src/scala/com/twitter/interaction_graph/scio/agg_flock/InteractionGraphAggFlockOption.scala create mode 100644 src/scala/com/twitter/interaction_graph/scio/agg_flock/InteractionGraphAggFlockSource.scala create mode 100644 src/scala/com/twitter/interaction_graph/scio/agg_flock/InteractionGraphAggFlockUtil.scala create mode 100644 src/scala/com/twitter/interaction_graph/scio/agg_flock/README.md create mode 100644 src/scala/com/twitter/interaction_graph/scio/agg_negative/BUILD create mode 100644 src/scala/com/twitter/interaction_graph/scio/agg_negative/InteractionGraphNegativeJob.scala create mode 100644 src/scala/com/twitter/interaction_graph/scio/agg_negative/InteractionGraphNegativeOption.scala create mode 100644 src/scala/com/twitter/interaction_graph/scio/agg_negative/README.md create mode 100644 src/scala/com/twitter/interaction_graph/scio/agg_notifications/BUILD create mode 100644 src/scala/com/twitter/interaction_graph/scio/agg_notifications/InteractionGraphNotificationUtil.scala create mode 100644 src/scala/com/twitter/interaction_graph/scio/agg_notifications/InteractionGraphNotificationsJob.scala create mode 100644 src/scala/com/twitter/interaction_graph/scio/agg_notifications/InteractionGraphNotificationsOption.scala create mode 100644 src/scala/com/twitter/interaction_graph/scio/agg_notifications/README.md create mode 100644 src/scala/com/twitter/interaction_graph/scio/common/BUILD create mode 100644 src/scala/com/twitter/interaction_graph/scio/common/CaseClasses.scala create mode 100644 src/scala/com/twitter/interaction_graph/scio/common/ConversionUtil.scala create mode 100644 src/scala/com/twitter/interaction_graph/scio/common/DateUtil.scala create mode 100644 src/scala/com/twitter/interaction_graph/scio/common/EdgeFeatureCombiner.scala create mode 100644 src/scala/com/twitter/interaction_graph/scio/common/FeatureGeneratorUtil.scala create mode 100644 src/scala/com/twitter/interaction_graph/scio/common/FeatureGroups.scala create mode 100644 src/scala/com/twitter/interaction_graph/scio/common/GraphUtil.scala create mode 100644 src/scala/com/twitter/interaction_graph/scio/common/InteractionGraphUtils.scala create mode 100644 src/scala/com/twitter/interaction_graph/scio/common/UserUtil.scala create mode 100644 src/scala/com/twitter/interaction_graph/scio/common/VertexFeatureCombiner.scala create mode 100644 src/scala/com/twitter/interaction_graph/scio/ml/labels/BUILD create mode 100644 src/scala/com/twitter/interaction_graph/scio/ml/labels/InteractionGraphLabelsJob.scala create mode 100644 src/scala/com/twitter/interaction_graph/scio/ml/labels/InteractionGraphLabelsOption.scala create mode 100644 src/scala/com/twitter/interaction_graph/scio/ml/labels/LabelUtil.scala create mode 100644 src/scala/com/twitter/interaction_graph/scio/ml/labels/README.md create mode 100644 src/scala/com/twitter/interaction_graph/scio/ml/scores/BUILD create mode 100644 src/scala/com/twitter/interaction_graph/scio/ml/scores/InteractionGraphScoreExportJob.scala create mode 100644 src/scala/com/twitter/interaction_graph/scio/ml/scores/InteractionGraphScoreExportOption.scala create mode 100644 src/scala/com/twitter/interaction_graph/scio/ml/scores/README.md create mode 100644 src/scala/com/twitter/recos/decider/BUILD create mode 100644 src/scala/com/twitter/recos/decider/BaseDecider.scala create mode 100644 src/scala/com/twitter/recos/decider/EndpointLoadShedder.scala create mode 100644 src/scala/com/twitter/recos/graph_common/ActionEdgeTypeMask.scala create mode 100644 src/scala/com/twitter/recos/graph_common/BUILD create mode 100644 src/scala/com/twitter/recos/graph_common/BipartiteGraphHelper.scala create mode 100644 src/scala/com/twitter/recos/graph_common/FinagleCounterWrapper.scala create mode 100644 src/scala/com/twitter/recos/graph_common/FinagleStatsReceiverWrapper.scala create mode 100644 src/scala/com/twitter/recos/graph_common/LeftIndexedPowerLawMultiSegmentBipartiteGraphBuilder.scala create mode 100644 src/scala/com/twitter/recos/graph_common/MultiSegmentPowerLawBipartiteGraphBuilder.scala create mode 100644 src/scala/com/twitter/recos/graph_common/NodeInfoHandler.scala create mode 100644 src/scala/com/twitter/recos/graph_common/NodeMetadataLeftIndexedPowerLawMultiSegmentBipartiteGraphBuilder.scala create mode 100644 src/scala/com/twitter/recos/graph_common/RightNodeMetadataLeftIndexedPowerLawMultiSegmentBipartiteGraphBuilder.scala create mode 100644 src/scala/com/twitter/recos/hose/common/BUILD create mode 100644 src/scala/com/twitter/recos/hose/common/BufferedEdgeWriter.scala create mode 100644 src/scala/com/twitter/recos/hose/common/EdgeCollector.scala create mode 100644 src/scala/com/twitter/recos/hose/common/RecosEdgeProcessor.scala create mode 100644 src/scala/com/twitter/recos/hose/common/UnifiedGraphWriter.scala create mode 100644 src/scala/com/twitter/recos/hose/common/UnifiedGraphWriterMulti.scala create mode 100644 src/scala/com/twitter/recos/user_tweet_entity_graph/BUILD create mode 100644 src/scala/com/twitter/recos/user_tweet_entity_graph/EntitySocialProofRunner.scala create mode 100644 src/scala/com/twitter/recos/user_tweet_entity_graph/LoggingUserTweetEntityGraph.scala create mode 100644 src/scala/com/twitter/recos/user_tweet_entity_graph/Main.scala create mode 100644 src/scala/com/twitter/recos/user_tweet_entity_graph/README.md create mode 100644 src/scala/com/twitter/recos/user_tweet_entity_graph/RecommendationHandler.scala create mode 100644 src/scala/com/twitter/recos/user_tweet_entity_graph/RecosConfig.scala create mode 100644 src/scala/com/twitter/recos/user_tweet_entity_graph/SocialProofHandler.scala create mode 100644 src/scala/com/twitter/recos/user_tweet_entity_graph/SocialProofHydrator.scala create mode 100644 src/scala/com/twitter/recos/user_tweet_entity_graph/TweetRecommendationsRunner.scala create mode 100644 src/scala/com/twitter/recos/user_tweet_entity_graph/TweetSocialProofHandler.scala create mode 100644 src/scala/com/twitter/recos/user_tweet_entity_graph/TweetSocialProofRunner.scala create mode 100644 src/scala/com/twitter/recos/user_tweet_entity_graph/UserTweetEdgeTypeMask.scala create mode 100644 src/scala/com/twitter/recos/user_tweet_entity_graph/UserTweetEntityGraph.scala create mode 100644 src/scala/com/twitter/recos/user_tweet_entity_graph/UserTweetEntityGraphWriter.scala create mode 100644 src/scala/com/twitter/recos/user_tweet_graph/BUILD create mode 100644 src/scala/com/twitter/recos/user_tweet_graph/Main.scala create mode 100644 src/scala/com/twitter/recos/user_tweet_graph/README.md create mode 100644 src/scala/com/twitter/recos/user_tweet_graph/UserTweetGraph.scala create mode 100644 src/scala/com/twitter/recos/user_tweet_graph/UserTweetGraphConfig.scala create mode 100644 src/scala/com/twitter/recos/user_tweet_graph/UserTweetGraphWriter.scala create mode 100644 src/scala/com/twitter/recos/user_tweet_graph/relatedTweetHandlers/BUILD create mode 100644 src/scala/com/twitter/recos/user_tweet_graph/relatedTweetHandlers/ConsumersBasedRelatedTweetsHandler.scala create mode 100644 src/scala/com/twitter/recos/user_tweet_graph/relatedTweetHandlers/ProducerBasedRelatedTweetsHandler.scala create mode 100644 src/scala/com/twitter/recos/user_tweet_graph/relatedTweetHandlers/TweetBasedRelatedTweetsHandler.scala create mode 100644 src/scala/com/twitter/recos/user_tweet_graph/store/BUILD create mode 100644 src/scala/com/twitter/recos/user_tweet_graph/store/UserRecentFollowersStore.scala create mode 100644 src/scala/com/twitter/recos/user_tweet_graph/util/BUILD create mode 100644 src/scala/com/twitter/recos/user_tweet_graph/util/FetchRHSTweetsUtil.scala create mode 100644 src/scala/com/twitter/recos/user_tweet_graph/util/FilterUtil.scala create mode 100644 src/scala/com/twitter/recos/user_tweet_graph/util/GetAllInternalTweetIdsUtil.scala create mode 100644 src/scala/com/twitter/recos/user_tweet_graph/util/GetRelatedTweetCandidatesUtil.scala create mode 100644 src/scala/com/twitter/recos/user_tweet_graph/util/SampleLHSUsersUtil.scala create mode 100644 src/scala/com/twitter/recos/user_tweet_graph/util/UserTweetEdgeTypeMask.scala create mode 100644 src/scala/com/twitter/recos/user_user_graph/BUILD create mode 100644 src/scala/com/twitter/recos/user_user_graph/KafkaConfig.scala create mode 100644 src/scala/com/twitter/recos/user_user_graph/LoggingUserUserGraph.scala create mode 100644 src/scala/com/twitter/recos/user_user_graph/Main.scala create mode 100644 src/scala/com/twitter/recos/user_user_graph/README.md create mode 100644 src/scala/com/twitter/recos/user_user_graph/RecommendUsersHandler.scala create mode 100644 src/scala/com/twitter/recos/user_user_graph/RecosConfig.scala create mode 100644 src/scala/com/twitter/recos/user_user_graph/UserEdgeTypeMask.scala create mode 100644 src/scala/com/twitter/recos/user_user_graph/UserUserGraph.scala create mode 100644 src/scala/com/twitter/recos/user_user_graph/UserUserGraphWriter.scala create mode 100644 src/scala/com/twitter/recos/user_video_graph/BUILD create mode 100644 src/scala/com/twitter/recos/user_video_graph/LoggingUserVideoGraph.scala create mode 100644 src/scala/com/twitter/recos/user_video_graph/Main.scala create mode 100644 src/scala/com/twitter/recos/user_video_graph/README.md create mode 100644 src/scala/com/twitter/recos/user_video_graph/UserVideoEdgeTypeMask.scala create mode 100644 src/scala/com/twitter/recos/user_video_graph/UserVideoGraph.scala create mode 100644 src/scala/com/twitter/recos/user_video_graph/UserVideoGraphConfig.scala create mode 100644 src/scala/com/twitter/recos/user_video_graph/UserVideoGraphEdgeHttpHandler.scala create mode 100644 src/scala/com/twitter/recos/user_video_graph/UserVideoGraphWriter.scala create mode 100644 src/scala/com/twitter/recos/user_video_graph/relatedTweetHandlers/BUILD create mode 100644 src/scala/com/twitter/recos/user_video_graph/relatedTweetHandlers/ConsumersBasedRelatedTweetsHandler.scala create mode 100644 src/scala/com/twitter/recos/user_video_graph/relatedTweetHandlers/ProducerBasedRelatedTweetsHandler.scala create mode 100644 src/scala/com/twitter/recos/user_video_graph/relatedTweetHandlers/TweetBasedRelatedTweetsHandler.scala create mode 100644 src/scala/com/twitter/recos/user_video_graph/store/BUILD create mode 100644 src/scala/com/twitter/recos/user_video_graph/store/UserRecentFollowersStore.scala create mode 100644 src/scala/com/twitter/recos/user_video_graph/util/BUILD create mode 100644 src/scala/com/twitter/recos/user_video_graph/util/FetchRHSTweetsUtil.scala create mode 100644 src/scala/com/twitter/recos/user_video_graph/util/FilterUtil.scala create mode 100644 src/scala/com/twitter/recos/user_video_graph/util/GetAllInternalTweetIdsUtil.scala create mode 100644 src/scala/com/twitter/recos/user_video_graph/util/GetRelatedTweetCandidatesUtil.scala create mode 100644 src/scala/com/twitter/recos/user_video_graph/util/SampleLHSUsersUtil.scala create mode 100644 src/scala/com/twitter/simclusters_v2/README.md create mode 100644 src/scala/com/twitter/simclusters_v2/candidate_source/BUILD create mode 100644 src/scala/com/twitter/simclusters_v2/candidate_source/ClusterRanker.scala create mode 100644 src/scala/com/twitter/simclusters_v2/candidate_source/HeavyRanker.scala create mode 100644 src/scala/com/twitter/simclusters_v2/candidate_source/SimClustersANNCandidateSource.scala create mode 100644 src/scala/com/twitter/simclusters_v2/candidate_source/SimClustersANNWrapperCandidateSource.scala create mode 100644 src/scala/com/twitter/simclusters_v2/common/BUILD create mode 100644 src/scala/com/twitter/simclusters_v2/common/CosineSimilarityUtil.scala create mode 100644 src/scala/com/twitter/simclusters_v2/common/DeciderGateBuilderWithIdHashing.scala create mode 100644 src/scala/com/twitter/simclusters_v2/common/ModelVersions.scala create mode 100644 src/scala/com/twitter/simclusters_v2/common/SeqStandardDeviation.scala create mode 100644 src/scala/com/twitter/simclusters_v2/common/SimClustersEmbedding.scala create mode 100644 src/scala/com/twitter/simclusters_v2/common/SimClustersEmbeddingId.scala create mode 100644 src/scala/com/twitter/simclusters_v2/common/SimClustersEmbeddingIdCacheKeyBuilder.scala create mode 100644 src/scala/com/twitter/simclusters_v2/common/SimClustersEmbeddingMonoid.scala create mode 100644 src/scala/com/twitter/simclusters_v2/common/SimClustersMultiEmbedding.scala create mode 100644 src/scala/com/twitter/simclusters_v2/common/SimClustersMultiEmbeddingId.scala create mode 100644 src/scala/com/twitter/simclusters_v2/common/clustering/BUILD create mode 100644 src/scala/com/twitter/simclusters_v2/common/clustering/ClusterRepresentativeSelectionMethod.scala create mode 100644 src/scala/com/twitter/simclusters_v2/common/clustering/ClusteringMethod.scala create mode 100644 src/scala/com/twitter/simclusters_v2/common/clustering/ConnectedComponentsClusteringMethod.scala create mode 100644 src/scala/com/twitter/simclusters_v2/common/clustering/LargestDimensionClusteringMethod.scala create mode 100644 src/scala/com/twitter/simclusters_v2/common/clustering/LouvainClusteringMethod.scala create mode 100644 src/scala/com/twitter/simclusters_v2/common/clustering/MaxFavScoreRepresentativeSelectionMethod.scala create mode 100644 src/scala/com/twitter/simclusters_v2/common/clustering/MedoidRepresentativeSelectionMethod.scala create mode 100644 src/scala/com/twitter/simclusters_v2/common/clustering/SimilarityFunctions.scala create mode 100644 src/scala/com/twitter/simclusters_v2/common/ml/BUILD create mode 100644 src/scala/com/twitter/simclusters_v2/common/ml/SimClustersEmbeddingAdapter.scala create mode 100644 src/scala/com/twitter/simclusters_v2/common/package.scala create mode 100644 src/scala/com/twitter/simclusters_v2/hdfs_sources/AdhocSources.scala create mode 100644 src/scala/com/twitter/simclusters_v2/hdfs_sources/BUILD create mode 100644 src/scala/com/twitter/simclusters_v2/hdfs_sources/DataPaths.scala create mode 100644 src/scala/com/twitter/simclusters_v2/hdfs_sources/DataSources.scala create mode 100644 src/scala/com/twitter/simclusters_v2/hdfs_sources/EntityEmbeddingsSources.scala create mode 100644 src/scala/com/twitter/simclusters_v2/hdfs_sources/InterestedInSources.scala create mode 100644 src/scala/com/twitter/simclusters_v2/hdfs_sources/ProducerEmbeddingSources.scala create mode 100644 src/scala/com/twitter/simclusters_v2/hdfs_sources/injections/BUILD create mode 100644 src/scala/com/twitter/simclusters_v2/hdfs_sources/injections/ClusterDetailsInjection.scala create mode 100644 src/scala/com/twitter/simclusters_v2/hdfs_sources/injections/ClusterTopMediaTweetsInjection.scala create mode 100644 src/scala/com/twitter/simclusters_v2/hdfs_sources/injections/ClusterTopTweetsInjection.scala create mode 100644 src/scala/com/twitter/simclusters_v2/hdfs_sources/injections/ClusteringInjections.scala create mode 100644 src/scala/com/twitter/simclusters_v2/hdfs_sources/injections/EntityEmbeddingsInjections.scala create mode 100644 src/scala/com/twitter/simclusters_v2/hdfs_sources/injections/InferredEntitiesInjections.scala create mode 100644 src/scala/com/twitter/simclusters_v2/hdfs_sources/injections/InterestedInInjection.scala create mode 100644 src/scala/com/twitter/simclusters_v2/hdfs_sources/injections/KnownForInjection.scala create mode 100644 src/scala/com/twitter/simclusters_v2/hdfs_sources/injections/MultiTypeGraphInjections.scala create mode 100644 src/scala/com/twitter/simclusters_v2/hdfs_sources/injections/ProducerEmbeddingsInjections.scala create mode 100644 src/scala/com/twitter/simclusters_v2/hdfs_sources/injections/SemanticCoreEntitiesInjections.scala create mode 100644 src/scala/com/twitter/simclusters_v2/hdfs_sources/injections/SingleSideUserScoresInjection.scala create mode 100644 src/scala/com/twitter/simclusters_v2/hdfs_sources/presto_hdfs_sources/BUILD create mode 100644 src/scala/com/twitter/simclusters_v2/hdfs_sources/presto_hdfs_sources/EntityEmbeddingsPrestoSources.scala create mode 100644 src/scala/com/twitter/simclusters_v2/images/bipartite_graph.png create mode 100644 src/scala/com/twitter/simclusters_v2/images/interestedin.png create mode 100644 src/scala/com/twitter/simclusters_v2/images/knownfor.png create mode 100644 src/scala/com/twitter/simclusters_v2/images/producer_embeddings.png create mode 100644 src/scala/com/twitter/simclusters_v2/images/producer_producer_similarity.png create mode 100644 src/scala/com/twitter/simclusters_v2/images/topic_embeddings.png create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/BUILD create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/BipartiteClusterEvaluation.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/BipartiteClusterEvaluationClasses.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/ClusterDetailsJob.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/ClusterEvaluation.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/CompareClusters.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/EigenVectorsForSparseSymmetric.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/InterestedInFromAggregatableProducerEmbeddings.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/InterestedInFromKnownFor.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/InterestedInFromKnownForLite.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/InterestedInFromProducerEmbeddingsAdhocApp.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/KnownForSources.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/ProducerNormsAndCounts.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/TopUsersSimilarityGraph.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/UpdateKnownFor.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/UpdateKnownForApps.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/UserUserFavGraph.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/UserUserGraph.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/UserUserNormalizedGraph.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/common/BUILD create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/common/PersistentTweetEmbeddingSource.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/common/QTreeMultiAggregator.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/common/TypedRichPipe.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/common/Util.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/common/matrix/BUILD create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/common/matrix/DenseRowMatrix.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/common/matrix/SparseMatrix.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/common/matrix/SparseRowMatrix.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/common/matrix/TypedPipeMatrix.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/embedding/BUILD create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/embedding/EntityEmbeddingFromProducerEmbeddingJob.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/embedding/EntityToSimClustersEmbeddingsJob.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/embedding/GlobalSimClustersLanguageEmbedding.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/embedding/LocaleEntitySimClustersEmbeddingV2Job.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/embedding/LocaleEntitySimClustersEmbeddingsJob.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/embedding/ProducerEmbeddingsFromInterestedIn.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/embedding/SimilarUsersBySimClustersEmbedding.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/embedding/abuse/AbuseSimclusterFeaturesScaldingJob.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/embedding/abuse/AdhocAbuseSimClusterFeaturesScaldingJob.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/embedding/abuse/BUILD create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/embedding/abuse/CrossSimClusterFeaturesScaldingJob.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/embedding/abuse/DataSources.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/embedding/abuse/PairedinteractionFeatures.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/embedding/abuse/SingleSideInteractionTransformation.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/embedding/common/EmbeddingUtil.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/embedding/common/EntityEmbeddingUtil.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/embedding/common/ExternalDataSources.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/embedding/common/SimClustersEmbeddingJob.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/embedding/producer/AggregatableFavBasedProducerEmbeddings.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/embedding/producer/AggregatableFollowBasedProducerEmbeddings.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/embedding/producer/AggregatableLogFavBasedProducerEmbeddings.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/embedding/producer/AggregatableProducerEmbeddings.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/embedding/producer/BUILD.bazel create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/embedding/tfg/BUILD create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/embedding/tfg/EngagementWeightedTfgBasedTopicEmbeddingsJob.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/embedding/tfg/FavInferredLanguageTfgBasedTopicEmbeddings.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/embedding/tfg/FavTfgBasedTopicEmbeddings.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/embedding/tfg/InferredLanguageTfgBasedTopicEmbeddingsBaseApp.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/embedding/tfg/LogFavTfgBasedTopicEmbeddings.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/embedding/tfg/README create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/embedding/tfg/TfgBasedTopicEmbeddingsBaseApp.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/embedding/twice/BUILD.bazel create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/embedding/twice/InterestedInTwice.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/embedding/twice/InterestedInTwiceBaseApp.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/evaluation/BUILD.bazel create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/evaluation/CandidateEvaluationBase.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/evaluation/EvaluationMetricHelper.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/evaluation/EvaluationReferenceDataExtraction.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/evaluation/LabelCorrelationsHelper.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/evaluation/SimClustersEvaluationAdhocApp.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/inferred_entities/BUILD.bazel create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/inferred_entities/InferredEntities.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/inferred_entities/InferredEntitiesFromInterestedIn.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/inferred_entities/InferredSemanticCoreEntitiesFromKnownFor.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/inferred_entities/ProdSources.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/mbcg/AllFeatures.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/mbcg/BUILD.bazel create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/mbcg/RecordAdapters.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/mbcg/TweetEmbeddingGenerationJob.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/mbcg/UserEmbeddingGenerationJob.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/multi_type_graph/assemble_multi_type_graph/AssembleMultiTypeGraph.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/multi_type_graph/assemble_multi_type_graph/AssembleMultiTypeGraphApp.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/multi_type_graph/assemble_multi_type_graph/AssembleMultiTypeGraphBaseApp.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/multi_type_graph/assemble_multi_type_graph/BUILD create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/multi_type_graph/assemble_multi_type_graph/Config.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/offline_job/BUILD.bazel create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/offline_job/OfflineTweetRecommendation.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/offline_job/SimClustersOfflineJob.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/offline_job/SimClustersOfflineJobAdhocApp.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/offline_job/SimClustersOfflineJobScheduledApp.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/offline_job/SimClustersOfflineJobUtil.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/offline_job/adhoc/BUILD.bazel create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/offline_job/adhoc/README create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/offline_job/adhoc/SimClustersTweetEmbeddingAdhocApp.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/offline_job/adhoc/TweetSimilarityEvaluationAdhocApp.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/offline_tweets/BUILD.bazel create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/offline_tweets/ClusterTopMediaTweetsJob.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/optout/BUILD.bazel create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/optout/InterestedInOptOut.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/optout/KnownForOptOut.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/optout/SimClustersOptOutUtil.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/topic_recommendations/BUILD create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/topic_recommendations/GeoPopularTopicsApp.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/topic_recommendations/ProducersForTopicsFromTopicFollowGraph.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/topic_recommendations/SimilarTopicsFromTopicFollowGraphApp.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/topic_recommendations/TopicsForProducersFromEM.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/topic_recommendations/TopicsForProducersUtils.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/topic_recommendations/model_based_topic_recommendations/BUILD create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/topic_recommendations/model_based_topic_recommendations/DataSources.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/topic_recommendations/model_based_topic_recommendations/UserFeatures.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/topic_recommendations/model_based_topic_recommendations/UserTopicDataRecordAdapter.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/topic_recommendations/model_based_topic_recommendations/UserTopicModellingTrainingDataCollectionJob.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/tweet_similarity/BUILD create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/tweet_similarity/DatasetTopKAnalysisJob.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/tweet_similarity/TrainingDataCollectionJob.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/tweet_similarity/TrainingDataCollectionUtil.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/tweet_similarity/TweetPairFeatureHydrationUtil.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/tweet_similarity/TweetPairLabelCollectionUtil.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/tweet_similarity/UnhydratedPairsCollectionJob.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/tweet_similarity/evaluation/BUILD.bazel create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/tweet_similarity/evaluation/ModelEvalAdhocApp.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/tweet_similarity/evaluation/RUXLandingDdgAnalysisAdhocApp.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/update_known_for/BUILD.bazel create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/update_known_for/UpdateKnownFor20M145K2020.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scalding/update_known_for/UpdateKnownForSBFRunner.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scio/bq_generation/common/BQGenerationUtil.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scio/bq_generation/common/BUILD create mode 100644 src/scala/com/twitter/simclusters_v2/scio/bq_generation/common/IndexGenerationUtil.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scio/bq_generation/ftr_tweet/BUILD create mode 100644 src/scala/com/twitter/simclusters_v2/scio/bq_generation/ftr_tweet/Config.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scio/bq_generation/ftr_tweet/FTRJob.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scio/bq_generation/ftr_tweet/FtrClusterToTweetIndexGenerationJob.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scio/bq_generation/ftr_tweet/README.md create mode 100644 src/scala/com/twitter/simclusters_v2/scio/bq_generation/ftr_tweet/ftr-based-simclusters-index-generation-job.d6w create mode 100644 src/scala/com/twitter/simclusters_v2/scio/bq_generation/ftr_tweet/ftr-tweets-ann-adhoc-job.d6w create mode 100644 src/scala/com/twitter/simclusters_v2/scio/bq_generation/ftr_tweet/iikf2020-decayed-sum-ann-batch-job.d6w create mode 100644 src/scala/com/twitter/simclusters_v2/scio/bq_generation/ftr_tweet/iikf2020-ftrat5-pop1000-ann-batch-job.d6w create mode 100644 src/scala/com/twitter/simclusters_v2/scio/bq_generation/ftr_tweet/iikf2020-ftrat5-pop10000-ann-batch-job.d6w create mode 100644 src/scala/com/twitter/simclusters_v2/scio/bq_generation/ftr_tweet/sql/BUILD create mode 100644 src/scala/com/twitter/simclusters_v2/scio/bq_generation/ftr_tweet/sql/ftr_tweet_embeddings.sql create mode 100644 src/scala/com/twitter/simclusters_v2/scio/bq_generation/simclusters_index_generation/BUILD create mode 100644 src/scala/com/twitter/simclusters_v2/scio/bq_generation/simclusters_index_generation/Config.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scio/bq_generation/simclusters_index_generation/EngagementEventBasedClusterToTweetIndexFromBQ.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scio/bq_generation/simclusters_index_generation/EngagementEventBasedClusterToTweetIndexGenerationJob.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scio/bq_generation/simclusters_index_generation/README create mode 100644 src/scala/com/twitter/simclusters_v2/scio/bq_generation/simclusters_index_generation/engagement-event-based-simclusters-index-generation-job.d6w create mode 100644 src/scala/com/twitter/simclusters_v2/scio/bq_generation/sql/BUILD create mode 100644 src/scala/com/twitter/simclusters_v2/scio/bq_generation/sql/ads_user_tweet_action_pair_generation.sql create mode 100644 src/scala/com/twitter/simclusters_v2/scio/bq_generation/sql/cluster_top_tweets.sql create mode 100644 src/scala/com/twitter/simclusters_v2/scio/bq_generation/sql/cluster_top_tweets_intersection_with_fav_based_index.sql create mode 100644 src/scala/com/twitter/simclusters_v2/scio/bq_generation/sql/combined_user_tweet_action_pair_generation.sql create mode 100644 src/scala/com/twitter/simclusters_v2/scio/bq_generation/sql/engagement_based_index_generation.sql create mode 100644 src/scala/com/twitter/simclusters_v2/scio/bq_generation/sql/evergreen_content_user_tweet_action_pair_generation.sql create mode 100644 src/scala/com/twitter/simclusters_v2/scio/bq_generation/sql/nsfw_tweet_denylist.sql create mode 100644 src/scala/com/twitter/simclusters_v2/scio/bq_generation/sql/tweet_embeddings_generation.sql create mode 100644 src/scala/com/twitter/simclusters_v2/scio/bq_generation/sql/tweet_fav_count.sql create mode 100644 src/scala/com/twitter/simclusters_v2/scio/bq_generation/sql/tweets_ann.sql create mode 100644 src/scala/com/twitter/simclusters_v2/scio/bq_generation/sql/unified_user_tweet_action_pair_generation.sql create mode 100644 src/scala/com/twitter/simclusters_v2/scio/bq_generation/sql/user_video_tweet_fav_engagement_generation.sql create mode 100644 src/scala/com/twitter/simclusters_v2/scio/bq_generation/tweets_ann/BUILD create mode 100644 src/scala/com/twitter/simclusters_v2/scio/bq_generation/tweets_ann/Config.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scio/bq_generation/tweets_ann/README create mode 100644 src/scala/com/twitter/simclusters_v2/scio/bq_generation/tweets_ann/TweetsANNFromBQ.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scio/bq_generation/tweets_ann/TweetsANNJob.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scio/bq_generation/tweets_ann/iikf-hl-0-el-15-tweets-ann-batch-job.d6w create mode 100644 src/scala/com/twitter/simclusters_v2/scio/bq_generation/tweets_ann/iikf-hl-2-el-15-tweets-ann-batch-job.d6w create mode 100644 src/scala/com/twitter/simclusters_v2/scio/bq_generation/tweets_ann/iikf-hl-2-el-50-tweets-ann-batch-job.d6w create mode 100644 src/scala/com/twitter/simclusters_v2/scio/bq_generation/tweets_ann/iikf-hl-8-el-50-tweets-ann-adhoc-job.d6w create mode 100644 src/scala/com/twitter/simclusters_v2/scio/bq_generation/tweets_ann/iikf-hl-8-el-50-tweets-ann-batch-job.d6w create mode 100644 src/scala/com/twitter/simclusters_v2/scio/bq_generation/tweets_ann/iikf-tweets-ann-adhoc-job.d6w create mode 100644 src/scala/com/twitter/simclusters_v2/scio/bq_generation/tweets_ann/iikf-tweets-ann-batch-job.d6w create mode 100644 src/scala/com/twitter/simclusters_v2/scio/bq_generation/tweets_ann/mts-consumer-embeddings-tweets-ann-adhoc-job.d6w create mode 100644 src/scala/com/twitter/simclusters_v2/scio/bq_generation/tweets_ann/mts-consumer-embeddings-tweets-ann-batch-job.d6w create mode 100644 src/scala/com/twitter/simclusters_v2/scio/common/BUILD create mode 100644 src/scala/com/twitter/simclusters_v2/scio/common/ExternalDataSources.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scio/multi_type_graph/assemble_multi_type_graph/AssembleMultiTypeGraphScioApp.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scio/multi_type_graph/assemble_multi_type_graph/AssembleMultiTypeGraphScioBaseApp.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scio/multi_type_graph/assemble_multi_type_graph/BUILD create mode 100644 src/scala/com/twitter/simclusters_v2/scio/multi_type_graph/assemble_multi_type_graph/Config.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scio/multi_type_graph/assemble_multi_type_graph/README.md create mode 100644 src/scala/com/twitter/simclusters_v2/scio/multi_type_graph/assemble_multi_type_graph/assemble-multi-type-graph-scio-adhoc.d6w create mode 100644 src/scala/com/twitter/simclusters_v2/scio/multi_type_graph/assemble_multi_type_graph/assemble-multi-type-graph-scio-batch.d6w create mode 100644 src/scala/com/twitter/simclusters_v2/scio/multi_type_graph/common/BUILD create mode 100644 src/scala/com/twitter/simclusters_v2/scio/multi_type_graph/common/MultiTypeGraphUtil.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scio/multi_type_graph/multi_type_graph_sims/BUILD create mode 100644 src/scala/com/twitter/simclusters_v2/scio/multi_type_graph/multi_type_graph_sims/Config.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scio/multi_type_graph/multi_type_graph_sims/RightNodeCosineSimilarityScioApp.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scio/multi_type_graph/multi_type_graph_sims/RightNodeCosineSimilarityScioBaseApp.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scio/multi_type_graph/multi_type_graph_sims/RightNodeSimHashScioApp.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scio/multi_type_graph/multi_type_graph_sims/RightNodeSimHashScioBaseApp.scala create mode 100644 src/scala/com/twitter/simclusters_v2/scio/multi_type_graph/multi_type_graph_sims/cosine-similarity-scio-adhoc.d6w create mode 100644 src/scala/com/twitter/simclusters_v2/scio/multi_type_graph/multi_type_graph_sims/cosine-similarity-scio-batch.d6w create mode 100644 src/scala/com/twitter/simclusters_v2/scio/multi_type_graph/multi_type_graph_sims/sim-hash-scio-adhoc.d6w create mode 100644 src/scala/com/twitter/simclusters_v2/scio/multi_type_graph/multi_type_graph_sims/sim-hash-scio-batch.d6w create mode 100644 src/scala/com/twitter/simclusters_v2/score/AggregatedScoreStore.scala create mode 100644 src/scala/com/twitter/simclusters_v2/score/BUILD create mode 100644 src/scala/com/twitter/simclusters_v2/score/Score.scala create mode 100644 src/scala/com/twitter/simclusters_v2/score/ScoreFacadeStore.scala create mode 100644 src/scala/com/twitter/simclusters_v2/score/ScoreId.scala create mode 100644 src/scala/com/twitter/simclusters_v2/score/ScoreStore.scala create mode 100644 src/scala/com/twitter/simclusters_v2/score/SimClustersEmbeddingPairScoreStore.scala create mode 100644 src/scala/com/twitter/simclusters_v2/score/WeightedSumAggregatedScoreStore.scala create mode 100644 src/scala/com/twitter/simclusters_v2/stores/BUILD create mode 100644 src/scala/com/twitter/simclusters_v2/stores/LanguageFilteredLocaleEntityEmbeddingStore.scala create mode 100644 src/scala/com/twitter/simclusters_v2/stores/MultiTypeGraphStore.scala create mode 100644 src/scala/com/twitter/simclusters_v2/stores/SimClustersEmbeddingStore.scala create mode 100644 src/scala/com/twitter/simclusters_v2/stores/SimClustersMultiEmbeddingStore.scala create mode 100644 src/scala/com/twitter/simclusters_v2/stores/TopicTopProducersStore.scala create mode 100644 src/scala/com/twitter/simclusters_v2/stores/WtfMbcgStore.scala create mode 100644 src/scala/com/twitter/simclusters_v2/summingbird/BUILD create mode 100644 src/scala/com/twitter/simclusters_v2/summingbird/README.md create mode 100644 src/scala/com/twitter/simclusters_v2/summingbird/common/BUILD create mode 100644 src/scala/com/twitter/simclusters_v2/summingbird/common/ClientConfigs.scala create mode 100644 src/scala/com/twitter/simclusters_v2/summingbird/common/Configs.scala create mode 100644 src/scala/com/twitter/simclusters_v2/summingbird/common/EntityUtil.scala create mode 100644 src/scala/com/twitter/simclusters_v2/summingbird/common/Implicits.scala create mode 100644 src/scala/com/twitter/simclusters_v2/summingbird/common/ModelVersionProfile.scala create mode 100644 src/scala/com/twitter/simclusters_v2/summingbird/common/Monoids.scala create mode 100644 src/scala/com/twitter/simclusters_v2/summingbird/common/SimClustersEmbeddingWithMetadataMonoid.scala create mode 100644 src/scala/com/twitter/simclusters_v2/summingbird/common/SimClustersHashUtil.scala create mode 100644 src/scala/com/twitter/simclusters_v2/summingbird/common/SimClustersInterestedInUtil.scala create mode 100644 src/scala/com/twitter/simclusters_v2/summingbird/common/SimClustersProfile.scala create mode 100644 src/scala/com/twitter/simclusters_v2/summingbird/common/StatsUtil.scala create mode 100644 src/scala/com/twitter/simclusters_v2/summingbird/common/SummerWithSumValues.scala create mode 100644 src/scala/com/twitter/simclusters_v2/summingbird/common/ThriftDecayedValueMonoid.scala create mode 100644 src/scala/com/twitter/simclusters_v2/summingbird/common/TweetEntityExtractor.scala create mode 100644 src/scala/com/twitter/simclusters_v2/summingbird/stores/ApeTopicEmbeddingStore.scala create mode 100644 src/scala/com/twitter/simclusters_v2/summingbird/stores/BUILD create mode 100644 src/scala/com/twitter/simclusters_v2/summingbird/stores/ClusterDetailsReadableStore.scala create mode 100644 src/scala/com/twitter/simclusters_v2/summingbird/stores/EntityClusterScoreReadableStore.scala create mode 100644 src/scala/com/twitter/simclusters_v2/summingbird/stores/ManhattanFromStratoStore.scala create mode 100644 src/scala/com/twitter/simclusters_v2/summingbird/stores/PersistentTweetEmbeddingStore.scala create mode 100644 src/scala/com/twitter/simclusters_v2/summingbird/stores/ProducerClusterEmbeddingReadableStores.scala create mode 100644 src/scala/com/twitter/simclusters_v2/summingbird/stores/SemanticCoreEntityEmbeddingStore.scala create mode 100644 src/scala/com/twitter/simclusters_v2/summingbird/stores/SimClustersManhattanReadableStoreForReadWriteDataset.scala create mode 100644 src/scala/com/twitter/simclusters_v2/summingbird/stores/TfgTopicEmbeddingsStore.scala create mode 100644 src/scala/com/twitter/simclusters_v2/summingbird/stores/TopKClustersForEntityReadableStore.scala create mode 100644 src/scala/com/twitter/simclusters_v2/summingbird/stores/TopKClustersForTweetReadableStore.scala create mode 100644 src/scala/com/twitter/simclusters_v2/summingbird/stores/TopKTweetsForClusterReadableStore.scala create mode 100644 src/scala/com/twitter/simclusters_v2/summingbird/stores/TweetStatusCountsStore.scala create mode 100644 src/scala/com/twitter/simclusters_v2/summingbird/stores/UserInterestedInReadableStore.scala create mode 100644 src/scala/com/twitter/simclusters_v2/summingbird/stores/UserKnownForReadableStore.scala create mode 100644 src/scala/com/twitter/simclusters_v2/summingbird/storm/BUILD create mode 100644 src/scala/com/twitter/simclusters_v2/summingbird/storm/PersistentTweetJob.scala create mode 100644 src/scala/com/twitter/simclusters_v2/summingbird/storm/PersistentTweetJobRunner.scala create mode 100644 src/scala/com/twitter/simclusters_v2/summingbird/storm/TweetJob.scala create mode 100644 src/scala/com/twitter/simclusters_v2/summingbird/storm/TweetJobRunner.scala create mode 100755 src/scala/com/twitter/simclusters_v2/summingbird/storm/persistent_tweet_job_deploy.sh create mode 100755 src/scala/com/twitter/simclusters_v2/summingbird/storm/tweet_alt_job_deploy.sh create mode 100755 src/scala/com/twitter/simclusters_v2/summingbird/storm/tweet_job_deploy.sh create mode 100644 src/scala/com/twitter/simclusters_v2/tweet_similarity/BUILD create mode 100644 src/scala/com/twitter/simclusters_v2/tweet_similarity/ModelBasedTweetSimilaritySimClustersEmbeddingAdapter.scala create mode 100644 src/scala/com/twitter/simclusters_v2/tweet_similarity/TweetSimilarityFeatures.scala create mode 100644 src/thrift/com/twitter/interaction_graph/BUILD create mode 100644 src/thrift/com/twitter/interaction_graph/interaction_graph.thrift create mode 100644 src/thrift/com/twitter/recos/recos.thrift create mode 100644 src/thrift/com/twitter/recos/recos_common.thrift create mode 100644 src/thrift/com/twitter/recos/recos_injector.thrift create mode 100644 src/thrift/com/twitter/recos/user_tweet_entity_graph/BUILD create mode 100644 src/thrift/com/twitter/recos/user_tweet_entity_graph/CONFIG.ini create mode 100644 src/thrift/com/twitter/recos/user_tweet_entity_graph/user_tweet_entity_graph.thrift create mode 100644 src/thrift/com/twitter/recos/user_tweet_graph/BUILD create mode 100644 src/thrift/com/twitter/recos/user_tweet_graph/CONFIG.ini create mode 100644 src/thrift/com/twitter/recos/user_tweet_graph/user_tweet_graph.thrift create mode 100644 src/thrift/com/twitter/recos/user_user_graph/BUILD create mode 100644 src/thrift/com/twitter/recos/user_user_graph/CONFIG.ini create mode 100644 src/thrift/com/twitter/recos/user_user_graph/user_user_graph.thrift create mode 100644 src/thrift/com/twitter/recos/user_video_graph/BUILD create mode 100644 src/thrift/com/twitter/recos/user_video_graph/CONFIG.ini create mode 100644 src/thrift/com/twitter/recos/user_video_graph/user_video_graph.thrift create mode 100644 src/thrift/com/twitter/search/common/ranking/ranking.thrift create mode 100644 src/thrift/com/twitter/search/earlybird/thrift/earlybird.thrift create mode 100644 src/thrift/com/twitter/simclusters_v2/BUILD create mode 100644 src/thrift/com/twitter/simclusters_v2/abuse.thrift create mode 100644 src/thrift/com/twitter/simclusters_v2/clustering.thrift create mode 100644 src/thrift/com/twitter/simclusters_v2/embedding.thrift create mode 100644 src/thrift/com/twitter/simclusters_v2/entity.thrift create mode 100644 src/thrift/com/twitter/simclusters_v2/evaluation.thrift create mode 100644 src/thrift/com/twitter/simclusters_v2/graph.thrift create mode 100644 src/thrift/com/twitter/simclusters_v2/identifier.thrift create mode 100644 src/thrift/com/twitter/simclusters_v2/inferred_entities.thrift create mode 100644 src/thrift/com/twitter/simclusters_v2/interests.thrift create mode 100644 src/thrift/com/twitter/simclusters_v2/multi_type_graph.thrift create mode 100644 src/thrift/com/twitter/simclusters_v2/offline_job_internal.thrift create mode 100644 src/thrift/com/twitter/simclusters_v2/online_store.thrift create mode 100644 src/thrift/com/twitter/simclusters_v2/online_store_internal.thrift create mode 100644 src/thrift/com/twitter/simclusters_v2/score.thrift create mode 100644 src/thrift/com/twitter/simclusters_v2/simclusters_presto.thrift create mode 100644 src/thrift/com/twitter/simclusters_v2/top_k_map.thrift create mode 100644 src/thrift/com/twitter/simclusters_v2/tweet_similarity.thrift create mode 100644 timelineranker/README.md create mode 100644 timelineranker/client/builder/BUILD create mode 100644 timelineranker/client/builder/README.md create mode 100644 timelineranker/client/builder/src/main/scala/BUILD create mode 100644 timelineranker/client/builder/src/main/scala/com/twitter/timelineranker/client/TimelineRankerClient.scala create mode 100644 timelineranker/client/builder/src/main/scala/com/twitter/timelineranker/client/TimelineRankerClientBuilder.scala create mode 100644 timelineranker/common/BUILD create mode 100644 timelineranker/common/src/main/scala/BUILD create mode 100644 timelineranker/common/src/main/scala/com/twitter/timelineranker/adapter/BUILD create mode 100644 timelineranker/common/src/main/scala/com/twitter/timelineranker/adapter/TimelineServiceAdapter.scala create mode 100644 timelineranker/common/src/main/scala/com/twitter/timelineranker/model/BUILD create mode 100644 timelineranker/common/src/main/scala/com/twitter/timelineranker/model/CandidateTweet.scala create mode 100644 timelineranker/common/src/main/scala/com/twitter/timelineranker/model/CandidateTweetsResult.scala create mode 100644 timelineranker/common/src/main/scala/com/twitter/timelineranker/model/HydratedTweetEntry.scala create mode 100644 timelineranker/common/src/main/scala/com/twitter/timelineranker/model/Language.scala create mode 100644 timelineranker/common/src/main/scala/com/twitter/timelineranker/model/LanguageScope.scala create mode 100644 timelineranker/common/src/main/scala/com/twitter/timelineranker/model/PartiallyHydratedTweet.scala create mode 100644 timelineranker/common/src/main/scala/com/twitter/timelineranker/model/PriorSeenEntries.scala create mode 100644 timelineranker/common/src/main/scala/com/twitter/timelineranker/model/RankedTimelineQuery.scala create mode 100644 timelineranker/common/src/main/scala/com/twitter/timelineranker/model/RankedTimelineQueryOptions.scala create mode 100644 timelineranker/common/src/main/scala/com/twitter/timelineranker/model/RecapQuery.scala create mode 100644 timelineranker/common/src/main/scala/com/twitter/timelineranker/model/ReverseChronTimelineQuery.scala create mode 100644 timelineranker/common/src/main/scala/com/twitter/timelineranker/model/ReverseChronTimelineQueryOptions.scala create mode 100644 timelineranker/common/src/main/scala/com/twitter/timelineranker/model/TimeRange.scala create mode 100644 timelineranker/common/src/main/scala/com/twitter/timelineranker/model/Timeline.scala create mode 100644 timelineranker/common/src/main/scala/com/twitter/timelineranker/model/TimelineEntry.scala create mode 100644 timelineranker/common/src/main/scala/com/twitter/timelineranker/model/TimelineEntryEnvelope.scala create mode 100644 timelineranker/common/src/main/scala/com/twitter/timelineranker/model/TimelineQuery.scala create mode 100644 timelineranker/common/src/main/scala/com/twitter/timelineranker/model/TimelineQueryOptions.scala create mode 100644 timelineranker/common/src/main/scala/com/twitter/timelineranker/model/TimelineRange.scala create mode 100644 timelineranker/common/src/main/scala/com/twitter/timelineranker/model/Tweet.scala create mode 100644 timelineranker/common/src/main/scala/com/twitter/timelineranker/model/TweetIdRange.scala create mode 100644 timelineranker/common/src/main/scala/com/twitter/timelineranker/model/UtegLikedByTweetsOptions.scala create mode 100644 timelineranker/server/BUILD.bazel create mode 100644 timelineranker/server/config/BUILD create mode 100644 timelineranker/server/config/decider.yml create mode 100644 timelineranker/server/src/main/resources/BUILD.bazel create mode 100644 timelineranker/server/src/main/resources/logback-timelineranker.xml create mode 100644 timelineranker/server/src/main/scala/BUILD.bazel create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/clients/BUILD create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/clients/CortexTweetQueryServiceClient.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/clients/MemcacheFactory.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/clients/content_features_cache/BUILD create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/clients/content_features_cache/ContentFeaturesMemcacheBuilder.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/common/BUILD create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/common/CandidateGenerationTransform.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/common/ContentFeaturesHydrationTransform.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/common/CreateCandidateEnvelopeTransform.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/common/FeatureHydrationDataTransform.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/common/FollowAndRealGraphCombiningTransform.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/common/FollowGraphDataTransform.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/common/HydrateTweetsAndSourceTweetsInParallelTransform.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/common/HydratedTweetsFilterTransform.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/common/InNetworkTweetsSearchFeaturesHydrationTransform.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/common/MarkRandomTweetTransform.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/common/OutOfNetworkRepliesToUserIdSearchResultsTransform.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/common/OutOfNetworkTweetsSearchFeaturesHydrationTransform.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/common/RecapHydrationSearchResultsTransformBase.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/common/RecapSearchResultsTransform.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/common/RecapSearchResultsTruncationTransform.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/common/SearchResultDedupAndSortingTransform.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/common/SourceTweetsSearchResultsTransform.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/common/TrimToMatchHydratedTweetsTransform.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/common/TrimToMatchSearchResultsTransform.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/common/TweetHydrationTransform.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/common/TweetKindOptionHydratedTweetsFilterTransform.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/common/UserLanguagesTransform.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/common/UserProfileInfoTransform.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/common/VisibilityEnforcingTransform.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/config/BUILD create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/config/CallInfo.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/config/ClientAccessPermissions.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/config/ClientWrapperFactories.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/config/ClientWrappers.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/config/DefaultUnderlyingClientConfiguration.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/config/RequestScopes.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/config/RuntimeConfiguration.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/config/StagingUnderlyingConfiguration.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/config/TimelineRankerConstants.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/config/TimelineRankerFlags.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/config/UnderlyingClientConfiguration.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/contentfeatures/BUILD create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/contentfeatures/package.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/core/BUILD create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/core/CandidateEnvelope.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/core/FollowGraphData.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/core/FollowGraphDataFuture.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/core/HydratedCandidatesAndFeaturesEnvelope.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/core/HydratedTweets.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/core/package.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/decider/BUILD create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/decider/DeciderKey.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/entity_tweets/BUILD.bazel create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/entity_tweets/EntityTweetsRepository.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/entity_tweets/EntityTweetsRepositoryBuilder.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/entity_tweets/EntityTweetsSearchResultsTransform.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/entity_tweets/EntityTweetsSource.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/in_network_tweets/BUILD create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/in_network_tweets/InNetworkTweetRepository.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/in_network_tweets/InNetworkTweetRepositoryBuilder.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/in_network_tweets/InNetworkTweetSource.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/monitoring/BUILD create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/monitoring/UsersSearchResultMonitoringTransform.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/observe/BUILD.bazel create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/observe/DebugObserverBuilder.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/observe/ObservedRequests.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/parameters/BUILD create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/parameters/ConfigBuilder.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/parameters/entity_tweets/BUILD create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/parameters/entity_tweets/EntityTweetsParams.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/parameters/entity_tweets/EntityTweetsProduction.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/parameters/in_network_tweets/BUILD create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/parameters/in_network_tweets/InNetworkTweetParams.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/parameters/in_network_tweets/InNetworkTweetProduction.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/parameters/monitoring/BUILD create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/parameters/monitoring/MonitoringParams.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/parameters/monitoring/MonitoringProduction.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/parameters/recap/BUILD create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/parameters/recap/RecapParams.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/parameters/recap/RecapProduction.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/parameters/recap/RecapQueryContext.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/parameters/recap_author/BUILD create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/parameters/recap_author/RecapAuthorParams.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/parameters/recap_author/RecapAuthorProduction.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/parameters/recap_hydration/BUILD create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/parameters/recap_hydration/RecapHydrationParams.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/parameters/recap_hydration/RecapHydrationProduction.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/parameters/revchron/BUILD create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/parameters/revchron/ReverseChronParams.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/parameters/revchron/ReverseChronProduction.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/parameters/revchron/ReverseChronTimelineQueryContext.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/parameters/revchron/ReverseChronTimelineQueryContextBuilder.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/parameters/uteg_liked_by_tweets/BUILD create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/parameters/uteg_liked_by_tweets/UtegLikedByTweetsParams.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/parameters/uteg_liked_by_tweets/UtegLikedByTweetsProduction.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/parameters/util/BUILD create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/parameters/util/CommonRequestContext.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/parameters/util/ConfigHelper.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/parameters/util/RecapQueryParamInitializer.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/recap/BUILD create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/recap/model/BUILD create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/recap/model/ContentFeatures.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/recap_author/BUILD.bazel create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/recap_author/RecapAuthorRepository.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/recap_author/RecapAuthorRepositoryBuilder.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/recap_author/RecapAuthorSearchResultsTransform.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/recap_author/RecapAuthorSource.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/recap_hydration/BUILD.bazel create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/recap_hydration/RecapHydrationRepository.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/recap_hydration/RecapHydrationRepositoryBuilder.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/recap_hydration/RecapHydrationSearchResultsTransform.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/recap_hydration/RecapHydrationSource.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/repository/BUILD create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/repository/CandidatesRepositoryBuilder.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/repository/RankedHomeTimelineRepository.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/repository/RepositoryBuilder.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/repository/ReverseChronHomeTimelineRepository.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/repository/ReverseChronHomeTimelineRepositoryBuilder.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/repository/RoutingTimelineRepository.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/repository/RoutingTimelineRepositoryBuilder.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/repository/TimelineRepository.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/server/BUILD.bazel create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/server/Main.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/server/TimelineRanker.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/server/TimelineRankerBuilder.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/server/TimelineRankerThriftWebForms.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/server/Warmup.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/source/BUILD create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/source/ReverseChronHomeTimelineSource.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/source/TimelineSource.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/uteg_liked_by_tweets/BUILD.bazel create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/uteg_liked_by_tweets/CombinedScoreAndTruncateTransform.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/uteg_liked_by_tweets/MinNumNonAuthorFavoritedByUserIdsFilterTransform.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/uteg_liked_by_tweets/RemoveCandidatesAuthoredByWeightedFollowingsTransform.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/uteg_liked_by_tweets/SocialProofAndUTEGScoreHydrationTransform.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/uteg_liked_by_tweets/UTEGResultsTransform.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/uteg_liked_by_tweets/UtegLikedByTweetsRepository.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/uteg_liked_by_tweets/UtegLikedByTweetsRepositoryBuilder.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/uteg_liked_by_tweets/UtegLikedByTweetsSearchResultsTransform.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/uteg_liked_by_tweets/UtegLikedByTweetsSource.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/util/BUILD create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/util/CachingContentFeaturesProvider.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/util/CopyContentFeaturesIntoHydratedTweetsTransform.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/util/CopyContentFeaturesIntoThriftTweetFeaturesTransform.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/util/ExtendedRepliesFilter.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/util/LatentRepository.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/util/RecommendedRepliesFilter.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/util/ReverseExtendedRepliesFilter.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/util/SearchResultUtil.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/util/SearchResultWithVisibilityActors.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/util/SnowflakeUtils.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/util/SourceTweetsUtil.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/util/TweetAnnotationFeaturesExtractor.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/util/TweetHydrator.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/util/TweetMediaFeatureExtractor.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/util/TweetTextFeaturesExtractor.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/util/TweetsPostFilter.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/util/TweetsPostFilterBasedOnSearchMetadata.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/util/TweetypieContentFeaturesProvider.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/visibility/BUILD create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/visibility/FollowGraphDataProvider.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/visibility/RealGraphFollowGraphDataProvider.scala create mode 100644 timelineranker/server/src/main/scala/com/twitter/timelineranker/visibility/SgsFollowGraphDataProvider.scala create mode 100644 timelines/data_processing/ad_hoc/earlybird_ranking/earlybird_ranking/BUILD create mode 100644 timelines/data_processing/ad_hoc/earlybird_ranking/earlybird_ranking/common/BUILD create mode 100644 timelines/data_processing/ad_hoc/earlybird_ranking/earlybird_ranking/common/EarlybirdTrainingConfiguration.scala create mode 100644 timelines/data_processing/ad_hoc/earlybird_ranking/earlybird_ranking/common/EarlybirdTrainingRecapConfiguration.scala create mode 100644 timelines/data_processing/ad_hoc/earlybird_ranking/earlybird_ranking/common/EarlybirdTrainingRectweetConfiguration.scala create mode 100644 timelines/data_processing/ad_hoc/earlybird_ranking/earlybird_ranking/model_evaluation/BUILD create mode 100644 timelines/data_processing/ad_hoc/earlybird_ranking/earlybird_ranking/model_evaluation/EarlybirdEvaluationMetric.scala create mode 100644 timelines/data_processing/ad_hoc/earlybird_ranking/earlybird_ranking/model_evaluation/EarlybirdModelEvaluationJob.scala create mode 100644 timelines/data_processing/ad_hoc/earlybird_ranking/earlybird_ranking/training_data_generation/BUILD create mode 100644 timelines/data_processing/ad_hoc/earlybird_ranking/earlybird_ranking/training_data_generation/EarlybirdExampleSampler.scala create mode 100644 timelines/data_processing/ad_hoc/earlybird_ranking/earlybird_ranking/training_data_generation/EarlybirdStatsJob.scala create mode 100644 timelines/data_processing/ad_hoc/earlybird_ranking/earlybird_ranking/training_data_generation/EarlybirdTrainingDataJob.scala create mode 100644 trust_and_safety_models/README.md create mode 100644 trust_and_safety_models/abusive/abusive_model.py create mode 100644 trust_and_safety_models/nsfw/nsfw_media.py create mode 100644 trust_and_safety_models/nsfw/nsfw_text.py create mode 100644 trust_and_safety_models/toxicity/__init__.py create mode 100644 trust_and_safety_models/toxicity/data/__init__.py create mode 100644 trust_and_safety_models/toxicity/data/data_preprocessing.py create mode 100644 trust_and_safety_models/toxicity/data/dataframe_loader.py create mode 100644 trust_and_safety_models/toxicity/data/mb_generator.py create mode 100644 trust_and_safety_models/toxicity/load_model.py create mode 100644 trust_and_safety_models/toxicity/optim/__init__.py create mode 100644 trust_and_safety_models/toxicity/optim/callbacks.py create mode 100644 trust_and_safety_models/toxicity/optim/losses.py create mode 100644 trust_and_safety_models/toxicity/optim/schedulers.py create mode 100644 trust_and_safety_models/toxicity/rescoring.py create mode 100644 trust_and_safety_models/toxicity/settings/__init__.py create mode 100644 trust_and_safety_models/toxicity/settings/default_settings_tox.py create mode 100644 trust_and_safety_models/toxicity/train.py create mode 100644 trust_and_safety_models/toxicity/utils/__init__.py create mode 100644 trust_and_safety_models/toxicity/utils/helpers.py create mode 100644 twml/BUILD create mode 100644 twml/README.md create mode 100644 twml/libtwml/BUILD create mode 100644 twml/libtwml/include/twml.h create mode 100644 twml/libtwml/include/twml/BatchPredictionRequest.h create mode 100644 twml/libtwml/include/twml/BatchPredictionResponse.h create mode 100644 twml/libtwml/include/twml/BlockFormatReader.h create mode 100644 twml/libtwml/include/twml/BlockFormatWriter.h create mode 100644 twml/libtwml/include/twml/DataRecord.h create mode 100644 twml/libtwml/include/twml/DataRecordReader.h create mode 100644 twml/libtwml/include/twml/DataRecordWriter.h create mode 100644 twml/libtwml/include/twml/Error.h create mode 100644 twml/libtwml/include/twml/HashedDataRecord.h create mode 100644 twml/libtwml/include/twml/HashedDataRecordReader.h create mode 100644 twml/libtwml/include/twml/Hashmap.h create mode 100644 twml/libtwml/include/twml/RawTensor.h create mode 100644 twml/libtwml/include/twml/Tensor.h create mode 100644 twml/libtwml/include/twml/TensorRecord.h create mode 100644 twml/libtwml/include/twml/TensorRecordReader.h create mode 100644 twml/libtwml/include/twml/TensorRecordWriter.h create mode 100644 twml/libtwml/include/twml/ThriftReader.h create mode 100644 twml/libtwml/include/twml/ThriftWriter.h create mode 100644 twml/libtwml/include/twml/Type.h create mode 100644 twml/libtwml/include/twml/common.h create mode 100644 twml/libtwml/include/twml/defines.h create mode 100644 twml/libtwml/include/twml/discretizer_impl.h create mode 100644 twml/libtwml/include/twml/functions.h create mode 100644 twml/libtwml/include/twml/hashing_discretizer_impl.h create mode 100644 twml/libtwml/include/twml/io/IOError.h create mode 100644 twml/libtwml/include/twml/optim.h create mode 100644 twml/libtwml/include/twml/utilities.h create mode 100644 twml/libtwml/setup.cfg create mode 100644 twml/libtwml/setup.py create mode 100644 twml/libtwml/src/lib/BatchPredictionRequest.cpp create mode 100644 twml/libtwml/src/lib/BatchPredictionResponse.cpp create mode 100644 twml/libtwml/src/lib/BlockFormatReader.cpp create mode 100644 twml/libtwml/src/lib/BlockFormatWriter.cpp create mode 100644 twml/libtwml/src/lib/CMakeLists.txt create mode 100644 twml/libtwml/src/lib/CPPLINT.cfg create mode 100644 twml/libtwml/src/lib/DataRecord.cpp create mode 100644 twml/libtwml/src/lib/DataRecordReader.cpp create mode 100644 twml/libtwml/src/lib/DataRecordWriter.cpp create mode 100644 twml/libtwml/src/lib/HashedDataRecord.cpp create mode 100644 twml/libtwml/src/lib/HashedDataRecordReader.cpp create mode 100644 twml/libtwml/src/lib/Hashmap.cpp create mode 100644 twml/libtwml/src/lib/Tensor.cpp create mode 100644 twml/libtwml/src/lib/TensorRecordReader.cpp create mode 100644 twml/libtwml/src/lib/TensorRecordWriter.cpp create mode 100644 twml/libtwml/src/lib/ThriftReader.cpp create mode 100644 twml/libtwml/src/lib/ThriftWriter.cpp create mode 100644 twml/libtwml/src/lib/discretizer_impl.cpp create mode 100644 twml/libtwml/src/lib/functions.cpp create mode 100644 twml/libtwml/src/lib/hashing_discretizer_impl.cpp create mode 100644 twml/libtwml/src/lib/internal/endianutils.h create mode 100644 twml/libtwml/src/lib/internal/error.h create mode 100644 twml/libtwml/src/lib/internal/interpolate.h create mode 100644 twml/libtwml/src/lib/internal/khash.h create mode 100644 twml/libtwml/src/lib/internal/linear_search.h create mode 100644 twml/libtwml/src/lib/internal/murmur_hash3.h create mode 100644 twml/libtwml/src/lib/internal/thrift.h create mode 100644 twml/libtwml/src/lib/internal/utf_converter.h create mode 100644 twml/libtwml/src/lib/io/IOError.cpp create mode 100644 twml/libtwml/src/lib/murmur_hash3.cpp create mode 100644 twml/libtwml/src/lib/optim.cpp create mode 100644 twml/libtwml/src/lib/utf_converter.cpp create mode 100644 twml/libtwml/src/ops/CMakeLists.txt create mode 100644 twml/libtwml/src/ops/add1.cpp create mode 100644 twml/libtwml/src/ops/batch_prediction_request.cpp create mode 100644 twml/libtwml/src/ops/batch_prediction_request_v2.cpp create mode 100644 twml/libtwml/src/ops/batch_prediction_response_writer.cpp create mode 100644 twml/libtwml/src/ops/batch_prediction_tensor_response_writer.cpp create mode 100644 twml/libtwml/src/ops/binary_sparse_dense_matmul.cpp create mode 100644 twml/libtwml/src/ops/binary_sparse_dense_matmul.h create mode 100644 twml/libtwml/src/ops/binary_sparse_dense_matmul_impl.h create mode 100644 twml/libtwml/src/ops/block_format_dataset.cpp create mode 100644 twml/libtwml/src/ops/block_format_reader.h create mode 100644 twml/libtwml/src/ops/compress_sample_ids.cpp create mode 100644 twml/libtwml/src/ops/contrib/get_substrings.cpp create mode 100644 twml/libtwml/src/ops/data_record.cpp create mode 100644 twml/libtwml/src/ops/data_record_tensor_writer.cpp create mode 100644 twml/libtwml/src/ops/discretizer.cpp create mode 100644 twml/libtwml/src/ops/feature_extractor.cpp create mode 100644 twml/libtwml/src/ops/feature_id.cpp create mode 100644 twml/libtwml/src/ops/feature_mask.cpp create mode 100644 twml/libtwml/src/ops/fixed_length_tensor.cpp create mode 100644 twml/libtwml/src/ops/hashed_data_record.cpp create mode 100644 twml/libtwml/src/ops/hashing_discretizer.cpp create mode 100644 twml/libtwml/src/ops/hashmap.cpp create mode 100644 twml/libtwml/src/ops/isotonic_calibration.cpp create mode 100644 twml/libtwml/src/ops/num_intra_op_threads.cpp create mode 100644 twml/libtwml/src/ops/par_add.cpp create mode 100644 twml/libtwml/src/ops/partition_sparse_tensor.cpp create mode 100644 twml/libtwml/src/ops/percentile_discretizer_v2.cpp create mode 100644 twml/libtwml/src/ops/resource_utils.h create mode 100644 twml/libtwml/src/ops/scripts/get_inc.py create mode 100755 twml/libtwml/src/ops/scripts/get_inc.sh create mode 100644 twml/libtwml/src/ops/scripts/get_lib.py create mode 100755 twml/libtwml/src/ops/scripts/get_lib.sh create mode 100755 twml/libtwml/src/ops/scripts/symlink.sh create mode 100644 twml/libtwml/src/ops/sleep_op.cpp create mode 100644 twml/libtwml/src/ops/sparse_normalization.cpp create mode 100644 twml/libtwml/src/ops/tensor_record.cpp create mode 100644 twml/libtwml/src/ops/tensorflow_utils.cpp create mode 100644 twml/libtwml/src/ops/tensorflow_utils.h create mode 100644 twml/libtwml/src/ops/var_length_reader.cpp create mode 100644 twml/setup.cfg create mode 100644 twml/setup.py create mode 100644 twml/twml/__init__.py create mode 100644 twml/twml/argument_parser.py create mode 100644 twml/twml/array.py create mode 100644 twml/twml/block_format_writer.py create mode 100644 twml/twml/constants.py create mode 100644 twml/twml/contrib/__init__.py create mode 100644 twml/twml/contrib/build_graphs_fns.py create mode 100644 twml/twml/contrib/calibrators/__init__.py create mode 100644 twml/twml/contrib/calibrators/calibrator.py create mode 100644 twml/twml/contrib/calibrators/common_calibrators.py create mode 100644 twml/twml/contrib/calibrators/hashed_percentile_discretizer.py create mode 100644 twml/twml/contrib/calibrators/hashing_discretizer.py create mode 100644 twml/twml/contrib/calibrators/isotonic.py create mode 100644 twml/twml/contrib/calibrators/mdl.py create mode 100644 twml/twml/contrib/calibrators/percentile_discretizer.py create mode 100644 twml/twml/contrib/eventbus/input_fn.py create mode 100644 twml/twml/contrib/eventbus/reader.py create mode 100644 twml/twml/contrib/export/__init__.py create mode 100644 twml/twml/contrib/export/export_fn.py create mode 100644 twml/twml/contrib/export/exporters.py create mode 100644 twml/twml/contrib/feature_config.py create mode 100644 twml/twml/contrib/feature_config_parsers.py create mode 100644 twml/twml/contrib/feature_importances/__init__.py create mode 100644 twml/twml/contrib/feature_importances/feature_importances.py create mode 100644 twml/twml/contrib/feature_importances/feature_permutation.py create mode 100644 twml/twml/contrib/feature_importances/helpers.py create mode 100644 twml/twml/contrib/hooks.py create mode 100644 twml/twml/contrib/initializers.py create mode 100644 twml/twml/contrib/layers/__init__.py create mode 100644 twml/twml/contrib/layers/embedding_lookup.py create mode 100644 twml/twml/contrib/layers/factorization_machine.py create mode 100644 twml/twml/contrib/layers/full_dense.py create mode 100644 twml/twml/contrib/layers/hashed_percentile_discretizer.py create mode 100644 twml/twml/contrib/layers/hashing_discretizer.py create mode 100644 twml/twml/contrib/layers/mask_layer.py create mode 100644 twml/twml/contrib/layers/stacked_rnn.py create mode 100644 twml/twml/contrib/layers/zscore_normalization.py create mode 100644 twml/twml/contrib/metrics/__init__.py create mode 100644 twml/twml/contrib/metrics/metrics.py create mode 100644 twml/twml/contrib/metrics/search_metrics.py create mode 100644 twml/twml/contrib/optimizers/__init__.py create mode 100644 twml/twml/contrib/optimizers/deep_gradient_compression_optimizer.py create mode 100644 twml/twml/contrib/optimizers/pruning_optimizer.py create mode 100644 twml/twml/contrib/parsers.py create mode 100644 twml/twml/contrib/pruning.py create mode 100644 twml/twml/contrib/readers/__init__.py create mode 100644 twml/twml/contrib/readers/batch_prediction_request.py create mode 100644 twml/twml/contrib/readers/data_record.py create mode 100644 twml/twml/contrib/readers/hashed_batch_prediction_request.py create mode 100644 twml/twml/contrib/trainers/__init__.py create mode 100644 twml/twml/contrib/trainers/batch_prediction_request_trainer.py create mode 100644 twml/twml/contrib/trainers/pruning_data_record_trainer.py create mode 100644 twml/twml/contrib/trainers/trainer_utils.py create mode 100644 twml/twml/contrib/utils/__init__.py create mode 100644 twml/twml/contrib/utils/datasets.py create mode 100644 twml/twml/contrib/utils/device.py create mode 100644 twml/twml/contrib/utils/interp.py create mode 100644 twml/twml/contrib/utils/loss_fns.py create mode 100644 twml/twml/contrib/utils/masks.py create mode 100644 twml/twml/contrib/utils/math_fns.py create mode 100644 twml/twml/contrib/utils/normalizer.py create mode 100644 twml/twml/contrib/utils/scores.py create mode 100644 twml/twml/contrib/utils/similarities.py create mode 100644 twml/twml/dataset.py create mode 100644 twml/twml/errors.py create mode 100644 twml/twml/export_output_fns.py create mode 100644 twml/twml/feature_config.py create mode 100644 twml/twml/filters.py create mode 100644 twml/twml/hooks.py create mode 100644 twml/twml/input_fns.py create mode 100644 twml/twml/layers/__init__.py create mode 100644 twml/twml/layers/batch_prediction_tensor_writer.py create mode 100644 twml/twml/layers/batch_prediction_writer.py create mode 100644 twml/twml/layers/data_record_tensor_writer.py create mode 100644 twml/twml/layers/full_dense.py create mode 100644 twml/twml/layers/full_sparse.py create mode 100644 twml/twml/layers/isotonic.py create mode 100644 twml/twml/layers/layer.py create mode 100644 twml/twml/layers/mdl.py create mode 100644 twml/twml/layers/partition.py create mode 100644 twml/twml/layers/percentile_discretizer.py create mode 100644 twml/twml/layers/sequential.py create mode 100644 twml/twml/layers/sparse_max_norm.py create mode 100644 twml/twml/layers/stitch.py create mode 100644 twml/twml/learning_rate_decay.py create mode 100644 twml/twml/lookup/__init__.py create mode 100644 twml/twml/metrics.py create mode 100644 twml/twml/optimizers/__init__.py create mode 100644 twml/twml/parsers.py create mode 100644 twml/twml/readers/__init__.py create mode 100644 twml/twml/readers/batch_prediction_request.py create mode 100644 twml/twml/readers/data_record.py create mode 100644 twml/twml/readers/hashed_batch_prediction_request.py create mode 100644 twml/twml/readers/hashed_data_record.py create mode 100644 twml/twml/saved_model_cli/__init__.py create mode 100644 twml/twml/saved_model_cli/__main__.py create mode 100644 twml/twml/summary/__init__.py create mode 100644 twml/twml/tensorboard/__init__.py create mode 100644 twml/twml/tensorboard/__main__.py create mode 100644 twml/twml/tensorio.py create mode 100644 twml/twml/tracking/__init__.py create mode 100644 twml/twml/tracking/experiment_tracker.py create mode 100644 twml/twml/trainers/__init__.py create mode 100644 twml/twml/trainers/data_record_trainer.py create mode 100644 twml/twml/trainers/trainer.py create mode 100644 twml/twml/util.py create mode 100644 twml/twml_common/__init__.py create mode 100644 twml/twml_common/initializer.py create mode 100644 twml/twml_common/serialize.py create mode 100644 twml/twml_common/sparse_inputs.py create mode 100644 visibilitylib/BUILD create mode 100644 visibilitylib/README.md create mode 100644 visibilitylib/src/main/resources/config/BUILD create mode 100644 visibilitylib/src/main/resources/config/com/twitter/visibility/decider.yml create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/BUILD create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/VisibilityLibrary.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/builder/BUILD create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/builder/FeatureMapBuilder.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/builder/VerdictLogger.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/builder/VisibilityResult.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/builder/VisibilityResultBuilder.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/builder/common/BUILD create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/builder/common/MutedKeywordFeatures.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/builder/dms/BUILD create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/builder/dms/DmConversationFeatures.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/builder/dms/DmEventFeatures.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/builder/media/BUILD create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/builder/media/MediaFeatures.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/builder/media/MediaMetadataFeatures.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/builder/spaces/BUILD create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/builder/spaces/SpaceFeatures.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/builder/tweets/BUILD create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/builder/tweets/BlenderContextFeatures.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/builder/tweets/CommunityNotificationFeatures.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/builder/tweets/CommunityTweetFeatures.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/builder/tweets/CommunityTweetFeaturesPartitioned.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/builder/tweets/CommunityTweetFeaturesV2.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/builder/tweets/ConversationControlFeatures.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/builder/tweets/EditTweetFeatures.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/builder/tweets/ExclusiveTweetFeatures.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/builder/tweets/FosnrPefetchedLabelsRelationshipFeatures.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/builder/tweets/FosnrRelationshipFeatures.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/builder/tweets/MisinformationPolicyFeatures.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/builder/tweets/ModerationFeatures.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/builder/tweets/SearchContextFeatures.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/builder/tweets/ToxicReplyFilterFeature.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/builder/tweets/TrustedFriendsFeatures.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/builder/tweets/TweetFeatures.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/builder/tweets/TweetIdFeatures.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/builder/tweets/TweetMediaMetadataFeatures.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/builder/tweets/TweetPerspectiveFeatures.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/builder/tweets/TweetVisibilityNudgeSourceWrapper.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/builder/tweets/UnmentionNotificationFeatures.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/builder/users/AuthorDeviceFeatures.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/builder/users/AuthorFeatures.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/builder/users/BUILD create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/builder/users/QuotedTweetFeatures.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/builder/users/RelationshipFeatures.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/builder/users/RelationshipVerbHelpers.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/builder/users/SearchFeatures.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/builder/users/UserUnavailableFeatures.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/builder/users/ViewerAdvancedFilteringFeatures.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/builder/users/ViewerFeatures.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/builder/users/ViewerSearchSafetyFeatures.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/builder/users/ViewerSensitiveMediaSettingsFeatures.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/configapi/BUILD create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/configapi/ConfigBuilder.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/configapi/VisibilityParams.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/configapi/VisibilityRequestContext.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/configapi/VisibilityRequestContextFactory.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/configapi/configs/BUILD create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/configapi/configs/DeciderKey.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/configapi/configs/ExperimentsHelper.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/configapi/configs/VisibilityDeciderGates.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/configapi/configs/VisibilityDeciders.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/configapi/configs/VisibilityExperimentsConfig.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/configapi/configs/VisibilityFeatureSwitches.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/configapi/configs/overrides/BUILD create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/configapi/configs/overrides/VisibilityLibraryDeciderOverrides.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/configapi/params/BUILD create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/configapi/params/FSRuleParams.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/configapi/params/GlobalParams.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/configapi/params/LabelSourceParams.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/configapi/params/RuleParams.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/configapi/params/SafetyLevelParams.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/configapi/params/TimelineConversationsDownrankingSpecificParams.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/configapi/params/VisibilityExperiment.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/configapi/params/VisibilityExperiments.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/engine/BUILD create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/engine/DeciderableVisibilityRuleEngine.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/engine/VisibilityResultsMetricRecorder.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/engine/VisibilityRuleEngine.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/engine/VisibilityRulePreprocessor.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/features/AdvancedFilteringFeatures.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/features/BUILD create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/features/Feature.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/features/FeatureMap.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/features/Features.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/generators/BUILD create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/generators/CountryNameGenerator.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/generators/EpitaphToLocalizedMessage.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/generators/InterstitialReasonToLocalizedMessage.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/generators/LocalizedInterstitialGenerator.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/generators/TombstoneGenerator.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/interfaces/blender/BUILD create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/interfaces/blender/BlenderVisibilityLibrary.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/interfaces/blender/BlenderVisibilityRequest.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/interfaces/blender/CombinedVisibilityResult.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/interfaces/cards/BUILD create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/interfaces/cards/CardVisibilityLibrary.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/interfaces/cards/CardVisibilityLibraryParityTest.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/interfaces/cards/CardVisibilityRequest.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/interfaces/common/BUILD.bazel create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/interfaces/common/blender/BUILD create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/interfaces/common/blender/BlenderVFRequestContext.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/interfaces/common/search/BUILD create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/interfaces/common/search/SearchVFRequestContext.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/interfaces/common/tweets/BUILD create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/interfaces/common/tweets/StratoSafetyLabelFetcher.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/interfaces/common/tweets/StratoSafetyLabelMapFetcher.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/interfaces/common/tweets/package.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/interfaces/conversations/AdAvoidanceLibrary.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/interfaces/conversations/BUILD create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/interfaces/conversations/TimelineConversationsVisibilityLibrary.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/interfaces/conversations/TimelineConversationsVisibilityRequest.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/interfaces/conversations/TimelineConversationsVisibilityResponse.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/interfaces/conversations/Tombstone.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/interfaces/conversations/TombstoneVisibilityLibrary.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/interfaces/conversations/package.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/interfaces/des/BUILD create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/interfaces/des/DESRealtimeVisibilityLibrary.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/interfaces/des/DESVisibilityLibrary.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/interfaces/dms/BUILD create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/interfaces/dms/DmConversationVisibilityLibrary.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/interfaces/dms/DmConversationVisibilityRequest.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/interfaces/dms/DmEventVisibilityLibrary.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/interfaces/dms/DmEventVisibilityRequest.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/interfaces/dms/DmVisibilityLibrary.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/interfaces/dms/package.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/interfaces/media/BUILD.bazel create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/interfaces/media/MediaVisibilityLibrary.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/interfaces/media/MediaVisibilityRequest.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/interfaces/notifications/BUILD create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/interfaces/notifications/NotificationVFRequest.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/interfaces/notifications/NotificationsFilteringResponse.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/interfaces/notifications/NotificationsPlatformFilteringResponse.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/interfaces/notifications/NotificationsPlatformVisibilityLibrary.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/interfaces/notifications/NotificationsVisibilityLibrary.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/interfaces/push_service/BUILD.bazel create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/interfaces/push_service/PushServiceSafetyLabelMapFetcher.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/interfaces/push_service/PushServiceVisibilityLibrary.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/interfaces/push_service/PushServiceVisibilityLibraryParity.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/interfaces/push_service/PushServiceVisibilityLibraryUtil.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/interfaces/push_service/PushServiceVisibilityRequest.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/interfaces/push_service/PushServiceVisibilityResponse.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/interfaces/search/BUILD create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/interfaces/search/BatchSearchVisibilityRequest.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/interfaces/search/BatchSearchVisibilityResponse.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/interfaces/search/CombinedVisibilityResult.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/interfaces/search/SearchVisibilityLibrary.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/interfaces/search/TweetContext.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/interfaces/spaces/BUILD create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/interfaces/spaces/SpaceVisibilityLibrary.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/interfaces/spaces/SpaceVisibilityRequest.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/interfaces/tweets/BUILD create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/interfaces/tweets/DeletedTweetVisibilityLibrary.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/interfaces/tweets/QuotedTweetVisibilityLibrary.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/interfaces/tweets/TweetVisibilityLibrary.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/interfaces/tweets/TweetVisibilityLibraryParityTest.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/interfaces/tweets/TweetVisibilityRequest.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/interfaces/tweets/TweetypieContext.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/interfaces/tweets/UserUnavailableStateVisibilityLibrary.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/interfaces/tweets/UserUnavailableStateVisibilityRequest.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/interfaces/tweets/enrichments/BUILD create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/interfaces/tweets/enrichments/ComplianceTweetNoticeEnrichment.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/interfaces/tweets/enrichments/LimitedActionsPolicyEnrichment.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/interfaces/tweets/enrichments/TweetVisibilityNudgeEnrichment.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/interfaces/users/BUILD create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/interfaces/users/UserVisibilityLibrary.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/models/BUILD create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/models/CommunityTweet.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/models/ContentId.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/models/LabelSource.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/models/MediaSafetyLabelType.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/models/MisinformationPolicy.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/models/MutedKeyword.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/models/SafetyLabel.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/models/SafetyLabelMetadata.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/models/SafetyLabelType.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/models/SafetyLevel.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/models/SafetyLevelGroup.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/models/SemanticCoreAnnotation.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/models/SpaceSafetyLabelType.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/models/TweetDeleteReason.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/models/TweetModelMetadata.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/models/TweetSafetyLabel.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/models/UnitOfDiversion.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/models/UserAge.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/models/UserLabel.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/models/UserSensitiveMediaSettings.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/models/UserUnavailableStateEnum.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/models/ViewerContext.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/models/ViolationLevel.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/models/package.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/rules/Action.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/rules/AdvancedFilteringRules.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/rules/BUILD create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/rules/CardRules.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/rules/ComposableActions.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/rules/Condition.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/rules/DmConversationRules.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/rules/DmEventRules.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/rules/DmVisibilityPolicies.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/rules/DownrankingRules.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/rules/EvaluationContext.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/rules/ExperimentBase.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/rules/FailClosedException.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/rules/FollowerRelations.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/rules/ForEmergencyUseOnly.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/rules/FreedomOfSpeechNotReach.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/rules/InterstitialIf.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/rules/PublicInterestRules.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/rules/Rule.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/rules/RuleActionSourceBuilder.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/rules/RuleBase.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/rules/Rules.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/rules/SafeSearchRules.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/rules/SearchBlenderRules.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/rules/SensitiveMediaSettingsRules.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/rules/SpaceRules.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/rules/TombstoneIf.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/rules/ToxicityReplyFilterRules.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/rules/TweetLabelRules.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/rules/TweetRules.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/rules/UserLabelRules.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/rules/UserUnavailableStateTombstoneRules.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/rules/VisibilityPolicy.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/rules/generators/BUILD create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/rules/generators/RuleGenerator.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/rules/generators/TweetRuleGenerator.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/rules/generators/TweetVisibilityPolicy.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/rules/package.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/rules/providers/BUILD create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/rules/providers/InjectedPolicyProvider.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/rules/providers/PolicyProvider.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/rules/providers/ProvidedEvaluationContext.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/rules/utils/BUILD create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/rules/utils/ShimUtils.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/util/BUILD create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/util/DeciderUtil.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/util/FeatureSwitchUtil.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/util/LoggingUtil.scala create mode 100644 visibilitylib/src/main/scala/com/twitter/visibility/util/NamingUtils.scala diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000000..5ca0973f8f --- /dev/null +++ b/.gitignore @@ -0,0 +1,2 @@ +.DS_Store + diff --git a/COPYING b/COPYING new file mode 100644 index 0000000000..be3f7b28e5 --- /dev/null +++ b/COPYING @@ -0,0 +1,661 @@ + GNU AFFERO GENERAL PUBLIC LICENSE + Version 3, 19 November 2007 + + Copyright (C) 2007 Free Software Foundation, Inc. + Everyone is permitted to copy and distribute verbatim copies + of this license document, but changing it is not allowed. + + Preamble + + The GNU Affero General Public License is a free, copyleft license for +software and other kinds of works, specifically designed to ensure +cooperation with the community in the case of network server software. + + The licenses for most software and other practical works are designed +to take away your freedom to share and change the works. By contrast, +our General Public Licenses are intended to guarantee your freedom to +share and change all versions of a program--to make sure it remains free +software for all its users. + + When we speak of free software, we are referring to freedom, not +price. Our General Public Licenses are designed to make sure that you +have the freedom to distribute copies of free software (and charge for +them if you wish), that you receive source code or can get it if you +want it, that you can change the software or use pieces of it in new +free programs, and that you know you can do these things. + + Developers that use our General Public Licenses protect your rights +with two steps: (1) assert copyright on the software, and (2) offer +you this License which gives you legal permission to copy, distribute +and/or modify the software. + + A secondary benefit of defending all users' freedom is that +improvements made in alternate versions of the program, if they +receive widespread use, become available for other developers to +incorporate. Many developers of free software are heartened and +encouraged by the resulting cooperation. However, in the case of +software used on network servers, this result may fail to come about. +The GNU General Public License permits making a modified version and +letting the public access it on a server without ever releasing its +source code to the public. + + The GNU Affero General Public License is designed specifically to +ensure that, in such cases, the modified source code becomes available +to the community. It requires the operator of a network server to +provide the source code of the modified version running there to the +users of that server. Therefore, public use of a modified version, on +a publicly accessible server, gives the public access to the source +code of the modified version. + + An older license, called the Affero General Public License and +published by Affero, was designed to accomplish similar goals. This is +a different license, not a version of the Affero GPL, but Affero has +released a new version of the Affero GPL which permits relicensing under +this license. + + The precise terms and conditions for copying, distribution and +modification follow. + + TERMS AND CONDITIONS + + 0. Definitions. + + "This License" refers to version 3 of the GNU Affero General Public License. + + "Copyright" also means copyright-like laws that apply to other kinds of +works, such as semiconductor masks. + + "The Program" refers to any copyrightable work licensed under this +License. Each licensee is addressed as "you". "Licensees" and +"recipients" may be individuals or organizations. + + To "modify" a work means to copy from or adapt all or part of the work +in a fashion requiring copyright permission, other than the making of an +exact copy. The resulting work is called a "modified version" of the +earlier work or a work "based on" the earlier work. + + A "covered work" means either the unmodified Program or a work based +on the Program. + + To "propagate" a work means to do anything with it that, without +permission, would make you directly or secondarily liable for +infringement under applicable copyright law, except executing it on a +computer or modifying a private copy. Propagation includes copying, +distribution (with or without modification), making available to the +public, and in some countries other activities as well. + + To "convey" a work means any kind of propagation that enables other +parties to make or receive copies. Mere interaction with a user through +a computer network, with no transfer of a copy, is not conveying. + + An interactive user interface displays "Appropriate Legal Notices" +to the extent that it includes a convenient and prominently visible +feature that (1) displays an appropriate copyright notice, and (2) +tells the user that there is no warranty for the work (except to the +extent that warranties are provided), that licensees may convey the +work under this License, and how to view a copy of this License. If +the interface presents a list of user commands or options, such as a +menu, a prominent item in the list meets this criterion. + + 1. Source Code. + + The "source code" for a work means the preferred form of the work +for making modifications to it. "Object code" means any non-source +form of a work. + + A "Standard Interface" means an interface that either is an official +standard defined by a recognized standards body, or, in the case of +interfaces specified for a particular programming language, one that +is widely used among developers working in that language. + + The "System Libraries" of an executable work include anything, other +than the work as a whole, that (a) is included in the normal form of +packaging a Major Component, but which is not part of that Major +Component, and (b) serves only to enable use of the work with that +Major Component, or to implement a Standard Interface for which an +implementation is available to the public in source code form. A +"Major Component", in this context, means a major essential component +(kernel, window system, and so on) of the specific operating system +(if any) on which the executable work runs, or a compiler used to +produce the work, or an object code interpreter used to run it. + + The "Corresponding Source" for a work in object code form means all +the source code needed to generate, install, and (for an executable +work) run the object code and to modify the work, including scripts to +control those activities. However, it does not include the work's +System Libraries, or general-purpose tools or generally available free +programs which are used unmodified in performing those activities but +which are not part of the work. For example, Corresponding Source +includes interface definition files associated with source files for +the work, and the source code for shared libraries and dynamically +linked subprograms that the work is specifically designed to require, +such as by intimate data communication or control flow between those +subprograms and other parts of the work. + + The Corresponding Source need not include anything that users +can regenerate automatically from other parts of the Corresponding +Source. + + The Corresponding Source for a work in source code form is that +same work. + + 2. Basic Permissions. + + All rights granted under this License are granted for the term of +copyright on the Program, and are irrevocable provided the stated +conditions are met. This License explicitly affirms your unlimited +permission to run the unmodified Program. The output from running a +covered work is covered by this License only if the output, given its +content, constitutes a covered work. This License acknowledges your +rights of fair use or other equivalent, as provided by copyright law. + + You may make, run and propagate covered works that you do not +convey, without conditions so long as your license otherwise remains +in force. You may convey covered works to others for the sole purpose +of having them make modifications exclusively for you, or provide you +with facilities for running those works, provided that you comply with +the terms of this License in conveying all material for which you do +not control copyright. Those thus making or running the covered works +for you must do so exclusively on your behalf, under your direction +and control, on terms that prohibit them from making any copies of +your copyrighted material outside their relationship with you. + + Conveying under any other circumstances is permitted solely under +the conditions stated below. Sublicensing is not allowed; section 10 +makes it unnecessary. + + 3. Protecting Users' Legal Rights From Anti-Circumvention Law. + + No covered work shall be deemed part of an effective technological +measure under any applicable law fulfilling obligations under article +11 of the WIPO copyright treaty adopted on 20 December 1996, or +similar laws prohibiting or restricting circumvention of such +measures. + + When you convey a covered work, you waive any legal power to forbid +circumvention of technological measures to the extent such circumvention +is effected by exercising rights under this License with respect to +the covered work, and you disclaim any intention to limit operation or +modification of the work as a means of enforcing, against the work's +users, your or third parties' legal rights to forbid circumvention of +technological measures. + + 4. Conveying Verbatim Copies. + + You may convey verbatim copies of the Program's source code as you +receive it, in any medium, provided that you conspicuously and +appropriately publish on each copy an appropriate copyright notice; +keep intact all notices stating that this License and any +non-permissive terms added in accord with section 7 apply to the code; +keep intact all notices of the absence of any warranty; and give all +recipients a copy of this License along with the Program. + + You may charge any price or no price for each copy that you convey, +and you may offer support or warranty protection for a fee. + + 5. Conveying Modified Source Versions. + + You may convey a work based on the Program, or the modifications to +produce it from the Program, in the form of source code under the +terms of section 4, provided that you also meet all of these conditions: + + a) The work must carry prominent notices stating that you modified + it, and giving a relevant date. + + b) The work must carry prominent notices stating that it is + released under this License and any conditions added under section + 7. This requirement modifies the requirement in section 4 to + "keep intact all notices". + + c) You must license the entire work, as a whole, under this + License to anyone who comes into possession of a copy. This + License will therefore apply, along with any applicable section 7 + additional terms, to the whole of the work, and all its parts, + regardless of how they are packaged. This License gives no + permission to license the work in any other way, but it does not + invalidate such permission if you have separately received it. + + d) If the work has interactive user interfaces, each must display + Appropriate Legal Notices; however, if the Program has interactive + interfaces that do not display Appropriate Legal Notices, your + work need not make them do so. + + A compilation of a covered work with other separate and independent +works, which are not by their nature extensions of the covered work, +and which are not combined with it such as to form a larger program, +in or on a volume of a storage or distribution medium, is called an +"aggregate" if the compilation and its resulting copyright are not +used to limit the access or legal rights of the compilation's users +beyond what the individual works permit. Inclusion of a covered work +in an aggregate does not cause this License to apply to the other +parts of the aggregate. + + 6. Conveying Non-Source Forms. + + You may convey a covered work in object code form under the terms +of sections 4 and 5, provided that you also convey the +machine-readable Corresponding Source under the terms of this License, +in one of these ways: + + a) Convey the object code in, or embodied in, a physical product + (including a physical distribution medium), accompanied by the + Corresponding Source fixed on a durable physical medium + customarily used for software interchange. + + b) Convey the object code in, or embodied in, a physical product + (including a physical distribution medium), accompanied by a + written offer, valid for at least three years and valid for as + long as you offer spare parts or customer support for that product + model, to give anyone who possesses the object code either (1) a + copy of the Corresponding Source for all the software in the + product that is covered by this License, on a durable physical + medium customarily used for software interchange, for a price no + more than your reasonable cost of physically performing this + conveying of source, or (2) access to copy the + Corresponding Source from a network server at no charge. + + c) Convey individual copies of the object code with a copy of the + written offer to provide the Corresponding Source. This + alternative is allowed only occasionally and noncommercially, and + only if you received the object code with such an offer, in accord + with subsection 6b. + + d) Convey the object code by offering access from a designated + place (gratis or for a charge), and offer equivalent access to the + Corresponding Source in the same way through the same place at no + further charge. You need not require recipients to copy the + Corresponding Source along with the object code. If the place to + copy the object code is a network server, the Corresponding Source + may be on a different server (operated by you or a third party) + that supports equivalent copying facilities, provided you maintain + clear directions next to the object code saying where to find the + Corresponding Source. Regardless of what server hosts the + Corresponding Source, you remain obligated to ensure that it is + available for as long as needed to satisfy these requirements. + + e) Convey the object code using peer-to-peer transmission, provided + you inform other peers where the object code and Corresponding + Source of the work are being offered to the general public at no + charge under subsection 6d. + + A separable portion of the object code, whose source code is excluded +from the Corresponding Source as a System Library, need not be +included in conveying the object code work. + + A "User Product" is either (1) a "consumer product", which means any +tangible personal property which is normally used for personal, family, +or household purposes, or (2) anything designed or sold for incorporation +into a dwelling. In determining whether a product is a consumer product, +doubtful cases shall be resolved in favor of coverage. For a particular +product received by a particular user, "normally used" refers to a +typical or common use of that class of product, regardless of the status +of the particular user or of the way in which the particular user +actually uses, or expects or is expected to use, the product. A product +is a consumer product regardless of whether the product has substantial +commercial, industrial or non-consumer uses, unless such uses represent +the only significant mode of use of the product. + + "Installation Information" for a User Product means any methods, +procedures, authorization keys, or other information required to install +and execute modified versions of a covered work in that User Product from +a modified version of its Corresponding Source. The information must +suffice to ensure that the continued functioning of the modified object +code is in no case prevented or interfered with solely because +modification has been made. + + If you convey an object code work under this section in, or with, or +specifically for use in, a User Product, and the conveying occurs as +part of a transaction in which the right of possession and use of the +User Product is transferred to the recipient in perpetuity or for a +fixed term (regardless of how the transaction is characterized), the +Corresponding Source conveyed under this section must be accompanied +by the Installation Information. But this requirement does not apply +if neither you nor any third party retains the ability to install +modified object code on the User Product (for example, the work has +been installed in ROM). + + The requirement to provide Installation Information does not include a +requirement to continue to provide support service, warranty, or updates +for a work that has been modified or installed by the recipient, or for +the User Product in which it has been modified or installed. Access to a +network may be denied when the modification itself materially and +adversely affects the operation of the network or violates the rules and +protocols for communication across the network. + + Corresponding Source conveyed, and Installation Information provided, +in accord with this section must be in a format that is publicly +documented (and with an implementation available to the public in +source code form), and must require no special password or key for +unpacking, reading or copying. + + 7. Additional Terms. + + "Additional permissions" are terms that supplement the terms of this +License by making exceptions from one or more of its conditions. +Additional permissions that are applicable to the entire Program shall +be treated as though they were included in this License, to the extent +that they are valid under applicable law. If additional permissions +apply only to part of the Program, that part may be used separately +under those permissions, but the entire Program remains governed by +this License without regard to the additional permissions. + + When you convey a copy of a covered work, you may at your option +remove any additional permissions from that copy, or from any part of +it. (Additional permissions may be written to require their own +removal in certain cases when you modify the work.) You may place +additional permissions on material, added by you to a covered work, +for which you have or can give appropriate copyright permission. + + Notwithstanding any other provision of this License, for material you +add to a covered work, you may (if authorized by the copyright holders of +that material) supplement the terms of this License with terms: + + a) Disclaiming warranty or limiting liability differently from the + terms of sections 15 and 16 of this License; or + + b) Requiring preservation of specified reasonable legal notices or + author attributions in that material or in the Appropriate Legal + Notices displayed by works containing it; or + + c) Prohibiting misrepresentation of the origin of that material, or + requiring that modified versions of such material be marked in + reasonable ways as different from the original version; or + + d) Limiting the use for publicity purposes of names of licensors or + authors of the material; or + + e) Declining to grant rights under trademark law for use of some + trade names, trademarks, or service marks; or + + f) Requiring indemnification of licensors and authors of that + material by anyone who conveys the material (or modified versions of + it) with contractual assumptions of liability to the recipient, for + any liability that these contractual assumptions directly impose on + those licensors and authors. + + All other non-permissive additional terms are considered "further +restrictions" within the meaning of section 10. If the Program as you +received it, or any part of it, contains a notice stating that it is +governed by this License along with a term that is a further +restriction, you may remove that term. If a license document contains +a further restriction but permits relicensing or conveying under this +License, you may add to a covered work material governed by the terms +of that license document, provided that the further restriction does +not survive such relicensing or conveying. + + If you add terms to a covered work in accord with this section, you +must place, in the relevant source files, a statement of the +additional terms that apply to those files, or a notice indicating +where to find the applicable terms. + + Additional terms, permissive or non-permissive, may be stated in the +form of a separately written license, or stated as exceptions; +the above requirements apply either way. + + 8. Termination. + + You may not propagate or modify a covered work except as expressly +provided under this License. Any attempt otherwise to propagate or +modify it is void, and will automatically terminate your rights under +this License (including any patent licenses granted under the third +paragraph of section 11). + + However, if you cease all violation of this License, then your +license from a particular copyright holder is reinstated (a) +provisionally, unless and until the copyright holder explicitly and +finally terminates your license, and (b) permanently, if the copyright +holder fails to notify you of the violation by some reasonable means +prior to 60 days after the cessation. + + Moreover, your license from a particular copyright holder is +reinstated permanently if the copyright holder notifies you of the +violation by some reasonable means, this is the first time you have +received notice of violation of this License (for any work) from that +copyright holder, and you cure the violation prior to 30 days after +your receipt of the notice. + + Termination of your rights under this section does not terminate the +licenses of parties who have received copies or rights from you under +this License. If your rights have been terminated and not permanently +reinstated, you do not qualify to receive new licenses for the same +material under section 10. + + 9. Acceptance Not Required for Having Copies. + + You are not required to accept this License in order to receive or +run a copy of the Program. Ancillary propagation of a covered work +occurring solely as a consequence of using peer-to-peer transmission +to receive a copy likewise does not require acceptance. However, +nothing other than this License grants you permission to propagate or +modify any covered work. These actions infringe copyright if you do +not accept this License. Therefore, by modifying or propagating a +covered work, you indicate your acceptance of this License to do so. + + 10. Automatic Licensing of Downstream Recipients. + + Each time you convey a covered work, the recipient automatically +receives a license from the original licensors, to run, modify and +propagate that work, subject to this License. You are not responsible +for enforcing compliance by third parties with this License. + + An "entity transaction" is a transaction transferring control of an +organization, or substantially all assets of one, or subdividing an +organization, or merging organizations. If propagation of a covered +work results from an entity transaction, each party to that +transaction who receives a copy of the work also receives whatever +licenses to the work the party's predecessor in interest had or could +give under the previous paragraph, plus a right to possession of the +Corresponding Source of the work from the predecessor in interest, if +the predecessor has it or can get it with reasonable efforts. + + You may not impose any further restrictions on the exercise of the +rights granted or affirmed under this License. For example, you may +not impose a license fee, royalty, or other charge for exercise of +rights granted under this License, and you may not initiate litigation +(including a cross-claim or counterclaim in a lawsuit) alleging that +any patent claim is infringed by making, using, selling, offering for +sale, or importing the Program or any portion of it. + + 11. Patents. + + A "contributor" is a copyright holder who authorizes use under this +License of the Program or a work on which the Program is based. The +work thus licensed is called the contributor's "contributor version". + + A contributor's "essential patent claims" are all patent claims +owned or controlled by the contributor, whether already acquired or +hereafter acquired, that would be infringed by some manner, permitted +by this License, of making, using, or selling its contributor version, +but do not include claims that would be infringed only as a +consequence of further modification of the contributor version. For +purposes of this definition, "control" includes the right to grant +patent sublicenses in a manner consistent with the requirements of +this License. + + Each contributor grants you a non-exclusive, worldwide, royalty-free +patent license under the contributor's essential patent claims, to +make, use, sell, offer for sale, import and otherwise run, modify and +propagate the contents of its contributor version. + + In the following three paragraphs, a "patent license" is any express +agreement or commitment, however denominated, not to enforce a patent +(such as an express permission to practice a patent or covenant not to +sue for patent infringement). To "grant" such a patent license to a +party means to make such an agreement or commitment not to enforce a +patent against the party. + + If you convey a covered work, knowingly relying on a patent license, +and the Corresponding Source of the work is not available for anyone +to copy, free of charge and under the terms of this License, through a +publicly available network server or other readily accessible means, +then you must either (1) cause the Corresponding Source to be so +available, or (2) arrange to deprive yourself of the benefit of the +patent license for this particular work, or (3) arrange, in a manner +consistent with the requirements of this License, to extend the patent +license to downstream recipients. "Knowingly relying" means you have +actual knowledge that, but for the patent license, your conveying the +covered work in a country, or your recipient's use of the covered work +in a country, would infringe one or more identifiable patents in that +country that you have reason to believe are valid. + + If, pursuant to or in connection with a single transaction or +arrangement, you convey, or propagate by procuring conveyance of, a +covered work, and grant a patent license to some of the parties +receiving the covered work authorizing them to use, propagate, modify +or convey a specific copy of the covered work, then the patent license +you grant is automatically extended to all recipients of the covered +work and works based on it. + + A patent license is "discriminatory" if it does not include within +the scope of its coverage, prohibits the exercise of, or is +conditioned on the non-exercise of one or more of the rights that are +specifically granted under this License. You may not convey a covered +work if you are a party to an arrangement with a third party that is +in the business of distributing software, under which you make payment +to the third party based on the extent of your activity of conveying +the work, and under which the third party grants, to any of the +parties who would receive the covered work from you, a discriminatory +patent license (a) in connection with copies of the covered work +conveyed by you (or copies made from those copies), or (b) primarily +for and in connection with specific products or compilations that +contain the covered work, unless you entered into that arrangement, +or that patent license was granted, prior to 28 March 2007. + + Nothing in this License shall be construed as excluding or limiting +any implied license or other defenses to infringement that may +otherwise be available to you under applicable patent law. + + 12. No Surrender of Others' Freedom. + + If conditions are imposed on you (whether by court order, agreement or +otherwise) that contradict the conditions of this License, they do not +excuse you from the conditions of this License. If you cannot convey a +covered work so as to satisfy simultaneously your obligations under this +License and any other pertinent obligations, then as a consequence you may +not convey it at all. For example, if you agree to terms that obligate you +to collect a royalty for further conveying from those to whom you convey +the Program, the only way you could satisfy both those terms and this +License would be to refrain entirely from conveying the Program. + + 13. Remote Network Interaction; Use with the GNU General Public License. + + Notwithstanding any other provision of this License, if you modify the +Program, your modified version must prominently offer all users +interacting with it remotely through a computer network (if your version +supports such interaction) an opportunity to receive the Corresponding +Source of your version by providing access to the Corresponding Source +from a network server at no charge, through some standard or customary +means of facilitating copying of software. This Corresponding Source +shall include the Corresponding Source for any work covered by version 3 +of the GNU General Public License that is incorporated pursuant to the +following paragraph. + + Notwithstanding any other provision of this License, you have +permission to link or combine any covered work with a work licensed +under version 3 of the GNU General Public License into a single +combined work, and to convey the resulting work. The terms of this +License will continue to apply to the part which is the covered work, +but the work with which it is combined will remain governed by version +3 of the GNU General Public License. + + 14. Revised Versions of this License. + + The Free Software Foundation may publish revised and/or new versions of +the GNU Affero General Public License from time to time. Such new versions +will be similar in spirit to the present version, but may differ in detail to +address new problems or concerns. + + Each version is given a distinguishing version number. If the +Program specifies that a certain numbered version of the GNU Affero General +Public License "or any later version" applies to it, you have the +option of following the terms and conditions either of that numbered +version or of any later version published by the Free Software +Foundation. If the Program does not specify a version number of the +GNU Affero General Public License, you may choose any version ever published +by the Free Software Foundation. + + If the Program specifies that a proxy can decide which future +versions of the GNU Affero General Public License can be used, that proxy's +public statement of acceptance of a version permanently authorizes you +to choose that version for the Program. + + Later license versions may give you additional or different +permissions. However, no additional obligations are imposed on any +author or copyright holder as a result of your choosing to follow a +later version. + + 15. Disclaimer of Warranty. + + THERE IS NO WARRANTY FOR THE PROGRAM, TO THE EXTENT PERMITTED BY +APPLICABLE LAW. EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT +HOLDERS AND/OR OTHER PARTIES PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY +OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO, +THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR +PURPOSE. THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE PROGRAM +IS WITH YOU. SHOULD THE PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF +ALL NECESSARY SERVICING, REPAIR OR CORRECTION. + + 16. Limitation of Liability. + + IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING +WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MODIFIES AND/OR CONVEYS +THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES, INCLUDING ANY +GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING OUT OF THE +USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED TO LOSS OF +DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD +PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER PROGRAMS), +EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF +SUCH DAMAGES. + + 17. Interpretation of Sections 15 and 16. + + If the disclaimer of warranty and limitation of liability provided +above cannot be given local legal effect according to their terms, +reviewing courts shall apply local law that most closely approximates +an absolute waiver of all civil liability in connection with the +Program, unless a warranty or assumption of liability accompanies a +copy of the Program in return for a fee. + + END OF TERMS AND CONDITIONS + + How to Apply These Terms to Your New Programs + + If you develop a new program, and you want it to be of the greatest +possible use to the public, the best way to achieve this is to make it +free software which everyone can redistribute and change under these terms. + + To do so, attach the following notices to the program. It is safest +to attach them to the start of each source file to most effectively +state the exclusion of warranty; and each file should have at least +the "copyright" line and a pointer to where the full notice is found. + + + Copyright (C) + + This program is free software: you can redistribute it and/or modify + it under the terms of the GNU Affero General Public License as published by + the Free Software Foundation, either version 3 of the License, or + (at your option) any later version. + + This program is distributed in the hope that it will be useful, + but WITHOUT ANY WARRANTY; without even the implied warranty of + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + GNU Affero General Public License for more details. + + You should have received a copy of the GNU Affero General Public License + along with this program. If not, see . + +Also add information on how to contact you by electronic and paper mail. + + If your software can interact with users remotely through a computer +network, you should also make sure that it provides a way for users to +get its source. For example, if your program is a web application, its +interface could display a "Source" link that leads users to an archive +of the code. There are many ways you could offer source, and different +solutions will be better for different programs; see section 13 for the +specific requirements. + + You should also get your employer (if you work as a programmer) or school, +if any, to sign a "copyright disclaimer" for the program, if necessary. +For more information on this, and how to apply and follow the GNU AGPL, see +. diff --git a/README.md b/README.md new file mode 100644 index 0000000000..09b81856fa --- /dev/null +++ b/README.md @@ -0,0 +1,39 @@ +# Twitter Recommendation Algorithm + +The Twitter Recommendation Algorithm is a set of services and jobs that are responsible for constructing and serving the +Home Timeline. For an introduction to how the algorithm works, please refer to our [engineering blog](https://blog.twitter.com/engineering/en_us/topics/open-source/2023/twitter-recommendation-algorithm). The +diagram below illustrates how major services and jobs interconnect. + +![](docs/system-diagram.png) + +These are the main components of the Recommendation Algorithm included in this repository: + +| Type | Component | Description | +|------------|------------|------------| +| Feature | [simclusters-ann](simclusters-ann/README.md) | Community detection and sparse embeddings into those communities. | +| | [TwHIN](https://github.com/twitter/the-algorithm-ml/blob/main/projects/twhin/README.md) | Dense knowledge graph embeddings for Users and Tweets. | +| | [trust-and-safety-models](trust_and_safety_models/README.md) | Models for detecting NSFW or abusive content. | +| | [real-graph](src/scala/com/twitter/interaction_graph/README.md) | Model to predict likelihood of a Twitter User interacting with another User. | +| | [tweepcred](src/scala/com/twitter/graph/batch/job/tweepcred/README) | Page-Rank algorithm for calculating Twitter User reputation. | +| | [recos-injector](recos-injector/README.md) | Streaming event processor for building input streams for [GraphJet](https://github.com/twitter/GraphJet) based services. | +| | [graph-feature-service](graph-feature-service/README.md) | Serves graph features for a directed pair of Users (e.g. how many of User A's following liked Tweets from User B). | +| Candidate Source | [search-index](src/java/com/twitter/search/README.md) | Find and rank In-Network Tweets. ~50% of Tweets come from this candidate source. | +| | [cr-mixer](cr-mixer/README.md) | Coordination layer for fetching Out-of-Network tweet candidates from underlying compute services. | +| | [user-tweet-entity-graph](src/scala/com/twitter/recos/user_tweet_entity_graph/README.md) (UTEG)| Maintains an in memory User to Tweet interaction graph, and finds candidates based on traversals of this graph. This is built on the [GraphJet](https://github.com/twitter/GraphJet) framework. Several other GraphJet based features and candidate sources are located [here](src/scala/com/twitter/recos) | +| | [follow-recommendation-service](follow-recommendations-service/README.md) (FRS)| Provides Users with recommendations for accounts to follow, and Tweets from those accounts. | +| Ranking | [light-ranker](src/python/twitter/deepbird/projects/timelines/scripts/models/earlybird/README.md) | Light ranker model used by search index (Earlybird) to rank Tweets. | +| | [heavy-ranker](https://github.com/twitter/the-algorithm-ml/blob/main/projects/home/recap/README.md) | Neural network for ranking candidate tweets. One of the main signals used to select timeline Tweets post candidate sourcing. | +| Tweet mixing & filtering | [home-mixer](home-mixer/README.md) | Main service used to construct and serve the Home Timeline. Built on [product-mixer](product-mixer/README.md) | +| | [visibility-filters](visibilitylib/README.md) | Responsible for filtering Twitter content to support legal compliance, improve product quality, increase user trust, protect revenue through the use of hard-filtering, visible product treatments, and coarse-grained downranking. | +| | [timelineranker](timelineranker/README.md) | Legacy service which provides relevance-scored tweets from the Earlybird Search Index and UTEG service. | +| Software framework | [navi](navi/navi/README.md) | High performance, machine learning model serving written in Rust. | +| | [product-mixer](product-mixer/README.md) | Software framework for building feeds of content. | +| | [twml](twml/README.md) | Legacy machine learning framework built on TensorFlow v1. | + +We include Bazel BUILD files for most components, but not a top level BUILD or WORKSPACE file. + +## Contributing + +We invite the community to submit GitHub issues and pull requests for suggestions on improving the recommendation algorithm. We are working on tools to manage these suggestions and sync changes to our internal repository. Any security concerns or issues should be routed to our official [bug bounty program](https://hackerone.com/twitter) through HackerOne. We hope to benefit from the collective intelligence and expertise of the global community in helping us identify issues and suggest improvements, ultimately leading to a better Twitter. + +Read our blog on the open source initiative [here](https://blog.twitter.com/en_us/topics/company/2023/a-new-era-of-transparency-for-twitter). diff --git a/ann/src/main/java/com/twitter/ann/faiss/BUILD b/ann/src/main/java/com/twitter/ann/faiss/BUILD new file mode 100644 index 0000000000..2320a1dae2 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/BUILD @@ -0,0 +1,15 @@ +target( + name = "faiss", + dependencies = [ + "ann/src/main/java/com/twitter/ann/faiss/swig:swig-artifactory", + ], +) + +java_library( + name = "swig-native-utils", + sources = ["*.java"], + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [], +) diff --git a/ann/src/main/java/com/twitter/ann/faiss/NativeUtils.java b/ann/src/main/java/com/twitter/ann/faiss/NativeUtils.java new file mode 100644 index 0000000000..424d28890a --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/NativeUtils.java @@ -0,0 +1,151 @@ +package com.twitter.ann.faiss; + +import java.io.File; +import java.io.FileNotFoundException; +import java.io.IOException; +import java.io.InputStream; +import java.nio.file.Files; +import java.nio.file.StandardCopyOption; +import java.util.Locale; + +public final class NativeUtils { + + private static final int MIN_PREFIX_LENGTH = 3; + public static final String NATIVE_FOLDER_PATH_PREFIX = "nativeutils"; + + public static File temporaryDir; + + private NativeUtils() { + } + + private static File unpackLibraryFromJarInternal(String path) throws IOException { + if (null == path || !path.startsWith("/")) { + throw new IllegalArgumentException("The path has to be absolute (start with '/')."); + } + + String[] parts = path.split("/"); + String filename = (parts.length > 1) ? parts[parts.length - 1] : null; + + if (filename == null || filename.length() < MIN_PREFIX_LENGTH) { + throw new IllegalArgumentException("The filename has to be at least 3 characters long."); + } + + if (temporaryDir == null) { + temporaryDir = createTempDirectory(NATIVE_FOLDER_PATH_PREFIX); + temporaryDir.deleteOnExit(); + } + + File temp = new File(temporaryDir, filename); + + try (InputStream is = NativeUtils.class.getResourceAsStream(path)) { + Files.copy(is, temp.toPath(), StandardCopyOption.REPLACE_EXISTING); + } catch (IOException e) { + temp.delete(); + throw e; + } catch (NullPointerException e) { + temp.delete(); + throw new FileNotFoundException("File " + path + " was not found inside JAR."); + } + + return temp; + } + + /** + * Unpack library from JAR into temporary path + * + * @param path The path of file inside JAR as absolute path (beginning with + * '/'), e.g. /package/File.ext + * @throws IOException If temporary file creation or read/write + * operation fails + * @throws IllegalArgumentException If source file (param path) does not exist + * @throws IllegalArgumentException If the path is not absolute or if the + * filename is shorter than three characters + * (restriction of + * {@link File#createTempFile(java.lang.String, java.lang.String)}). + * @throws FileNotFoundException If the file could not be found inside the + * JAR. + */ + public static void unpackLibraryFromJar(String path) throws IOException { + unpackLibraryFromJarInternal(path); + } + + /** + * Loads library from current JAR archive + *

+ * The file from JAR is copied into system temporary directory and then loaded. + * The temporary file is deleted after + * exiting. + * Method uses String as filename because the pathname is "abstract", not + * system-dependent. + * + * @param path The path of file inside JAR as absolute path (beginning with + * '/'), e.g. /package/File.ext + * @throws IOException If temporary file creation or read/write + * operation fails + * @throws IllegalArgumentException If source file (param path) does not exist + * @throws IllegalArgumentException If the path is not absolute or if the + * filename is shorter than three characters + * (restriction of + * {@link File#createTempFile(java.lang.String, java.lang.String)}). + * @throws FileNotFoundException If the file could not be found inside the + * JAR. + */ + public static void loadLibraryFromJar(String path) throws IOException { + File temp = unpackLibraryFromJarInternal(path); + + try (InputStream is = NativeUtils.class.getResourceAsStream(path)) { + Files.copy(is, temp.toPath(), StandardCopyOption.REPLACE_EXISTING); + } catch (IOException e) { + temp.delete(); + throw e; + } catch (NullPointerException e) { + temp.delete(); + throw new FileNotFoundException("File " + path + " was not found inside JAR."); + } + + try { + System.load(temp.getAbsolutePath()); + } finally { + temp.deleteOnExit(); + } + } + + private static File createTempDirectory(String prefix) throws IOException { + String tempDir = System.getProperty("java.io.tmpdir"); + File generatedDir = new File(tempDir, prefix + System.nanoTime()); + + if (!generatedDir.mkdir()) { + throw new IOException("Failed to create temp directory " + generatedDir.getName()); + } + + return generatedDir; + } + + public enum OSType { + Windows, MacOS, Linux, Other + } + + protected static OSType detectedOS; + + /** + * detect the operating system from the os.name System property and cache + * the result + * + * @returns - the operating system detected + */ + public static OSType getOperatingSystemType() { + if (detectedOS == null) { + String osname = System.getProperty("os.name", "generic").toLowerCase(Locale.ENGLISH); + if ((osname.contains("mac")) || (osname.contains("darwin"))) { + detectedOS = OSType.MacOS; + } else if (osname.contains("win")) { + detectedOS = OSType.Windows; + } else if (osname.contains("nux")) { + detectedOS = OSType.Linux; + } else { + detectedOS = OSType.Other; + } + } + return detectedOS; + } +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/AlignedTableFloat32.java b/ann/src/main/java/com/twitter/ann/faiss/swig/AlignedTableFloat32.java new file mode 100644 index 0000000000..9758bd20de --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/AlignedTableFloat32.java @@ -0,0 +1,98 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class AlignedTableFloat32 { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected AlignedTableFloat32(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(AlignedTableFloat32 obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_AlignedTableFloat32(swigCPtr); + } + swigCPtr = 0; + } + } + + public void setTab(SWIGTYPE_p_faiss__AlignedTableTightAllocT_float_32_t value) { + swigfaissJNI.AlignedTableFloat32_tab_set(swigCPtr, this, SWIGTYPE_p_faiss__AlignedTableTightAllocT_float_32_t.getCPtr(value)); + } + + public SWIGTYPE_p_faiss__AlignedTableTightAllocT_float_32_t getTab() { + long cPtr = swigfaissJNI.AlignedTableFloat32_tab_get(swigCPtr, this); + return (cPtr == 0) ? null : new SWIGTYPE_p_faiss__AlignedTableTightAllocT_float_32_t(cPtr, false); + } + + public void setNumel(long value) { + swigfaissJNI.AlignedTableFloat32_numel_set(swigCPtr, this, value); + } + + public long getNumel() { + return swigfaissJNI.AlignedTableFloat32_numel_get(swigCPtr, this); + } + + public static long round_capacity(long n) { + return swigfaissJNI.AlignedTableFloat32_round_capacity(n); + } + + public AlignedTableFloat32() { + this(swigfaissJNI.new_AlignedTableFloat32__SWIG_0(), true); + } + + public AlignedTableFloat32(long n) { + this(swigfaissJNI.new_AlignedTableFloat32__SWIG_1(n), true); + } + + public long itemsize() { + return swigfaissJNI.AlignedTableFloat32_itemsize(swigCPtr, this); + } + + public void resize(long n) { + swigfaissJNI.AlignedTableFloat32_resize(swigCPtr, this, n); + } + + public void clear() { + swigfaissJNI.AlignedTableFloat32_clear(swigCPtr, this); + } + + public long size() { + return swigfaissJNI.AlignedTableFloat32_size(swigCPtr, this); + } + + public long nbytes() { + return swigfaissJNI.AlignedTableFloat32_nbytes(swigCPtr, this); + } + + public SWIGTYPE_p_float get() { + long cPtr = swigfaissJNI.AlignedTableFloat32_get__SWIG_0(swigCPtr, this); + return (cPtr == 0) ? null : new SWIGTYPE_p_float(cPtr, false); + } + + public SWIGTYPE_p_float data() { + long cPtr = swigfaissJNI.AlignedTableFloat32_data__SWIG_0(swigCPtr, this); + return (cPtr == 0) ? null : new SWIGTYPE_p_float(cPtr, false); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/AlignedTableUint16.java b/ann/src/main/java/com/twitter/ann/faiss/swig/AlignedTableUint16.java new file mode 100644 index 0000000000..1ce3c67f78 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/AlignedTableUint16.java @@ -0,0 +1,98 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class AlignedTableUint16 { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected AlignedTableUint16(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(AlignedTableUint16 obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_AlignedTableUint16(swigCPtr); + } + swigCPtr = 0; + } + } + + public void setTab(SWIGTYPE_p_faiss__AlignedTableTightAllocT_uint16_t_32_t value) { + swigfaissJNI.AlignedTableUint16_tab_set(swigCPtr, this, SWIGTYPE_p_faiss__AlignedTableTightAllocT_uint16_t_32_t.getCPtr(value)); + } + + public SWIGTYPE_p_faiss__AlignedTableTightAllocT_uint16_t_32_t getTab() { + long cPtr = swigfaissJNI.AlignedTableUint16_tab_get(swigCPtr, this); + return (cPtr == 0) ? null : new SWIGTYPE_p_faiss__AlignedTableTightAllocT_uint16_t_32_t(cPtr, false); + } + + public void setNumel(long value) { + swigfaissJNI.AlignedTableUint16_numel_set(swigCPtr, this, value); + } + + public long getNumel() { + return swigfaissJNI.AlignedTableUint16_numel_get(swigCPtr, this); + } + + public static long round_capacity(long n) { + return swigfaissJNI.AlignedTableUint16_round_capacity(n); + } + + public AlignedTableUint16() { + this(swigfaissJNI.new_AlignedTableUint16__SWIG_0(), true); + } + + public AlignedTableUint16(long n) { + this(swigfaissJNI.new_AlignedTableUint16__SWIG_1(n), true); + } + + public long itemsize() { + return swigfaissJNI.AlignedTableUint16_itemsize(swigCPtr, this); + } + + public void resize(long n) { + swigfaissJNI.AlignedTableUint16_resize(swigCPtr, this, n); + } + + public void clear() { + swigfaissJNI.AlignedTableUint16_clear(swigCPtr, this); + } + + public long size() { + return swigfaissJNI.AlignedTableUint16_size(swigCPtr, this); + } + + public long nbytes() { + return swigfaissJNI.AlignedTableUint16_nbytes(swigCPtr, this); + } + + public SWIGTYPE_p_uint16_t get() { + long cPtr = swigfaissJNI.AlignedTableUint16_get__SWIG_0(swigCPtr, this); + return (cPtr == 0) ? null : new SWIGTYPE_p_uint16_t(cPtr, false); + } + + public SWIGTYPE_p_uint16_t data() { + long cPtr = swigfaissJNI.AlignedTableUint16_data__SWIG_0(swigCPtr, this); + return (cPtr == 0) ? null : new SWIGTYPE_p_uint16_t(cPtr, false); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/AlignedTableUint8.java b/ann/src/main/java/com/twitter/ann/faiss/swig/AlignedTableUint8.java new file mode 100644 index 0000000000..f1640baa8a --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/AlignedTableUint8.java @@ -0,0 +1,98 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class AlignedTableUint8 { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected AlignedTableUint8(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(AlignedTableUint8 obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_AlignedTableUint8(swigCPtr); + } + swigCPtr = 0; + } + } + + public void setTab(SWIGTYPE_p_faiss__AlignedTableTightAllocT_unsigned_char_32_t value) { + swigfaissJNI.AlignedTableUint8_tab_set(swigCPtr, this, SWIGTYPE_p_faiss__AlignedTableTightAllocT_unsigned_char_32_t.getCPtr(value)); + } + + public SWIGTYPE_p_faiss__AlignedTableTightAllocT_unsigned_char_32_t getTab() { + long cPtr = swigfaissJNI.AlignedTableUint8_tab_get(swigCPtr, this); + return (cPtr == 0) ? null : new SWIGTYPE_p_faiss__AlignedTableTightAllocT_unsigned_char_32_t(cPtr, false); + } + + public void setNumel(long value) { + swigfaissJNI.AlignedTableUint8_numel_set(swigCPtr, this, value); + } + + public long getNumel() { + return swigfaissJNI.AlignedTableUint8_numel_get(swigCPtr, this); + } + + public static long round_capacity(long n) { + return swigfaissJNI.AlignedTableUint8_round_capacity(n); + } + + public AlignedTableUint8() { + this(swigfaissJNI.new_AlignedTableUint8__SWIG_0(), true); + } + + public AlignedTableUint8(long n) { + this(swigfaissJNI.new_AlignedTableUint8__SWIG_1(n), true); + } + + public long itemsize() { + return swigfaissJNI.AlignedTableUint8_itemsize(swigCPtr, this); + } + + public void resize(long n) { + swigfaissJNI.AlignedTableUint8_resize(swigCPtr, this, n); + } + + public void clear() { + swigfaissJNI.AlignedTableUint8_clear(swigCPtr, this); + } + + public long size() { + return swigfaissJNI.AlignedTableUint8_size(swigCPtr, this); + } + + public long nbytes() { + return swigfaissJNI.AlignedTableUint8_nbytes(swigCPtr, this); + } + + public SWIGTYPE_p_unsigned_char get() { + long cPtr = swigfaissJNI.AlignedTableUint8_get__SWIG_0(swigCPtr, this); + return (cPtr == 0) ? null : new SWIGTYPE_p_unsigned_char(cPtr, false); + } + + public SWIGTYPE_p_unsigned_char data() { + long cPtr = swigfaissJNI.AlignedTableUint8_data__SWIG_0(swigCPtr, this); + return (cPtr == 0) ? null : new SWIGTYPE_p_unsigned_char(cPtr, false); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/ArrayInvertedLists.java b/ann/src/main/java/com/twitter/ann/faiss/swig/ArrayInvertedLists.java new file mode 100644 index 0000000000..8536e6549e --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/ArrayInvertedLists.java @@ -0,0 +1,86 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class ArrayInvertedLists extends InvertedLists { + private transient long swigCPtr; + + protected ArrayInvertedLists(long cPtr, boolean cMemoryOwn) { + super(swigfaissJNI.ArrayInvertedLists_SWIGUpcast(cPtr), cMemoryOwn); + swigCPtr = cPtr; + } + + protected static long getCPtr(ArrayInvertedLists obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_ArrayInvertedLists(swigCPtr); + } + swigCPtr = 0; + } + super.delete(); + } + + public void setCodes(ByteVectorVector value) { + swigfaissJNI.ArrayInvertedLists_codes_set(swigCPtr, this, ByteVectorVector.getCPtr(value), value); + } + + public ByteVectorVector getCodes() { + long cPtr = swigfaissJNI.ArrayInvertedLists_codes_get(swigCPtr, this); + return (cPtr == 0) ? null : new ByteVectorVector(cPtr, false); + } + + public void setIds(SWIGTYPE_p_std__vectorT_std__vectorT_int64_t_t_t value) { + swigfaissJNI.ArrayInvertedLists_ids_set(swigCPtr, this, SWIGTYPE_p_std__vectorT_std__vectorT_int64_t_t_t.getCPtr(value)); + } + + public SWIGTYPE_p_std__vectorT_std__vectorT_int64_t_t_t getIds() { + long cPtr = swigfaissJNI.ArrayInvertedLists_ids_get(swigCPtr, this); + return (cPtr == 0) ? null : new SWIGTYPE_p_std__vectorT_std__vectorT_int64_t_t_t(cPtr, false); + } + + public ArrayInvertedLists(long nlist, long code_size) { + this(swigfaissJNI.new_ArrayInvertedLists(nlist, code_size), true); + } + + public long list_size(long list_no) { + return swigfaissJNI.ArrayInvertedLists_list_size(swigCPtr, this, list_no); + } + + public SWIGTYPE_p_unsigned_char get_codes(long list_no) { + long cPtr = swigfaissJNI.ArrayInvertedLists_get_codes(swigCPtr, this, list_no); + return (cPtr == 0) ? null : new SWIGTYPE_p_unsigned_char(cPtr, false); + } + + public LongVector get_ids(long list_no) { + return new LongVector(swigfaissJNI.ArrayInvertedLists_get_ids(swigCPtr, this, list_no), false); +} + + public long add_entries(long list_no, long n_entry, LongVector ids, SWIGTYPE_p_unsigned_char code) { + return swigfaissJNI.ArrayInvertedLists_add_entries(swigCPtr, this, list_no, n_entry, SWIGTYPE_p_long_long.getCPtr(ids.data()), ids, SWIGTYPE_p_unsigned_char.getCPtr(code)); + } + + public void update_entries(long list_no, long offset, long n_entry, LongVector ids, SWIGTYPE_p_unsigned_char code) { + swigfaissJNI.ArrayInvertedLists_update_entries(swigCPtr, this, list_no, offset, n_entry, SWIGTYPE_p_long_long.getCPtr(ids.data()), ids, SWIGTYPE_p_unsigned_char.getCPtr(code)); + } + + public void resize(long list_no, long new_size) { + swigfaissJNI.ArrayInvertedLists_resize(swigCPtr, this, list_no, new_size); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/AutoTuneCriterion.java b/ann/src/main/java/com/twitter/ann/faiss/swig/AutoTuneCriterion.java new file mode 100644 index 0000000000..c9df33f9f8 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/AutoTuneCriterion.java @@ -0,0 +1,89 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class AutoTuneCriterion { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected AutoTuneCriterion(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(AutoTuneCriterion obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_AutoTuneCriterion(swigCPtr); + } + swigCPtr = 0; + } + } + + public void setNq(long value) { + swigfaissJNI.AutoTuneCriterion_nq_set(swigCPtr, this, value); + } + + public long getNq() { + return swigfaissJNI.AutoTuneCriterion_nq_get(swigCPtr, this); +} + + public void setNnn(long value) { + swigfaissJNI.AutoTuneCriterion_nnn_set(swigCPtr, this, value); + } + + public long getNnn() { + return swigfaissJNI.AutoTuneCriterion_nnn_get(swigCPtr, this); +} + + public void setGt_nnn(long value) { + swigfaissJNI.AutoTuneCriterion_gt_nnn_set(swigCPtr, this, value); + } + + public long getGt_nnn() { + return swigfaissJNI.AutoTuneCriterion_gt_nnn_get(swigCPtr, this); +} + + public void setGt_D(FloatVector value) { + swigfaissJNI.AutoTuneCriterion_gt_D_set(swigCPtr, this, FloatVector.getCPtr(value), value); + } + + public FloatVector getGt_D() { + long cPtr = swigfaissJNI.AutoTuneCriterion_gt_D_get(swigCPtr, this); + return (cPtr == 0) ? null : new FloatVector(cPtr, false); + } + + public void setGt_I(SWIGTYPE_p_std__vectorT_int64_t_t value) { + swigfaissJNI.AutoTuneCriterion_gt_I_set(swigCPtr, this, SWIGTYPE_p_std__vectorT_int64_t_t.getCPtr(value)); + } + + public SWIGTYPE_p_std__vectorT_int64_t_t getGt_I() { + long cPtr = swigfaissJNI.AutoTuneCriterion_gt_I_get(swigCPtr, this); + return (cPtr == 0) ? null : new SWIGTYPE_p_std__vectorT_int64_t_t(cPtr, false); + } + + public void set_groundtruth(int gt_nnn, SWIGTYPE_p_float gt_D_in, LongVector gt_I_in) { + swigfaissJNI.AutoTuneCriterion_set_groundtruth(swigCPtr, this, gt_nnn, SWIGTYPE_p_float.getCPtr(gt_D_in), SWIGTYPE_p_long_long.getCPtr(gt_I_in.data()), gt_I_in); + } + + public double evaluate(SWIGTYPE_p_float D, LongVector I) { + return swigfaissJNI.AutoTuneCriterion_evaluate(swigCPtr, this, SWIGTYPE_p_float.getCPtr(D), SWIGTYPE_p_long_long.getCPtr(I.data()), I); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/BUILD b/ann/src/main/java/com/twitter/ann/faiss/swig/BUILD new file mode 100644 index 0000000000..b8b12773aa --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/BUILD @@ -0,0 +1,26 @@ +java_library( + name = "swig-local", + sources = ["*.java"], + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = [ + "bazel-compatible", + "bazel-only", + ], + dependencies = [ + "ann/src/main/java/com/twitter/ann/faiss:swig-native-utils", + "ann/src/main/java/com/twitter/ann/faiss/swig/resources", + ], +) + +java_library( + name = "swig-artifactory", + sources = ["*.java"], + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/com/twitter/ann/faiss/swig:resources", + "ann/src/main/java/com/twitter/ann/faiss:swig-native-utils", + ], +) diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/BitstringReader.java b/ann/src/main/java/com/twitter/ann/faiss/swig/BitstringReader.java new file mode 100644 index 0000000000..042789dcaa --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/BitstringReader.java @@ -0,0 +1,72 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class BitstringReader { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected BitstringReader(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(BitstringReader obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_BitstringReader(swigCPtr); + } + swigCPtr = 0; + } + } + + public void setCode(SWIGTYPE_p_unsigned_char value) { + swigfaissJNI.BitstringReader_code_set(swigCPtr, this, SWIGTYPE_p_unsigned_char.getCPtr(value)); + } + + public SWIGTYPE_p_unsigned_char getCode() { + long cPtr = swigfaissJNI.BitstringReader_code_get(swigCPtr, this); + return (cPtr == 0) ? null : new SWIGTYPE_p_unsigned_char(cPtr, false); + } + + public void setCode_size(long value) { + swigfaissJNI.BitstringReader_code_size_set(swigCPtr, this, value); + } + + public long getCode_size() { + return swigfaissJNI.BitstringReader_code_size_get(swigCPtr, this); + } + + public void setI(long value) { + swigfaissJNI.BitstringReader_i_set(swigCPtr, this, value); + } + + public long getI() { + return swigfaissJNI.BitstringReader_i_get(swigCPtr, this); + } + + public BitstringReader(SWIGTYPE_p_unsigned_char code, long code_size) { + this(swigfaissJNI.new_BitstringReader(SWIGTYPE_p_unsigned_char.getCPtr(code), code_size), true); + } + + public long read(int nbit) { + return swigfaissJNI.BitstringReader_read(swigCPtr, this, nbit); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/BitstringWriter.java b/ann/src/main/java/com/twitter/ann/faiss/swig/BitstringWriter.java new file mode 100644 index 0000000000..8fc18d9c2d --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/BitstringWriter.java @@ -0,0 +1,72 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class BitstringWriter { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected BitstringWriter(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(BitstringWriter obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_BitstringWriter(swigCPtr); + } + swigCPtr = 0; + } + } + + public void setCode(SWIGTYPE_p_unsigned_char value) { + swigfaissJNI.BitstringWriter_code_set(swigCPtr, this, SWIGTYPE_p_unsigned_char.getCPtr(value)); + } + + public SWIGTYPE_p_unsigned_char getCode() { + long cPtr = swigfaissJNI.BitstringWriter_code_get(swigCPtr, this); + return (cPtr == 0) ? null : new SWIGTYPE_p_unsigned_char(cPtr, false); + } + + public void setCode_size(long value) { + swigfaissJNI.BitstringWriter_code_size_set(swigCPtr, this, value); + } + + public long getCode_size() { + return swigfaissJNI.BitstringWriter_code_size_get(swigCPtr, this); + } + + public void setI(long value) { + swigfaissJNI.BitstringWriter_i_set(swigCPtr, this, value); + } + + public long getI() { + return swigfaissJNI.BitstringWriter_i_get(swigCPtr, this); + } + + public BitstringWriter(SWIGTYPE_p_unsigned_char code, long code_size) { + this(swigfaissJNI.new_BitstringWriter(SWIGTYPE_p_unsigned_char.getCPtr(code), code_size), true); + } + + public void write(long x, int nbit) { + swigfaissJNI.BitstringWriter_write(swigCPtr, this, x, nbit); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/BufferList.java b/ann/src/main/java/com/twitter/ann/faiss/swig/BufferList.java new file mode 100644 index 0000000000..256fc1edde --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/BufferList.java @@ -0,0 +1,80 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class BufferList { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected BufferList(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(BufferList obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_BufferList(swigCPtr); + } + swigCPtr = 0; + } + } + + public void setBuffer_size(long value) { + swigfaissJNI.BufferList_buffer_size_set(swigCPtr, this, value); + } + + public long getBuffer_size() { + return swigfaissJNI.BufferList_buffer_size_get(swigCPtr, this); + } + + public void setBuffers(SWIGTYPE_p_std__vectorT_faiss__BufferList__Buffer_t value) { + swigfaissJNI.BufferList_buffers_set(swigCPtr, this, SWIGTYPE_p_std__vectorT_faiss__BufferList__Buffer_t.getCPtr(value)); + } + + public SWIGTYPE_p_std__vectorT_faiss__BufferList__Buffer_t getBuffers() { + long cPtr = swigfaissJNI.BufferList_buffers_get(swigCPtr, this); + return (cPtr == 0) ? null : new SWIGTYPE_p_std__vectorT_faiss__BufferList__Buffer_t(cPtr, false); + } + + public void setWp(long value) { + swigfaissJNI.BufferList_wp_set(swigCPtr, this, value); + } + + public long getWp() { + return swigfaissJNI.BufferList_wp_get(swigCPtr, this); + } + + public BufferList(long buffer_size) { + this(swigfaissJNI.new_BufferList(buffer_size), true); + } + + public void append_buffer() { + swigfaissJNI.BufferList_append_buffer(swigCPtr, this); + } + + public void add(long id, float dis) { + swigfaissJNI.BufferList_add(swigCPtr, this, id, dis); + } + + public void copy_range(long ofs, long n, LongVector dest_ids, SWIGTYPE_p_float dest_dis) { + swigfaissJNI.BufferList_copy_range(swigCPtr, this, ofs, n, SWIGTYPE_p_long_long.getCPtr(dest_ids.data()), dest_ids, SWIGTYPE_p_float.getCPtr(dest_dis)); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/ByteVector.java b/ann/src/main/java/com/twitter/ann/faiss/swig/ByteVector.java new file mode 100644 index 0000000000..f439dfa721 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/ByteVector.java @@ -0,0 +1,76 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class ByteVector { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected ByteVector(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(ByteVector obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_ByteVector(swigCPtr); + } + swigCPtr = 0; + } + } + + public ByteVector() { + this(swigfaissJNI.new_ByteVector(), true); + } + + public void push_back(short arg0) { + swigfaissJNI.ByteVector_push_back(swigCPtr, this, arg0); + } + + public void clear() { + swigfaissJNI.ByteVector_clear(swigCPtr, this); + } + + public SWIGTYPE_p_unsigned_char data() { + long cPtr = swigfaissJNI.ByteVector_data(swigCPtr, this); + return (cPtr == 0) ? null : new SWIGTYPE_p_unsigned_char(cPtr, false); + } + + public long size() { + return swigfaissJNI.ByteVector_size(swigCPtr, this); + } + + public short at(long n) { + return swigfaissJNI.ByteVector_at(swigCPtr, this, n); + } + + public void resize(long n) { + swigfaissJNI.ByteVector_resize(swigCPtr, this, n); + } + + public void reserve(long n) { + swigfaissJNI.ByteVector_reserve(swigCPtr, this, n); + } + + public void swap(ByteVector other) { + swigfaissJNI.ByteVector_swap(swigCPtr, this, ByteVector.getCPtr(other), other); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/ByteVectorVector.java b/ann/src/main/java/com/twitter/ann/faiss/swig/ByteVectorVector.java new file mode 100644 index 0000000000..fa0b3a7cc0 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/ByteVectorVector.java @@ -0,0 +1,76 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class ByteVectorVector { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected ByteVectorVector(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(ByteVectorVector obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_ByteVectorVector(swigCPtr); + } + swigCPtr = 0; + } + } + + public ByteVectorVector() { + this(swigfaissJNI.new_ByteVectorVector(), true); + } + + public void push_back(ByteVector arg0) { + swigfaissJNI.ByteVectorVector_push_back(swigCPtr, this, ByteVector.getCPtr(arg0), arg0); + } + + public void clear() { + swigfaissJNI.ByteVectorVector_clear(swigCPtr, this); + } + + public ByteVector data() { + long cPtr = swigfaissJNI.ByteVectorVector_data(swigCPtr, this); + return (cPtr == 0) ? null : new ByteVector(cPtr, false); + } + + public long size() { + return swigfaissJNI.ByteVectorVector_size(swigCPtr, this); + } + + public ByteVector at(long n) { + return new ByteVector(swigfaissJNI.ByteVectorVector_at(swigCPtr, this, n), true); + } + + public void resize(long n) { + swigfaissJNI.ByteVectorVector_resize(swigCPtr, this, n); + } + + public void reserve(long n) { + swigfaissJNI.ByteVectorVector_reserve(swigCPtr, this, n); + } + + public void swap(ByteVectorVector other) { + swigfaissJNI.ByteVectorVector_swap(swigCPtr, this, ByteVectorVector.getCPtr(other), other); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/CenteringTransform.java b/ann/src/main/java/com/twitter/ann/faiss/swig/CenteringTransform.java new file mode 100644 index 0000000000..e9abaf61ab --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/CenteringTransform.java @@ -0,0 +1,68 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class CenteringTransform extends VectorTransform { + private transient long swigCPtr; + + protected CenteringTransform(long cPtr, boolean cMemoryOwn) { + super(swigfaissJNI.CenteringTransform_SWIGUpcast(cPtr), cMemoryOwn); + swigCPtr = cPtr; + } + + protected static long getCPtr(CenteringTransform obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_CenteringTransform(swigCPtr); + } + swigCPtr = 0; + } + super.delete(); + } + + public void setMean(FloatVector value) { + swigfaissJNI.CenteringTransform_mean_set(swigCPtr, this, FloatVector.getCPtr(value), value); + } + + public FloatVector getMean() { + long cPtr = swigfaissJNI.CenteringTransform_mean_get(swigCPtr, this); + return (cPtr == 0) ? null : new FloatVector(cPtr, false); + } + + public CenteringTransform(int d) { + this(swigfaissJNI.new_CenteringTransform__SWIG_0(d), true); + } + + public CenteringTransform() { + this(swigfaissJNI.new_CenteringTransform__SWIG_1(), true); + } + + public void train(long n, SWIGTYPE_p_float x) { + swigfaissJNI.CenteringTransform_train(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x)); + } + + public void apply_noalloc(long n, SWIGTYPE_p_float x, SWIGTYPE_p_float xt) { + swigfaissJNI.CenteringTransform_apply_noalloc(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x), SWIGTYPE_p_float.getCPtr(xt)); + } + + public void reverse_transform(long n, SWIGTYPE_p_float xt, SWIGTYPE_p_float x) { + swigfaissJNI.CenteringTransform_reverse_transform(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(xt), SWIGTYPE_p_float.getCPtr(x)); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/CharVector.java b/ann/src/main/java/com/twitter/ann/faiss/swig/CharVector.java new file mode 100644 index 0000000000..e1b91127c2 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/CharVector.java @@ -0,0 +1,75 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class CharVector { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected CharVector(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(CharVector obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_CharVector(swigCPtr); + } + swigCPtr = 0; + } + } + + public CharVector() { + this(swigfaissJNI.new_CharVector(), true); + } + + public void push_back(char arg0) { + swigfaissJNI.CharVector_push_back(swigCPtr, this, arg0); + } + + public void clear() { + swigfaissJNI.CharVector_clear(swigCPtr, this); + } + + public String data() { + return swigfaissJNI.CharVector_data(swigCPtr, this); + } + + public long size() { + return swigfaissJNI.CharVector_size(swigCPtr, this); + } + + public char at(long n) { + return swigfaissJNI.CharVector_at(swigCPtr, this, n); + } + + public void resize(long n) { + swigfaissJNI.CharVector_resize(swigCPtr, this, n); + } + + public void reserve(long n) { + swigfaissJNI.CharVector_reserve(swigCPtr, this, n); + } + + public void swap(CharVector other) { + swigfaissJNI.CharVector_swap(swigCPtr, this, CharVector.getCPtr(other), other); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/Clustering.java b/ann/src/main/java/com/twitter/ann/faiss/swig/Clustering.java new file mode 100644 index 0000000000..d8fe517287 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/Clustering.java @@ -0,0 +1,101 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class Clustering extends ClusteringParameters { + private transient long swigCPtr; + + protected Clustering(long cPtr, boolean cMemoryOwn) { + super(swigfaissJNI.Clustering_SWIGUpcast(cPtr), cMemoryOwn); + swigCPtr = cPtr; + } + + protected static long getCPtr(Clustering obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_Clustering(swigCPtr); + } + swigCPtr = 0; + } + super.delete(); + } + + public void setD(long value) { + swigfaissJNI.Clustering_d_set(swigCPtr, this, value); + } + + public long getD() { + return swigfaissJNI.Clustering_d_get(swigCPtr, this); + } + + public void setK(long value) { + swigfaissJNI.Clustering_k_set(swigCPtr, this, value); + } + + public long getK() { + return swigfaissJNI.Clustering_k_get(swigCPtr, this); + } + + public void setCentroids(FloatVector value) { + swigfaissJNI.Clustering_centroids_set(swigCPtr, this, FloatVector.getCPtr(value), value); + } + + public FloatVector getCentroids() { + long cPtr = swigfaissJNI.Clustering_centroids_get(swigCPtr, this); + return (cPtr == 0) ? null : new FloatVector(cPtr, false); + } + + public void setIteration_stats(SWIGTYPE_p_std__vectorT_faiss__ClusteringIterationStats_t value) { + swigfaissJNI.Clustering_iteration_stats_set(swigCPtr, this, SWIGTYPE_p_std__vectorT_faiss__ClusteringIterationStats_t.getCPtr(value)); + } + + public SWIGTYPE_p_std__vectorT_faiss__ClusteringIterationStats_t getIteration_stats() { + long cPtr = swigfaissJNI.Clustering_iteration_stats_get(swigCPtr, this); + return (cPtr == 0) ? null : new SWIGTYPE_p_std__vectorT_faiss__ClusteringIterationStats_t(cPtr, false); + } + + public Clustering(int d, int k) { + this(swigfaissJNI.new_Clustering__SWIG_0(d, k), true); + } + + public Clustering(int d, int k, ClusteringParameters cp) { + this(swigfaissJNI.new_Clustering__SWIG_1(d, k, ClusteringParameters.getCPtr(cp), cp), true); + } + + public void train(long n, SWIGTYPE_p_float x, Index index, SWIGTYPE_p_float x_weights) { + swigfaissJNI.Clustering_train__SWIG_0(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x), Index.getCPtr(index), index, SWIGTYPE_p_float.getCPtr(x_weights)); + } + + public void train(long n, SWIGTYPE_p_float x, Index index) { + swigfaissJNI.Clustering_train__SWIG_1(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x), Index.getCPtr(index), index); + } + + public void train_encoded(long nx, SWIGTYPE_p_unsigned_char x_in, Index codec, Index index, SWIGTYPE_p_float weights) { + swigfaissJNI.Clustering_train_encoded__SWIG_0(swigCPtr, this, nx, SWIGTYPE_p_unsigned_char.getCPtr(x_in), Index.getCPtr(codec), codec, Index.getCPtr(index), index, SWIGTYPE_p_float.getCPtr(weights)); + } + + public void train_encoded(long nx, SWIGTYPE_p_unsigned_char x_in, Index codec, Index index) { + swigfaissJNI.Clustering_train_encoded__SWIG_1(swigCPtr, this, nx, SWIGTYPE_p_unsigned_char.getCPtr(x_in), Index.getCPtr(codec), codec, Index.getCPtr(index), index); + } + + public void post_process_centroids() { + swigfaissJNI.Clustering_post_process_centroids(swigCPtr, this); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/Clustering1D.java b/ann/src/main/java/com/twitter/ann/faiss/swig/Clustering1D.java new file mode 100644 index 0000000000..8d4bc658c3 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/Clustering1D.java @@ -0,0 +1,51 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class Clustering1D extends Clustering { + private transient long swigCPtr; + + protected Clustering1D(long cPtr, boolean cMemoryOwn) { + super(swigfaissJNI.Clustering1D_SWIGUpcast(cPtr), cMemoryOwn); + swigCPtr = cPtr; + } + + protected static long getCPtr(Clustering1D obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_Clustering1D(swigCPtr); + } + swigCPtr = 0; + } + super.delete(); + } + + public Clustering1D(int k) { + this(swigfaissJNI.new_Clustering1D__SWIG_0(k), true); + } + + public Clustering1D(int k, ClusteringParameters cp) { + this(swigfaissJNI.new_Clustering1D__SWIG_1(k, ClusteringParameters.getCPtr(cp), cp), true); + } + + public void train_exact(long n, SWIGTYPE_p_float x) { + swigfaissJNI.Clustering1D_train_exact(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x)); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/ClusteringIterationStats.java b/ann/src/main/java/com/twitter/ann/faiss/swig/ClusteringIterationStats.java new file mode 100644 index 0000000000..b0fcb5d09b --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/ClusteringIterationStats.java @@ -0,0 +1,83 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class ClusteringIterationStats { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected ClusteringIterationStats(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(ClusteringIterationStats obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_ClusteringIterationStats(swigCPtr); + } + swigCPtr = 0; + } + } + + public void setObj(float value) { + swigfaissJNI.ClusteringIterationStats_obj_set(swigCPtr, this, value); + } + + public float getObj() { + return swigfaissJNI.ClusteringIterationStats_obj_get(swigCPtr, this); + } + + public void setTime(double value) { + swigfaissJNI.ClusteringIterationStats_time_set(swigCPtr, this, value); + } + + public double getTime() { + return swigfaissJNI.ClusteringIterationStats_time_get(swigCPtr, this); + } + + public void setTime_search(double value) { + swigfaissJNI.ClusteringIterationStats_time_search_set(swigCPtr, this, value); + } + + public double getTime_search() { + return swigfaissJNI.ClusteringIterationStats_time_search_get(swigCPtr, this); + } + + public void setImbalance_factor(double value) { + swigfaissJNI.ClusteringIterationStats_imbalance_factor_set(swigCPtr, this, value); + } + + public double getImbalance_factor() { + return swigfaissJNI.ClusteringIterationStats_imbalance_factor_get(swigCPtr, this); + } + + public void setNsplit(int value) { + swigfaissJNI.ClusteringIterationStats_nsplit_set(swigCPtr, this, value); + } + + public int getNsplit() { + return swigfaissJNI.ClusteringIterationStats_nsplit_get(swigCPtr, this); + } + + public ClusteringIterationStats() { + this(swigfaissJNI.new_ClusteringIterationStats(), true); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/ClusteringParameters.java b/ann/src/main/java/com/twitter/ann/faiss/swig/ClusteringParameters.java new file mode 100644 index 0000000000..3c52d810f3 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/ClusteringParameters.java @@ -0,0 +1,131 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class ClusteringParameters { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected ClusteringParameters(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(ClusteringParameters obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_ClusteringParameters(swigCPtr); + } + swigCPtr = 0; + } + } + + public void setNiter(int value) { + swigfaissJNI.ClusteringParameters_niter_set(swigCPtr, this, value); + } + + public int getNiter() { + return swigfaissJNI.ClusteringParameters_niter_get(swigCPtr, this); + } + + public void setNredo(int value) { + swigfaissJNI.ClusteringParameters_nredo_set(swigCPtr, this, value); + } + + public int getNredo() { + return swigfaissJNI.ClusteringParameters_nredo_get(swigCPtr, this); + } + + public void setVerbose(boolean value) { + swigfaissJNI.ClusteringParameters_verbose_set(swigCPtr, this, value); + } + + public boolean getVerbose() { + return swigfaissJNI.ClusteringParameters_verbose_get(swigCPtr, this); + } + + public void setSpherical(boolean value) { + swigfaissJNI.ClusteringParameters_spherical_set(swigCPtr, this, value); + } + + public boolean getSpherical() { + return swigfaissJNI.ClusteringParameters_spherical_get(swigCPtr, this); + } + + public void setInt_centroids(boolean value) { + swigfaissJNI.ClusteringParameters_int_centroids_set(swigCPtr, this, value); + } + + public boolean getInt_centroids() { + return swigfaissJNI.ClusteringParameters_int_centroids_get(swigCPtr, this); + } + + public void setUpdate_index(boolean value) { + swigfaissJNI.ClusteringParameters_update_index_set(swigCPtr, this, value); + } + + public boolean getUpdate_index() { + return swigfaissJNI.ClusteringParameters_update_index_get(swigCPtr, this); + } + + public void setFrozen_centroids(boolean value) { + swigfaissJNI.ClusteringParameters_frozen_centroids_set(swigCPtr, this, value); + } + + public boolean getFrozen_centroids() { + return swigfaissJNI.ClusteringParameters_frozen_centroids_get(swigCPtr, this); + } + + public void setMin_points_per_centroid(int value) { + swigfaissJNI.ClusteringParameters_min_points_per_centroid_set(swigCPtr, this, value); + } + + public int getMin_points_per_centroid() { + return swigfaissJNI.ClusteringParameters_min_points_per_centroid_get(swigCPtr, this); + } + + public void setMax_points_per_centroid(int value) { + swigfaissJNI.ClusteringParameters_max_points_per_centroid_set(swigCPtr, this, value); + } + + public int getMax_points_per_centroid() { + return swigfaissJNI.ClusteringParameters_max_points_per_centroid_get(swigCPtr, this); + } + + public void setSeed(int value) { + swigfaissJNI.ClusteringParameters_seed_set(swigCPtr, this, value); + } + + public int getSeed() { + return swigfaissJNI.ClusteringParameters_seed_get(swigCPtr, this); + } + + public void setDecode_block_size(long value) { + swigfaissJNI.ClusteringParameters_decode_block_size_set(swigCPtr, this, value); + } + + public long getDecode_block_size() { + return swigfaissJNI.ClusteringParameters_decode_block_size_get(swigCPtr, this); + } + + public ClusteringParameters() { + this(swigfaissJNI.new_ClusteringParameters(), true); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/DistanceComputer.java b/ann/src/main/java/com/twitter/ann/faiss/swig/DistanceComputer.java new file mode 100644 index 0000000000..251ede16f1 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/DistanceComputer.java @@ -0,0 +1,47 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class DistanceComputer { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected DistanceComputer(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(DistanceComputer obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_DistanceComputer(swigCPtr); + } + swigCPtr = 0; + } + } + + public void set_query(SWIGTYPE_p_float x) { + swigfaissJNI.DistanceComputer_set_query(swigCPtr, this, SWIGTYPE_p_float.getCPtr(x)); + } + + public float symmetric_dis(long i, long j) { + return swigfaissJNI.DistanceComputer_symmetric_dis(swigCPtr, this, i, j); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/DoubleVector.java b/ann/src/main/java/com/twitter/ann/faiss/swig/DoubleVector.java new file mode 100644 index 0000000000..c58001498f --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/DoubleVector.java @@ -0,0 +1,76 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class DoubleVector { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected DoubleVector(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(DoubleVector obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_DoubleVector(swigCPtr); + } + swigCPtr = 0; + } + } + + public DoubleVector() { + this(swigfaissJNI.new_DoubleVector(), true); + } + + public void push_back(double arg0) { + swigfaissJNI.DoubleVector_push_back(swigCPtr, this, arg0); + } + + public void clear() { + swigfaissJNI.DoubleVector_clear(swigCPtr, this); + } + + public SWIGTYPE_p_double data() { + long cPtr = swigfaissJNI.DoubleVector_data(swigCPtr, this); + return (cPtr == 0) ? null : new SWIGTYPE_p_double(cPtr, false); + } + + public long size() { + return swigfaissJNI.DoubleVector_size(swigCPtr, this); + } + + public double at(long n) { + return swigfaissJNI.DoubleVector_at(swigCPtr, this, n); + } + + public void resize(long n) { + swigfaissJNI.DoubleVector_resize(swigCPtr, this, n); + } + + public void reserve(long n) { + swigfaissJNI.DoubleVector_reserve(swigCPtr, this, n); + } + + public void swap(DoubleVector other) { + swigfaissJNI.DoubleVector_swap(swigCPtr, this, DoubleVector.getCPtr(other), other); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/FloatVector.java b/ann/src/main/java/com/twitter/ann/faiss/swig/FloatVector.java new file mode 100644 index 0000000000..7374ce3a23 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/FloatVector.java @@ -0,0 +1,76 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class FloatVector { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected FloatVector(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(FloatVector obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_FloatVector(swigCPtr); + } + swigCPtr = 0; + } + } + + public FloatVector() { + this(swigfaissJNI.new_FloatVector(), true); + } + + public void push_back(float arg0) { + swigfaissJNI.FloatVector_push_back(swigCPtr, this, arg0); + } + + public void clear() { + swigfaissJNI.FloatVector_clear(swigCPtr, this); + } + + public SWIGTYPE_p_float data() { + long cPtr = swigfaissJNI.FloatVector_data(swigCPtr, this); + return (cPtr == 0) ? null : new SWIGTYPE_p_float(cPtr, false); + } + + public long size() { + return swigfaissJNI.FloatVector_size(swigCPtr, this); + } + + public float at(long n) { + return swigfaissJNI.FloatVector_at(swigCPtr, this, n); + } + + public void resize(long n) { + swigfaissJNI.FloatVector_resize(swigCPtr, this, n); + } + + public void reserve(long n) { + swigfaissJNI.FloatVector_reserve(swigCPtr, this, n); + } + + public void swap(FloatVector other) { + swigfaissJNI.FloatVector_swap(swigCPtr, this, FloatVector.getCPtr(other), other); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/FloatVectorVector.java b/ann/src/main/java/com/twitter/ann/faiss/swig/FloatVectorVector.java new file mode 100644 index 0000000000..2aa7afbd2a --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/FloatVectorVector.java @@ -0,0 +1,76 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class FloatVectorVector { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected FloatVectorVector(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(FloatVectorVector obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_FloatVectorVector(swigCPtr); + } + swigCPtr = 0; + } + } + + public FloatVectorVector() { + this(swigfaissJNI.new_FloatVectorVector(), true); + } + + public void push_back(FloatVector arg0) { + swigfaissJNI.FloatVectorVector_push_back(swigCPtr, this, FloatVector.getCPtr(arg0), arg0); + } + + public void clear() { + swigfaissJNI.FloatVectorVector_clear(swigCPtr, this); + } + + public FloatVector data() { + long cPtr = swigfaissJNI.FloatVectorVector_data(swigCPtr, this); + return (cPtr == 0) ? null : new FloatVector(cPtr, false); + } + + public long size() { + return swigfaissJNI.FloatVectorVector_size(swigCPtr, this); + } + + public FloatVector at(long n) { + return new FloatVector(swigfaissJNI.FloatVectorVector_at(swigCPtr, this, n), true); + } + + public void resize(long n) { + swigfaissJNI.FloatVectorVector_resize(swigCPtr, this, n); + } + + public void reserve(long n) { + swigfaissJNI.FloatVectorVector_reserve(swigCPtr, this, n); + } + + public void swap(FloatVectorVector other) { + swigfaissJNI.FloatVectorVector_swap(swigCPtr, this, FloatVectorVector.getCPtr(other), other); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/GenHammingComputer16.java b/ann/src/main/java/com/twitter/ann/faiss/swig/GenHammingComputer16.java new file mode 100644 index 0000000000..2986d07de7 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/GenHammingComputer16.java @@ -0,0 +1,63 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class GenHammingComputer16 { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected GenHammingComputer16(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(GenHammingComputer16 obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_GenHammingComputer16(swigCPtr); + } + swigCPtr = 0; + } + } + + public void setA0(long value) { + swigfaissJNI.GenHammingComputer16_a0_set(swigCPtr, this, value); + } + + public long getA0() { + return swigfaissJNI.GenHammingComputer16_a0_get(swigCPtr, this); + } + + public void setA1(long value) { + swigfaissJNI.GenHammingComputer16_a1_set(swigCPtr, this, value); + } + + public long getA1() { + return swigfaissJNI.GenHammingComputer16_a1_get(swigCPtr, this); + } + + public GenHammingComputer16(SWIGTYPE_p_unsigned_char a8, int code_size) { + this(swigfaissJNI.new_GenHammingComputer16(SWIGTYPE_p_unsigned_char.getCPtr(a8), code_size), true); + } + + public int hamming(SWIGTYPE_p_unsigned_char b8) { + return swigfaissJNI.GenHammingComputer16_hamming(swigCPtr, this, SWIGTYPE_p_unsigned_char.getCPtr(b8)); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/GenHammingComputer32.java b/ann/src/main/java/com/twitter/ann/faiss/swig/GenHammingComputer32.java new file mode 100644 index 0000000000..284a6ac8c0 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/GenHammingComputer32.java @@ -0,0 +1,79 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class GenHammingComputer32 { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected GenHammingComputer32(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(GenHammingComputer32 obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_GenHammingComputer32(swigCPtr); + } + swigCPtr = 0; + } + } + + public void setA0(long value) { + swigfaissJNI.GenHammingComputer32_a0_set(swigCPtr, this, value); + } + + public long getA0() { + return swigfaissJNI.GenHammingComputer32_a0_get(swigCPtr, this); + } + + public void setA1(long value) { + swigfaissJNI.GenHammingComputer32_a1_set(swigCPtr, this, value); + } + + public long getA1() { + return swigfaissJNI.GenHammingComputer32_a1_get(swigCPtr, this); + } + + public void setA2(long value) { + swigfaissJNI.GenHammingComputer32_a2_set(swigCPtr, this, value); + } + + public long getA2() { + return swigfaissJNI.GenHammingComputer32_a2_get(swigCPtr, this); + } + + public void setA3(long value) { + swigfaissJNI.GenHammingComputer32_a3_set(swigCPtr, this, value); + } + + public long getA3() { + return swigfaissJNI.GenHammingComputer32_a3_get(swigCPtr, this); + } + + public GenHammingComputer32(SWIGTYPE_p_unsigned_char a8, int code_size) { + this(swigfaissJNI.new_GenHammingComputer32(SWIGTYPE_p_unsigned_char.getCPtr(a8), code_size), true); + } + + public int hamming(SWIGTYPE_p_unsigned_char b8) { + return swigfaissJNI.GenHammingComputer32_hamming(swigCPtr, this, SWIGTYPE_p_unsigned_char.getCPtr(b8)); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/GenHammingComputer8.java b/ann/src/main/java/com/twitter/ann/faiss/swig/GenHammingComputer8.java new file mode 100644 index 0000000000..063b873dff --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/GenHammingComputer8.java @@ -0,0 +1,55 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class GenHammingComputer8 { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected GenHammingComputer8(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(GenHammingComputer8 obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_GenHammingComputer8(swigCPtr); + } + swigCPtr = 0; + } + } + + public void setA0(long value) { + swigfaissJNI.GenHammingComputer8_a0_set(swigCPtr, this, value); + } + + public long getA0() { + return swigfaissJNI.GenHammingComputer8_a0_get(swigCPtr, this); + } + + public GenHammingComputer8(SWIGTYPE_p_unsigned_char a, int code_size) { + this(swigfaissJNI.new_GenHammingComputer8(SWIGTYPE_p_unsigned_char.getCPtr(a), code_size), true); + } + + public int hamming(SWIGTYPE_p_unsigned_char b) { + return swigfaissJNI.GenHammingComputer8_hamming(swigCPtr, this, SWIGTYPE_p_unsigned_char.getCPtr(b)); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/GenHammingComputerM8.java b/ann/src/main/java/com/twitter/ann/faiss/swig/GenHammingComputerM8.java new file mode 100644 index 0000000000..89ba44588d --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/GenHammingComputerM8.java @@ -0,0 +1,64 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class GenHammingComputerM8 { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected GenHammingComputerM8(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(GenHammingComputerM8 obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_GenHammingComputerM8(swigCPtr); + } + swigCPtr = 0; + } + } + + public void setA(SWIGTYPE_p_unsigned_long value) { + swigfaissJNI.GenHammingComputerM8_a_set(swigCPtr, this, SWIGTYPE_p_unsigned_long.getCPtr(value)); + } + + public SWIGTYPE_p_unsigned_long getA() { + long cPtr = swigfaissJNI.GenHammingComputerM8_a_get(swigCPtr, this); + return (cPtr == 0) ? null : new SWIGTYPE_p_unsigned_long(cPtr, false); + } + + public void setN(int value) { + swigfaissJNI.GenHammingComputerM8_n_set(swigCPtr, this, value); + } + + public int getN() { + return swigfaissJNI.GenHammingComputerM8_n_get(swigCPtr, this); + } + + public GenHammingComputerM8(SWIGTYPE_p_unsigned_char a8, int code_size) { + this(swigfaissJNI.new_GenHammingComputerM8(SWIGTYPE_p_unsigned_char.getCPtr(a8), code_size), true); + } + + public int hamming(SWIGTYPE_p_unsigned_char b8) { + return swigfaissJNI.GenHammingComputerM8_hamming(swigCPtr, this, SWIGTYPE_p_unsigned_char.getCPtr(b8)); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/HNSW.java b/ann/src/main/java/com/twitter/ann/faiss/swig/HNSW.java new file mode 100644 index 0000000000..5be46f0428 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/HNSW.java @@ -0,0 +1,437 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class HNSW { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected HNSW(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(HNSW obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_HNSW(swigCPtr); + } + swigCPtr = 0; + } + } + + static public class MinimaxHeap { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected MinimaxHeap(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(MinimaxHeap obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_HNSW_MinimaxHeap(swigCPtr); + } + swigCPtr = 0; + } + } + + public void setN(int value) { + swigfaissJNI.HNSW_MinimaxHeap_n_set(swigCPtr, this, value); + } + + public int getN() { + return swigfaissJNI.HNSW_MinimaxHeap_n_get(swigCPtr, this); + } + + public void setK(int value) { + swigfaissJNI.HNSW_MinimaxHeap_k_set(swigCPtr, this, value); + } + + public int getK() { + return swigfaissJNI.HNSW_MinimaxHeap_k_get(swigCPtr, this); + } + + public void setNvalid(int value) { + swigfaissJNI.HNSW_MinimaxHeap_nvalid_set(swigCPtr, this, value); + } + + public int getNvalid() { + return swigfaissJNI.HNSW_MinimaxHeap_nvalid_get(swigCPtr, this); + } + + public void setIds(IntVector value) { + swigfaissJNI.HNSW_MinimaxHeap_ids_set(swigCPtr, this, IntVector.getCPtr(value), value); + } + + public IntVector getIds() { + long cPtr = swigfaissJNI.HNSW_MinimaxHeap_ids_get(swigCPtr, this); + return (cPtr == 0) ? null : new IntVector(cPtr, false); + } + + public void setDis(FloatVector value) { + swigfaissJNI.HNSW_MinimaxHeap_dis_set(swigCPtr, this, FloatVector.getCPtr(value), value); + } + + public FloatVector getDis() { + long cPtr = swigfaissJNI.HNSW_MinimaxHeap_dis_get(swigCPtr, this); + return (cPtr == 0) ? null : new FloatVector(cPtr, false); + } + + public MinimaxHeap(int n) { + this(swigfaissJNI.new_HNSW_MinimaxHeap(n), true); + } + + public void push(int i, float v) { + swigfaissJNI.HNSW_MinimaxHeap_push(swigCPtr, this, i, v); + } + + public float max() { + return swigfaissJNI.HNSW_MinimaxHeap_max(swigCPtr, this); + } + + public int size() { + return swigfaissJNI.HNSW_MinimaxHeap_size(swigCPtr, this); + } + + public void clear() { + swigfaissJNI.HNSW_MinimaxHeap_clear(swigCPtr, this); + } + + public int pop_min(SWIGTYPE_p_float vmin_out) { + return swigfaissJNI.HNSW_MinimaxHeap_pop_min__SWIG_0(swigCPtr, this, SWIGTYPE_p_float.getCPtr(vmin_out)); + } + + public int pop_min() { + return swigfaissJNI.HNSW_MinimaxHeap_pop_min__SWIG_1(swigCPtr, this); + } + + public int count_below(float thresh) { + return swigfaissJNI.HNSW_MinimaxHeap_count_below(swigCPtr, this, thresh); + } + + } + + static public class NodeDistCloser { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected NodeDistCloser(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(NodeDistCloser obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_HNSW_NodeDistCloser(swigCPtr); + } + swigCPtr = 0; + } + } + + public void setD(float value) { + swigfaissJNI.HNSW_NodeDistCloser_d_set(swigCPtr, this, value); + } + + public float getD() { + return swigfaissJNI.HNSW_NodeDistCloser_d_get(swigCPtr, this); + } + + public void setId(int value) { + swigfaissJNI.HNSW_NodeDistCloser_id_set(swigCPtr, this, value); + } + + public int getId() { + return swigfaissJNI.HNSW_NodeDistCloser_id_get(swigCPtr, this); + } + + public NodeDistCloser(float d, int id) { + this(swigfaissJNI.new_HNSW_NodeDistCloser(d, id), true); + } + + } + + static public class NodeDistFarther { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected NodeDistFarther(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(NodeDistFarther obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_HNSW_NodeDistFarther(swigCPtr); + } + swigCPtr = 0; + } + } + + public void setD(float value) { + swigfaissJNI.HNSW_NodeDistFarther_d_set(swigCPtr, this, value); + } + + public float getD() { + return swigfaissJNI.HNSW_NodeDistFarther_d_get(swigCPtr, this); + } + + public void setId(int value) { + swigfaissJNI.HNSW_NodeDistFarther_id_set(swigCPtr, this, value); + } + + public int getId() { + return swigfaissJNI.HNSW_NodeDistFarther_id_get(swigCPtr, this); + } + + public NodeDistFarther(float d, int id) { + this(swigfaissJNI.new_HNSW_NodeDistFarther(d, id), true); + } + + } + + public void setAssign_probas(DoubleVector value) { + swigfaissJNI.HNSW_assign_probas_set(swigCPtr, this, DoubleVector.getCPtr(value), value); + } + + public DoubleVector getAssign_probas() { + long cPtr = swigfaissJNI.HNSW_assign_probas_get(swigCPtr, this); + return (cPtr == 0) ? null : new DoubleVector(cPtr, false); + } + + public void setCum_nneighbor_per_level(IntVector value) { + swigfaissJNI.HNSW_cum_nneighbor_per_level_set(swigCPtr, this, IntVector.getCPtr(value), value); + } + + public IntVector getCum_nneighbor_per_level() { + long cPtr = swigfaissJNI.HNSW_cum_nneighbor_per_level_get(swigCPtr, this); + return (cPtr == 0) ? null : new IntVector(cPtr, false); + } + + public void setLevels(IntVector value) { + swigfaissJNI.HNSW_levels_set(swigCPtr, this, IntVector.getCPtr(value), value); + } + + public IntVector getLevels() { + long cPtr = swigfaissJNI.HNSW_levels_get(swigCPtr, this); + return (cPtr == 0) ? null : new IntVector(cPtr, false); + } + + public void setOffsets(Uint64Vector value) { + swigfaissJNI.HNSW_offsets_set(swigCPtr, this, Uint64Vector.getCPtr(value), value); + } + + public Uint64Vector getOffsets() { + long cPtr = swigfaissJNI.HNSW_offsets_get(swigCPtr, this); + return (cPtr == 0) ? null : new Uint64Vector(cPtr, false); + } + + public void setNeighbors(IntVector value) { + swigfaissJNI.HNSW_neighbors_set(swigCPtr, this, IntVector.getCPtr(value), value); + } + + public IntVector getNeighbors() { + long cPtr = swigfaissJNI.HNSW_neighbors_get(swigCPtr, this); + return (cPtr == 0) ? null : new IntVector(cPtr, false); + } + + public void setEntry_point(int value) { + swigfaissJNI.HNSW_entry_point_set(swigCPtr, this, value); + } + + public int getEntry_point() { + return swigfaissJNI.HNSW_entry_point_get(swigCPtr, this); + } + + public void setRng(SWIGTYPE_p_faiss__RandomGenerator value) { + swigfaissJNI.HNSW_rng_set(swigCPtr, this, SWIGTYPE_p_faiss__RandomGenerator.getCPtr(value)); + } + + public SWIGTYPE_p_faiss__RandomGenerator getRng() { + long cPtr = swigfaissJNI.HNSW_rng_get(swigCPtr, this); + return (cPtr == 0) ? null : new SWIGTYPE_p_faiss__RandomGenerator(cPtr, false); + } + + public void setMax_level(int value) { + swigfaissJNI.HNSW_max_level_set(swigCPtr, this, value); + } + + public int getMax_level() { + return swigfaissJNI.HNSW_max_level_get(swigCPtr, this); + } + + public void setEfConstruction(int value) { + swigfaissJNI.HNSW_efConstruction_set(swigCPtr, this, value); + } + + public int getEfConstruction() { + return swigfaissJNI.HNSW_efConstruction_get(swigCPtr, this); + } + + public void setEfSearch(int value) { + swigfaissJNI.HNSW_efSearch_set(swigCPtr, this, value); + } + + public int getEfSearch() { + return swigfaissJNI.HNSW_efSearch_get(swigCPtr, this); + } + + public void setCheck_relative_distance(boolean value) { + swigfaissJNI.HNSW_check_relative_distance_set(swigCPtr, this, value); + } + + public boolean getCheck_relative_distance() { + return swigfaissJNI.HNSW_check_relative_distance_get(swigCPtr, this); + } + + public void setUpper_beam(int value) { + swigfaissJNI.HNSW_upper_beam_set(swigCPtr, this, value); + } + + public int getUpper_beam() { + return swigfaissJNI.HNSW_upper_beam_get(swigCPtr, this); + } + + public void setSearch_bounded_queue(boolean value) { + swigfaissJNI.HNSW_search_bounded_queue_set(swigCPtr, this, value); + } + + public boolean getSearch_bounded_queue() { + return swigfaissJNI.HNSW_search_bounded_queue_get(swigCPtr, this); + } + + public void set_default_probas(int M, float levelMult) { + swigfaissJNI.HNSW_set_default_probas(swigCPtr, this, M, levelMult); + } + + public void set_nb_neighbors(int level_no, int n) { + swigfaissJNI.HNSW_set_nb_neighbors(swigCPtr, this, level_no, n); + } + + public int nb_neighbors(int layer_no) { + return swigfaissJNI.HNSW_nb_neighbors(swigCPtr, this, layer_no); + } + + public int cum_nb_neighbors(int layer_no) { + return swigfaissJNI.HNSW_cum_nb_neighbors(swigCPtr, this, layer_no); + } + + public void neighbor_range(long no, int layer_no, SWIGTYPE_p_unsigned_long begin, SWIGTYPE_p_unsigned_long end) { + swigfaissJNI.HNSW_neighbor_range(swigCPtr, this, no, layer_no, SWIGTYPE_p_unsigned_long.getCPtr(begin), SWIGTYPE_p_unsigned_long.getCPtr(end)); + } + + public HNSW(int M) { + this(swigfaissJNI.new_HNSW__SWIG_0(M), true); + } + + public HNSW() { + this(swigfaissJNI.new_HNSW__SWIG_1(), true); + } + + public int random_level() { + return swigfaissJNI.HNSW_random_level(swigCPtr, this); + } + + public void fill_with_random_links(long n) { + swigfaissJNI.HNSW_fill_with_random_links(swigCPtr, this, n); + } + + public void add_links_starting_from(DistanceComputer ptdis, int pt_id, int nearest, float d_nearest, int level, SWIGTYPE_p_omp_lock_t locks, VisitedTable vt) { + swigfaissJNI.HNSW_add_links_starting_from(swigCPtr, this, DistanceComputer.getCPtr(ptdis), ptdis, pt_id, nearest, d_nearest, level, SWIGTYPE_p_omp_lock_t.getCPtr(locks), VisitedTable.getCPtr(vt), vt); + } + + public void add_with_locks(DistanceComputer ptdis, int pt_level, int pt_id, SWIGTYPE_p_std__vectorT_omp_lock_t_t locks, VisitedTable vt) { + swigfaissJNI.HNSW_add_with_locks(swigCPtr, this, DistanceComputer.getCPtr(ptdis), ptdis, pt_level, pt_id, SWIGTYPE_p_std__vectorT_omp_lock_t_t.getCPtr(locks), VisitedTable.getCPtr(vt), vt); + } + + public int search_from_candidates(DistanceComputer qdis, int k, LongVector I, SWIGTYPE_p_float D, HNSW.MinimaxHeap candidates, VisitedTable vt, HNSWStats stats, int level, int nres_in) { + return swigfaissJNI.HNSW_search_from_candidates__SWIG_0(swigCPtr, this, DistanceComputer.getCPtr(qdis), qdis, k, SWIGTYPE_p_long_long.getCPtr(I.data()), I, SWIGTYPE_p_float.getCPtr(D), HNSW.MinimaxHeap.getCPtr(candidates), candidates, VisitedTable.getCPtr(vt), vt, HNSWStats.getCPtr(stats), stats, level, nres_in); + } + + public int search_from_candidates(DistanceComputer qdis, int k, LongVector I, SWIGTYPE_p_float D, HNSW.MinimaxHeap candidates, VisitedTable vt, HNSWStats stats, int level) { + return swigfaissJNI.HNSW_search_from_candidates__SWIG_1(swigCPtr, this, DistanceComputer.getCPtr(qdis), qdis, k, SWIGTYPE_p_long_long.getCPtr(I.data()), I, SWIGTYPE_p_float.getCPtr(D), HNSW.MinimaxHeap.getCPtr(candidates), candidates, VisitedTable.getCPtr(vt), vt, HNSWStats.getCPtr(stats), stats, level); + } + + public SWIGTYPE_p_std__priority_queueT_std__pairT_float_int_t_t search_from_candidate_unbounded(SWIGTYPE_p_std__pairT_float_int_t node, DistanceComputer qdis, int ef, VisitedTable vt, HNSWStats stats) { + return new SWIGTYPE_p_std__priority_queueT_std__pairT_float_int_t_t(swigfaissJNI.HNSW_search_from_candidate_unbounded(swigCPtr, this, SWIGTYPE_p_std__pairT_float_int_t.getCPtr(node), DistanceComputer.getCPtr(qdis), qdis, ef, VisitedTable.getCPtr(vt), vt, HNSWStats.getCPtr(stats), stats), true); + } + + public HNSWStats search(DistanceComputer qdis, int k, LongVector I, SWIGTYPE_p_float D, VisitedTable vt) { + return new HNSWStats(swigfaissJNI.HNSW_search(swigCPtr, this, DistanceComputer.getCPtr(qdis), qdis, k, SWIGTYPE_p_long_long.getCPtr(I.data()), I, SWIGTYPE_p_float.getCPtr(D), VisitedTable.getCPtr(vt), vt), true); + } + + public void reset() { + swigfaissJNI.HNSW_reset(swigCPtr, this); + } + + public void clear_neighbor_tables(int level) { + swigfaissJNI.HNSW_clear_neighbor_tables(swigCPtr, this, level); + } + + public void print_neighbor_stats(int level) { + swigfaissJNI.HNSW_print_neighbor_stats(swigCPtr, this, level); + } + + public int prepare_level_tab(long n, boolean preset_levels) { + return swigfaissJNI.HNSW_prepare_level_tab__SWIG_0(swigCPtr, this, n, preset_levels); + } + + public int prepare_level_tab(long n) { + return swigfaissJNI.HNSW_prepare_level_tab__SWIG_1(swigCPtr, this, n); + } + + public static void shrink_neighbor_list(DistanceComputer qdis, SWIGTYPE_p_std__priority_queueT_faiss__HNSW__NodeDistFarther_t input, SWIGTYPE_p_std__vectorT_faiss__HNSW__NodeDistFarther_t output, int max_size) { + swigfaissJNI.HNSW_shrink_neighbor_list(DistanceComputer.getCPtr(qdis), qdis, SWIGTYPE_p_std__priority_queueT_faiss__HNSW__NodeDistFarther_t.getCPtr(input), SWIGTYPE_p_std__vectorT_faiss__HNSW__NodeDistFarther_t.getCPtr(output), max_size); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/HNSWStats.java b/ann/src/main/java/com/twitter/ann/faiss/swig/HNSWStats.java new file mode 100644 index 0000000000..baf388f5af --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/HNSWStats.java @@ -0,0 +1,111 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class HNSWStats { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected HNSWStats(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(HNSWStats obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_HNSWStats(swigCPtr); + } + swigCPtr = 0; + } + } + + public void setN1(long value) { + swigfaissJNI.HNSWStats_n1_set(swigCPtr, this, value); + } + + public long getN1() { + return swigfaissJNI.HNSWStats_n1_get(swigCPtr, this); + } + + public void setN2(long value) { + swigfaissJNI.HNSWStats_n2_set(swigCPtr, this, value); + } + + public long getN2() { + return swigfaissJNI.HNSWStats_n2_get(swigCPtr, this); + } + + public void setN3(long value) { + swigfaissJNI.HNSWStats_n3_set(swigCPtr, this, value); + } + + public long getN3() { + return swigfaissJNI.HNSWStats_n3_get(swigCPtr, this); + } + + public void setNdis(long value) { + swigfaissJNI.HNSWStats_ndis_set(swigCPtr, this, value); + } + + public long getNdis() { + return swigfaissJNI.HNSWStats_ndis_get(swigCPtr, this); + } + + public void setNreorder(long value) { + swigfaissJNI.HNSWStats_nreorder_set(swigCPtr, this, value); + } + + public long getNreorder() { + return swigfaissJNI.HNSWStats_nreorder_get(swigCPtr, this); + } + + public HNSWStats(long n1, long n2, long n3, long ndis, long nreorder) { + this(swigfaissJNI.new_HNSWStats__SWIG_0(n1, n2, n3, ndis, nreorder), true); + } + + public HNSWStats(long n1, long n2, long n3, long ndis) { + this(swigfaissJNI.new_HNSWStats__SWIG_1(n1, n2, n3, ndis), true); + } + + public HNSWStats(long n1, long n2, long n3) { + this(swigfaissJNI.new_HNSWStats__SWIG_2(n1, n2, n3), true); + } + + public HNSWStats(long n1, long n2) { + this(swigfaissJNI.new_HNSWStats__SWIG_3(n1, n2), true); + } + + public HNSWStats(long n1) { + this(swigfaissJNI.new_HNSWStats__SWIG_4(n1), true); + } + + public HNSWStats() { + this(swigfaissJNI.new_HNSWStats__SWIG_5(), true); + } + + public void reset() { + swigfaissJNI.HNSWStats_reset(swigCPtr, this); + } + + public void combine(HNSWStats other) { + swigfaissJNI.HNSWStats_combine(swigCPtr, this, HNSWStats.getCPtr(other), other); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/HStackInvertedLists.java b/ann/src/main/java/com/twitter/ann/faiss/swig/HStackInvertedLists.java new file mode 100644 index 0000000000..52e5c3b8ce --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/HStackInvertedLists.java @@ -0,0 +1,86 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class HStackInvertedLists extends ReadOnlyInvertedLists { + private transient long swigCPtr; + + protected HStackInvertedLists(long cPtr, boolean cMemoryOwn) { + super(swigfaissJNI.HStackInvertedLists_SWIGUpcast(cPtr), cMemoryOwn); + swigCPtr = cPtr; + } + + protected static long getCPtr(HStackInvertedLists obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_HStackInvertedLists(swigCPtr); + } + swigCPtr = 0; + } + super.delete(); + } + + public void setIls(SWIGTYPE_p_std__vectorT_faiss__InvertedLists_const_p_t value) { + swigfaissJNI.HStackInvertedLists_ils_set(swigCPtr, this, SWIGTYPE_p_std__vectorT_faiss__InvertedLists_const_p_t.getCPtr(value)); + } + + public SWIGTYPE_p_std__vectorT_faiss__InvertedLists_const_p_t getIls() { + long cPtr = swigfaissJNI.HStackInvertedLists_ils_get(swigCPtr, this); + return (cPtr == 0) ? null : new SWIGTYPE_p_std__vectorT_faiss__InvertedLists_const_p_t(cPtr, false); + } + + public HStackInvertedLists(int nil, SWIGTYPE_p_p_faiss__InvertedLists ils) { + this(swigfaissJNI.new_HStackInvertedLists(nil, SWIGTYPE_p_p_faiss__InvertedLists.getCPtr(ils)), true); + } + + public long list_size(long list_no) { + return swigfaissJNI.HStackInvertedLists_list_size(swigCPtr, this, list_no); + } + + public SWIGTYPE_p_unsigned_char get_codes(long list_no) { + long cPtr = swigfaissJNI.HStackInvertedLists_get_codes(swigCPtr, this, list_no); + return (cPtr == 0) ? null : new SWIGTYPE_p_unsigned_char(cPtr, false); + } + + public LongVector get_ids(long list_no) { + return new LongVector(swigfaissJNI.HStackInvertedLists_get_ids(swigCPtr, this, list_no), false); +} + + public void prefetch_lists(LongVector list_nos, int nlist) { + swigfaissJNI.HStackInvertedLists_prefetch_lists(swigCPtr, this, SWIGTYPE_p_long_long.getCPtr(list_nos.data()), list_nos, nlist); + } + + public void release_codes(long list_no, SWIGTYPE_p_unsigned_char codes) { + swigfaissJNI.HStackInvertedLists_release_codes(swigCPtr, this, list_no, SWIGTYPE_p_unsigned_char.getCPtr(codes)); + } + + public void release_ids(long list_no, LongVector ids) { + swigfaissJNI.HStackInvertedLists_release_ids(swigCPtr, this, list_no, SWIGTYPE_p_long_long.getCPtr(ids.data()), ids); + } + + public long get_single_id(long list_no, long offset) { + return swigfaissJNI.HStackInvertedLists_get_single_id(swigCPtr, this, list_no, offset); +} + + public SWIGTYPE_p_unsigned_char get_single_code(long list_no, long offset) { + long cPtr = swigfaissJNI.HStackInvertedLists_get_single_code(swigCPtr, this, list_no, offset); + return (cPtr == 0) ? null : new SWIGTYPE_p_unsigned_char(cPtr, false); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/HammingComputer16.java b/ann/src/main/java/com/twitter/ann/faiss/swig/HammingComputer16.java new file mode 100644 index 0000000000..14727d0832 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/HammingComputer16.java @@ -0,0 +1,71 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class HammingComputer16 { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected HammingComputer16(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(HammingComputer16 obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_HammingComputer16(swigCPtr); + } + swigCPtr = 0; + } + } + + public void setA0(long value) { + swigfaissJNI.HammingComputer16_a0_set(swigCPtr, this, value); + } + + public long getA0() { + return swigfaissJNI.HammingComputer16_a0_get(swigCPtr, this); + } + + public void setA1(long value) { + swigfaissJNI.HammingComputer16_a1_set(swigCPtr, this, value); + } + + public long getA1() { + return swigfaissJNI.HammingComputer16_a1_get(swigCPtr, this); + } + + public HammingComputer16() { + this(swigfaissJNI.new_HammingComputer16__SWIG_0(), true); + } + + public HammingComputer16(SWIGTYPE_p_unsigned_char a8, int code_size) { + this(swigfaissJNI.new_HammingComputer16__SWIG_1(SWIGTYPE_p_unsigned_char.getCPtr(a8), code_size), true); + } + + public void set(SWIGTYPE_p_unsigned_char a8, int code_size) { + swigfaissJNI.HammingComputer16_set(swigCPtr, this, SWIGTYPE_p_unsigned_char.getCPtr(a8), code_size); + } + + public int hamming(SWIGTYPE_p_unsigned_char b8) { + return swigfaissJNI.HammingComputer16_hamming(swigCPtr, this, SWIGTYPE_p_unsigned_char.getCPtr(b8)); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/HammingComputer20.java b/ann/src/main/java/com/twitter/ann/faiss/swig/HammingComputer20.java new file mode 100644 index 0000000000..68857db2b4 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/HammingComputer20.java @@ -0,0 +1,79 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class HammingComputer20 { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected HammingComputer20(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(HammingComputer20 obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_HammingComputer20(swigCPtr); + } + swigCPtr = 0; + } + } + + public void setA0(long value) { + swigfaissJNI.HammingComputer20_a0_set(swigCPtr, this, value); + } + + public long getA0() { + return swigfaissJNI.HammingComputer20_a0_get(swigCPtr, this); + } + + public void setA1(long value) { + swigfaissJNI.HammingComputer20_a1_set(swigCPtr, this, value); + } + + public long getA1() { + return swigfaissJNI.HammingComputer20_a1_get(swigCPtr, this); + } + + public void setA2(SWIGTYPE_p_uint32_t value) { + swigfaissJNI.HammingComputer20_a2_set(swigCPtr, this, SWIGTYPE_p_uint32_t.getCPtr(value)); + } + + public SWIGTYPE_p_uint32_t getA2() { + return new SWIGTYPE_p_uint32_t(swigfaissJNI.HammingComputer20_a2_get(swigCPtr, this), true); + } + + public HammingComputer20() { + this(swigfaissJNI.new_HammingComputer20__SWIG_0(), true); + } + + public HammingComputer20(SWIGTYPE_p_unsigned_char a8, int code_size) { + this(swigfaissJNI.new_HammingComputer20__SWIG_1(SWIGTYPE_p_unsigned_char.getCPtr(a8), code_size), true); + } + + public void set(SWIGTYPE_p_unsigned_char a8, int code_size) { + swigfaissJNI.HammingComputer20_set(swigCPtr, this, SWIGTYPE_p_unsigned_char.getCPtr(a8), code_size); + } + + public int hamming(SWIGTYPE_p_unsigned_char b8) { + return swigfaissJNI.HammingComputer20_hamming(swigCPtr, this, SWIGTYPE_p_unsigned_char.getCPtr(b8)); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/HammingComputer32.java b/ann/src/main/java/com/twitter/ann/faiss/swig/HammingComputer32.java new file mode 100644 index 0000000000..78207d8eb6 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/HammingComputer32.java @@ -0,0 +1,87 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class HammingComputer32 { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected HammingComputer32(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(HammingComputer32 obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_HammingComputer32(swigCPtr); + } + swigCPtr = 0; + } + } + + public void setA0(long value) { + swigfaissJNI.HammingComputer32_a0_set(swigCPtr, this, value); + } + + public long getA0() { + return swigfaissJNI.HammingComputer32_a0_get(swigCPtr, this); + } + + public void setA1(long value) { + swigfaissJNI.HammingComputer32_a1_set(swigCPtr, this, value); + } + + public long getA1() { + return swigfaissJNI.HammingComputer32_a1_get(swigCPtr, this); + } + + public void setA2(long value) { + swigfaissJNI.HammingComputer32_a2_set(swigCPtr, this, value); + } + + public long getA2() { + return swigfaissJNI.HammingComputer32_a2_get(swigCPtr, this); + } + + public void setA3(long value) { + swigfaissJNI.HammingComputer32_a3_set(swigCPtr, this, value); + } + + public long getA3() { + return swigfaissJNI.HammingComputer32_a3_get(swigCPtr, this); + } + + public HammingComputer32() { + this(swigfaissJNI.new_HammingComputer32__SWIG_0(), true); + } + + public HammingComputer32(SWIGTYPE_p_unsigned_char a8, int code_size) { + this(swigfaissJNI.new_HammingComputer32__SWIG_1(SWIGTYPE_p_unsigned_char.getCPtr(a8), code_size), true); + } + + public void set(SWIGTYPE_p_unsigned_char a8, int code_size) { + swigfaissJNI.HammingComputer32_set(swigCPtr, this, SWIGTYPE_p_unsigned_char.getCPtr(a8), code_size); + } + + public int hamming(SWIGTYPE_p_unsigned_char b8) { + return swigfaissJNI.HammingComputer32_hamming(swigCPtr, this, SWIGTYPE_p_unsigned_char.getCPtr(b8)); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/HammingComputer4.java b/ann/src/main/java/com/twitter/ann/faiss/swig/HammingComputer4.java new file mode 100644 index 0000000000..3f8b676053 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/HammingComputer4.java @@ -0,0 +1,63 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class HammingComputer4 { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected HammingComputer4(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(HammingComputer4 obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_HammingComputer4(swigCPtr); + } + swigCPtr = 0; + } + } + + public void setA0(SWIGTYPE_p_uint32_t value) { + swigfaissJNI.HammingComputer4_a0_set(swigCPtr, this, SWIGTYPE_p_uint32_t.getCPtr(value)); + } + + public SWIGTYPE_p_uint32_t getA0() { + return new SWIGTYPE_p_uint32_t(swigfaissJNI.HammingComputer4_a0_get(swigCPtr, this), true); + } + + public HammingComputer4() { + this(swigfaissJNI.new_HammingComputer4__SWIG_0(), true); + } + + public HammingComputer4(SWIGTYPE_p_unsigned_char a, int code_size) { + this(swigfaissJNI.new_HammingComputer4__SWIG_1(SWIGTYPE_p_unsigned_char.getCPtr(a), code_size), true); + } + + public void set(SWIGTYPE_p_unsigned_char a, int code_size) { + swigfaissJNI.HammingComputer4_set(swigCPtr, this, SWIGTYPE_p_unsigned_char.getCPtr(a), code_size); + } + + public int hamming(SWIGTYPE_p_unsigned_char b) { + return swigfaissJNI.HammingComputer4_hamming(swigCPtr, this, SWIGTYPE_p_unsigned_char.getCPtr(b)); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/HammingComputer64.java b/ann/src/main/java/com/twitter/ann/faiss/swig/HammingComputer64.java new file mode 100644 index 0000000000..d1962d3676 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/HammingComputer64.java @@ -0,0 +1,119 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class HammingComputer64 { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected HammingComputer64(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(HammingComputer64 obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_HammingComputer64(swigCPtr); + } + swigCPtr = 0; + } + } + + public void setA0(long value) { + swigfaissJNI.HammingComputer64_a0_set(swigCPtr, this, value); + } + + public long getA0() { + return swigfaissJNI.HammingComputer64_a0_get(swigCPtr, this); + } + + public void setA1(long value) { + swigfaissJNI.HammingComputer64_a1_set(swigCPtr, this, value); + } + + public long getA1() { + return swigfaissJNI.HammingComputer64_a1_get(swigCPtr, this); + } + + public void setA2(long value) { + swigfaissJNI.HammingComputer64_a2_set(swigCPtr, this, value); + } + + public long getA2() { + return swigfaissJNI.HammingComputer64_a2_get(swigCPtr, this); + } + + public void setA3(long value) { + swigfaissJNI.HammingComputer64_a3_set(swigCPtr, this, value); + } + + public long getA3() { + return swigfaissJNI.HammingComputer64_a3_get(swigCPtr, this); + } + + public void setA4(long value) { + swigfaissJNI.HammingComputer64_a4_set(swigCPtr, this, value); + } + + public long getA4() { + return swigfaissJNI.HammingComputer64_a4_get(swigCPtr, this); + } + + public void setA5(long value) { + swigfaissJNI.HammingComputer64_a5_set(swigCPtr, this, value); + } + + public long getA5() { + return swigfaissJNI.HammingComputer64_a5_get(swigCPtr, this); + } + + public void setA6(long value) { + swigfaissJNI.HammingComputer64_a6_set(swigCPtr, this, value); + } + + public long getA6() { + return swigfaissJNI.HammingComputer64_a6_get(swigCPtr, this); + } + + public void setA7(long value) { + swigfaissJNI.HammingComputer64_a7_set(swigCPtr, this, value); + } + + public long getA7() { + return swigfaissJNI.HammingComputer64_a7_get(swigCPtr, this); + } + + public HammingComputer64() { + this(swigfaissJNI.new_HammingComputer64__SWIG_0(), true); + } + + public HammingComputer64(SWIGTYPE_p_unsigned_char a8, int code_size) { + this(swigfaissJNI.new_HammingComputer64__SWIG_1(SWIGTYPE_p_unsigned_char.getCPtr(a8), code_size), true); + } + + public void set(SWIGTYPE_p_unsigned_char a8, int code_size) { + swigfaissJNI.HammingComputer64_set(swigCPtr, this, SWIGTYPE_p_unsigned_char.getCPtr(a8), code_size); + } + + public int hamming(SWIGTYPE_p_unsigned_char b8) { + return swigfaissJNI.HammingComputer64_hamming(swigCPtr, this, SWIGTYPE_p_unsigned_char.getCPtr(b8)); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/HammingComputer8.java b/ann/src/main/java/com/twitter/ann/faiss/swig/HammingComputer8.java new file mode 100644 index 0000000000..16c7eb8a92 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/HammingComputer8.java @@ -0,0 +1,63 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class HammingComputer8 { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected HammingComputer8(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(HammingComputer8 obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_HammingComputer8(swigCPtr); + } + swigCPtr = 0; + } + } + + public void setA0(long value) { + swigfaissJNI.HammingComputer8_a0_set(swigCPtr, this, value); + } + + public long getA0() { + return swigfaissJNI.HammingComputer8_a0_get(swigCPtr, this); + } + + public HammingComputer8() { + this(swigfaissJNI.new_HammingComputer8__SWIG_0(), true); + } + + public HammingComputer8(SWIGTYPE_p_unsigned_char a, int code_size) { + this(swigfaissJNI.new_HammingComputer8__SWIG_1(SWIGTYPE_p_unsigned_char.getCPtr(a), code_size), true); + } + + public void set(SWIGTYPE_p_unsigned_char a, int code_size) { + swigfaissJNI.HammingComputer8_set(swigCPtr, this, SWIGTYPE_p_unsigned_char.getCPtr(a), code_size); + } + + public int hamming(SWIGTYPE_p_unsigned_char b) { + return swigfaissJNI.HammingComputer8_hamming(swigCPtr, this, SWIGTYPE_p_unsigned_char.getCPtr(b)); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/HammingComputerDefault.java b/ann/src/main/java/com/twitter/ann/faiss/swig/HammingComputerDefault.java new file mode 100644 index 0000000000..b569f8ed0e --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/HammingComputerDefault.java @@ -0,0 +1,80 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class HammingComputerDefault { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected HammingComputerDefault(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(HammingComputerDefault obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_HammingComputerDefault(swigCPtr); + } + swigCPtr = 0; + } + } + + public void setA8(SWIGTYPE_p_unsigned_char value) { + swigfaissJNI.HammingComputerDefault_a8_set(swigCPtr, this, SWIGTYPE_p_unsigned_char.getCPtr(value)); + } + + public SWIGTYPE_p_unsigned_char getA8() { + long cPtr = swigfaissJNI.HammingComputerDefault_a8_get(swigCPtr, this); + return (cPtr == 0) ? null : new SWIGTYPE_p_unsigned_char(cPtr, false); + } + + public void setQuotient8(int value) { + swigfaissJNI.HammingComputerDefault_quotient8_set(swigCPtr, this, value); + } + + public int getQuotient8() { + return swigfaissJNI.HammingComputerDefault_quotient8_get(swigCPtr, this); + } + + public void setRemainder8(int value) { + swigfaissJNI.HammingComputerDefault_remainder8_set(swigCPtr, this, value); + } + + public int getRemainder8() { + return swigfaissJNI.HammingComputerDefault_remainder8_get(swigCPtr, this); + } + + public HammingComputerDefault() { + this(swigfaissJNI.new_HammingComputerDefault__SWIG_0(), true); + } + + public HammingComputerDefault(SWIGTYPE_p_unsigned_char a8, int code_size) { + this(swigfaissJNI.new_HammingComputerDefault__SWIG_1(SWIGTYPE_p_unsigned_char.getCPtr(a8), code_size), true); + } + + public void set(SWIGTYPE_p_unsigned_char a8, int code_size) { + swigfaissJNI.HammingComputerDefault_set(swigCPtr, this, SWIGTYPE_p_unsigned_char.getCPtr(a8), code_size); + } + + public int hamming(SWIGTYPE_p_unsigned_char b8) { + return swigfaissJNI.HammingComputerDefault_hamming(swigCPtr, this, SWIGTYPE_p_unsigned_char.getCPtr(b8)); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/HammingComputerM4.java b/ann/src/main/java/com/twitter/ann/faiss/swig/HammingComputerM4.java new file mode 100644 index 0000000000..386fbda615 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/HammingComputerM4.java @@ -0,0 +1,72 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class HammingComputerM4 { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected HammingComputerM4(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(HammingComputerM4 obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_HammingComputerM4(swigCPtr); + } + swigCPtr = 0; + } + } + + public void setA(SWIGTYPE_p_uint32_t value) { + swigfaissJNI.HammingComputerM4_a_set(swigCPtr, this, SWIGTYPE_p_uint32_t.getCPtr(value)); + } + + public SWIGTYPE_p_uint32_t getA() { + long cPtr = swigfaissJNI.HammingComputerM4_a_get(swigCPtr, this); + return (cPtr == 0) ? null : new SWIGTYPE_p_uint32_t(cPtr, false); + } + + public void setN(int value) { + swigfaissJNI.HammingComputerM4_n_set(swigCPtr, this, value); + } + + public int getN() { + return swigfaissJNI.HammingComputerM4_n_get(swigCPtr, this); + } + + public HammingComputerM4() { + this(swigfaissJNI.new_HammingComputerM4__SWIG_0(), true); + } + + public HammingComputerM4(SWIGTYPE_p_unsigned_char a4, int code_size) { + this(swigfaissJNI.new_HammingComputerM4__SWIG_1(SWIGTYPE_p_unsigned_char.getCPtr(a4), code_size), true); + } + + public void set(SWIGTYPE_p_unsigned_char a4, int code_size) { + swigfaissJNI.HammingComputerM4_set(swigCPtr, this, SWIGTYPE_p_unsigned_char.getCPtr(a4), code_size); + } + + public int hamming(SWIGTYPE_p_unsigned_char b8) { + return swigfaissJNI.HammingComputerM4_hamming(swigCPtr, this, SWIGTYPE_p_unsigned_char.getCPtr(b8)); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/HammingComputerM8.java b/ann/src/main/java/com/twitter/ann/faiss/swig/HammingComputerM8.java new file mode 100644 index 0000000000..24a5204764 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/HammingComputerM8.java @@ -0,0 +1,72 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class HammingComputerM8 { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected HammingComputerM8(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(HammingComputerM8 obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_HammingComputerM8(swigCPtr); + } + swigCPtr = 0; + } + } + + public void setA(SWIGTYPE_p_unsigned_long value) { + swigfaissJNI.HammingComputerM8_a_set(swigCPtr, this, SWIGTYPE_p_unsigned_long.getCPtr(value)); + } + + public SWIGTYPE_p_unsigned_long getA() { + long cPtr = swigfaissJNI.HammingComputerM8_a_get(swigCPtr, this); + return (cPtr == 0) ? null : new SWIGTYPE_p_unsigned_long(cPtr, false); + } + + public void setN(int value) { + swigfaissJNI.HammingComputerM8_n_set(swigCPtr, this, value); + } + + public int getN() { + return swigfaissJNI.HammingComputerM8_n_get(swigCPtr, this); + } + + public HammingComputerM8() { + this(swigfaissJNI.new_HammingComputerM8__SWIG_0(), true); + } + + public HammingComputerM8(SWIGTYPE_p_unsigned_char a8, int code_size) { + this(swigfaissJNI.new_HammingComputerM8__SWIG_1(SWIGTYPE_p_unsigned_char.getCPtr(a8), code_size), true); + } + + public void set(SWIGTYPE_p_unsigned_char a8, int code_size) { + swigfaissJNI.HammingComputerM8_set(swigCPtr, this, SWIGTYPE_p_unsigned_char.getCPtr(a8), code_size); + } + + public int hamming(SWIGTYPE_p_unsigned_char b8) { + return swigfaissJNI.HammingComputerM8_hamming(swigCPtr, this, SWIGTYPE_p_unsigned_char.getCPtr(b8)); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/IDSelector.java b/ann/src/main/java/com/twitter/ann/faiss/swig/IDSelector.java new file mode 100644 index 0000000000..893f28bb28 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/IDSelector.java @@ -0,0 +1,43 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class IDSelector { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected IDSelector(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(IDSelector obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_IDSelector(swigCPtr); + } + swigCPtr = 0; + } + } + + public boolean is_member(long id) { + return swigfaissJNI.IDSelector_is_member(swigCPtr, this, id); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/IDSelectorArray.java b/ann/src/main/java/com/twitter/ann/faiss/swig/IDSelectorArray.java new file mode 100644 index 0000000000..42c4860f23 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/IDSelectorArray.java @@ -0,0 +1,63 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class IDSelectorArray extends IDSelector { + private transient long swigCPtr; + + protected IDSelectorArray(long cPtr, boolean cMemoryOwn) { + super(swigfaissJNI.IDSelectorArray_SWIGUpcast(cPtr), cMemoryOwn); + swigCPtr = cPtr; + } + + protected static long getCPtr(IDSelectorArray obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_IDSelectorArray(swigCPtr); + } + swigCPtr = 0; + } + super.delete(); + } + + public void setN(long value) { + swigfaissJNI.IDSelectorArray_n_set(swigCPtr, this, value); + } + + public long getN() { + return swigfaissJNI.IDSelectorArray_n_get(swigCPtr, this); + } + + public void setIds(LongVector value) { + swigfaissJNI.IDSelectorArray_ids_set(swigCPtr, this, SWIGTYPE_p_long_long.getCPtr(value.data()), value); + } + + public LongVector getIds() { + return new LongVector(swigfaissJNI.IDSelectorArray_ids_get(swigCPtr, this), false); +} + + public IDSelectorArray(long n, LongVector ids) { + this(swigfaissJNI.new_IDSelectorArray(n, SWIGTYPE_p_long_long.getCPtr(ids.data()), ids), true); + } + + public boolean is_member(long id) { + return swigfaissJNI.IDSelectorArray_is_member(swigCPtr, this, id); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/IDSelectorBatch.java b/ann/src/main/java/com/twitter/ann/faiss/swig/IDSelectorBatch.java new file mode 100644 index 0000000000..69988c1be6 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/IDSelectorBatch.java @@ -0,0 +1,63 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class IDSelectorBatch extends IDSelector { + private transient long swigCPtr; + + protected IDSelectorBatch(long cPtr, boolean cMemoryOwn) { + super(swigfaissJNI.IDSelectorBatch_SWIGUpcast(cPtr), cMemoryOwn); + swigCPtr = cPtr; + } + + protected static long getCPtr(IDSelectorBatch obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_IDSelectorBatch(swigCPtr); + } + swigCPtr = 0; + } + super.delete(); + } + + public void setNbits(int value) { + swigfaissJNI.IDSelectorBatch_nbits_set(swigCPtr, this, value); + } + + public int getNbits() { + return swigfaissJNI.IDSelectorBatch_nbits_get(swigCPtr, this); + } + + public void setMask(long value) { + swigfaissJNI.IDSelectorBatch_mask_set(swigCPtr, this, value); + } + + public long getMask() { + return swigfaissJNI.IDSelectorBatch_mask_get(swigCPtr, this); +} + + public IDSelectorBatch(long n, LongVector indices) { + this(swigfaissJNI.new_IDSelectorBatch(n, SWIGTYPE_p_long_long.getCPtr(indices.data()), indices), true); + } + + public boolean is_member(long id) { + return swigfaissJNI.IDSelectorBatch_is_member(swigCPtr, this, id); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/IDSelectorRange.java b/ann/src/main/java/com/twitter/ann/faiss/swig/IDSelectorRange.java new file mode 100644 index 0000000000..c8f49eb882 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/IDSelectorRange.java @@ -0,0 +1,63 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class IDSelectorRange extends IDSelector { + private transient long swigCPtr; + + protected IDSelectorRange(long cPtr, boolean cMemoryOwn) { + super(swigfaissJNI.IDSelectorRange_SWIGUpcast(cPtr), cMemoryOwn); + swigCPtr = cPtr; + } + + protected static long getCPtr(IDSelectorRange obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_IDSelectorRange(swigCPtr); + } + swigCPtr = 0; + } + super.delete(); + } + + public void setImin(long value) { + swigfaissJNI.IDSelectorRange_imin_set(swigCPtr, this, value); + } + + public long getImin() { + return swigfaissJNI.IDSelectorRange_imin_get(swigCPtr, this); +} + + public void setImax(long value) { + swigfaissJNI.IDSelectorRange_imax_set(swigCPtr, this, value); + } + + public long getImax() { + return swigfaissJNI.IDSelectorRange_imax_get(swigCPtr, this); +} + + public IDSelectorRange(long imin, long imax) { + this(swigfaissJNI.new_IDSelectorRange(imin, imax), true); + } + + public boolean is_member(long id) { + return swigfaissJNI.IDSelectorRange_is_member(swigCPtr, this, id); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/ITQMatrix.java b/ann/src/main/java/com/twitter/ann/faiss/swig/ITQMatrix.java new file mode 100644 index 0000000000..333577c44e --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/ITQMatrix.java @@ -0,0 +1,76 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class ITQMatrix extends LinearTransform { + private transient long swigCPtr; + + protected ITQMatrix(long cPtr, boolean cMemoryOwn) { + super(swigfaissJNI.ITQMatrix_SWIGUpcast(cPtr), cMemoryOwn); + swigCPtr = cPtr; + } + + protected static long getCPtr(ITQMatrix obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_ITQMatrix(swigCPtr); + } + swigCPtr = 0; + } + super.delete(); + } + + public void setMax_iter(int value) { + swigfaissJNI.ITQMatrix_max_iter_set(swigCPtr, this, value); + } + + public int getMax_iter() { + return swigfaissJNI.ITQMatrix_max_iter_get(swigCPtr, this); + } + + public void setSeed(int value) { + swigfaissJNI.ITQMatrix_seed_set(swigCPtr, this, value); + } + + public int getSeed() { + return swigfaissJNI.ITQMatrix_seed_get(swigCPtr, this); + } + + public void setInit_rotation(DoubleVector value) { + swigfaissJNI.ITQMatrix_init_rotation_set(swigCPtr, this, DoubleVector.getCPtr(value), value); + } + + public DoubleVector getInit_rotation() { + long cPtr = swigfaissJNI.ITQMatrix_init_rotation_get(swigCPtr, this); + return (cPtr == 0) ? null : new DoubleVector(cPtr, false); + } + + public ITQMatrix(int d) { + this(swigfaissJNI.new_ITQMatrix__SWIG_0(d), true); + } + + public ITQMatrix() { + this(swigfaissJNI.new_ITQMatrix__SWIG_1(), true); + } + + public void train(long n, SWIGTYPE_p_float x) { + swigfaissJNI.ITQMatrix_train(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x)); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/ITQTransform.java b/ann/src/main/java/com/twitter/ann/faiss/swig/ITQTransform.java new file mode 100644 index 0000000000..42196648d6 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/ITQTransform.java @@ -0,0 +1,106 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class ITQTransform extends VectorTransform { + private transient long swigCPtr; + + protected ITQTransform(long cPtr, boolean cMemoryOwn) { + super(swigfaissJNI.ITQTransform_SWIGUpcast(cPtr), cMemoryOwn); + swigCPtr = cPtr; + } + + protected static long getCPtr(ITQTransform obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_ITQTransform(swigCPtr); + } + swigCPtr = 0; + } + super.delete(); + } + + public void setMean(FloatVector value) { + swigfaissJNI.ITQTransform_mean_set(swigCPtr, this, FloatVector.getCPtr(value), value); + } + + public FloatVector getMean() { + long cPtr = swigfaissJNI.ITQTransform_mean_get(swigCPtr, this); + return (cPtr == 0) ? null : new FloatVector(cPtr, false); + } + + public void setDo_pca(boolean value) { + swigfaissJNI.ITQTransform_do_pca_set(swigCPtr, this, value); + } + + public boolean getDo_pca() { + return swigfaissJNI.ITQTransform_do_pca_get(swigCPtr, this); + } + + public void setItq(ITQMatrix value) { + swigfaissJNI.ITQTransform_itq_set(swigCPtr, this, ITQMatrix.getCPtr(value), value); + } + + public ITQMatrix getItq() { + long cPtr = swigfaissJNI.ITQTransform_itq_get(swigCPtr, this); + return (cPtr == 0) ? null : new ITQMatrix(cPtr, false); + } + + public void setMax_train_per_dim(int value) { + swigfaissJNI.ITQTransform_max_train_per_dim_set(swigCPtr, this, value); + } + + public int getMax_train_per_dim() { + return swigfaissJNI.ITQTransform_max_train_per_dim_get(swigCPtr, this); + } + + public void setPca_then_itq(LinearTransform value) { + swigfaissJNI.ITQTransform_pca_then_itq_set(swigCPtr, this, LinearTransform.getCPtr(value), value); + } + + public LinearTransform getPca_then_itq() { + long cPtr = swigfaissJNI.ITQTransform_pca_then_itq_get(swigCPtr, this); + return (cPtr == 0) ? null : new LinearTransform(cPtr, false); + } + + public ITQTransform(int d_in, int d_out, boolean do_pca) { + this(swigfaissJNI.new_ITQTransform__SWIG_0(d_in, d_out, do_pca), true); + } + + public ITQTransform(int d_in, int d_out) { + this(swigfaissJNI.new_ITQTransform__SWIG_1(d_in, d_out), true); + } + + public ITQTransform(int d_in) { + this(swigfaissJNI.new_ITQTransform__SWIG_2(d_in), true); + } + + public ITQTransform() { + this(swigfaissJNI.new_ITQTransform__SWIG_3(), true); + } + + public void train(long n, SWIGTYPE_p_float x) { + swigfaissJNI.ITQTransform_train(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x)); + } + + public void apply_noalloc(long n, SWIGTYPE_p_float x, SWIGTYPE_p_float xt) { + swigfaissJNI.ITQTransform_apply_noalloc(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x), SWIGTYPE_p_float.getCPtr(xt)); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/IVFPQSearchParameters.java b/ann/src/main/java/com/twitter/ann/faiss/swig/IVFPQSearchParameters.java new file mode 100644 index 0000000000..2146f2f569 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/IVFPQSearchParameters.java @@ -0,0 +1,59 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class IVFPQSearchParameters extends IVFSearchParameters { + private transient long swigCPtr; + + protected IVFPQSearchParameters(long cPtr, boolean cMemoryOwn) { + super(swigfaissJNI.IVFPQSearchParameters_SWIGUpcast(cPtr), cMemoryOwn); + swigCPtr = cPtr; + } + + protected static long getCPtr(IVFPQSearchParameters obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_IVFPQSearchParameters(swigCPtr); + } + swigCPtr = 0; + } + super.delete(); + } + + public void setScan_table_threshold(long value) { + swigfaissJNI.IVFPQSearchParameters_scan_table_threshold_set(swigCPtr, this, value); + } + + public long getScan_table_threshold() { + return swigfaissJNI.IVFPQSearchParameters_scan_table_threshold_get(swigCPtr, this); + } + + public void setPolysemous_ht(int value) { + swigfaissJNI.IVFPQSearchParameters_polysemous_ht_set(swigCPtr, this, value); + } + + public int getPolysemous_ht() { + return swigfaissJNI.IVFPQSearchParameters_polysemous_ht_get(swigCPtr, this); + } + + public IVFPQSearchParameters() { + this(swigfaissJNI.new_IVFPQSearchParameters(), true); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/IVFSearchParameters.java b/ann/src/main/java/com/twitter/ann/faiss/swig/IVFSearchParameters.java new file mode 100644 index 0000000000..c5c21dfd75 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/IVFSearchParameters.java @@ -0,0 +1,59 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class IVFSearchParameters { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected IVFSearchParameters(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(IVFSearchParameters obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_IVFSearchParameters(swigCPtr); + } + swigCPtr = 0; + } + } + + public void setNprobe(long value) { + swigfaissJNI.IVFSearchParameters_nprobe_set(swigCPtr, this, value); + } + + public long getNprobe() { + return swigfaissJNI.IVFSearchParameters_nprobe_get(swigCPtr, this); + } + + public void setMax_codes(long value) { + swigfaissJNI.IVFSearchParameters_max_codes_set(swigCPtr, this, value); + } + + public long getMax_codes() { + return swigfaissJNI.IVFSearchParameters_max_codes_get(swigCPtr, this); + } + + public IVFSearchParameters() { + this(swigfaissJNI.new_IVFSearchParameters(), true); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/Index.java b/ann/src/main/java/com/twitter/ann/faiss/swig/Index.java new file mode 100644 index 0000000000..a2f04c1941 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/Index.java @@ -0,0 +1,165 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class Index { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected Index(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(Index obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_Index(swigCPtr); + } + swigCPtr = 0; + } + } + + public void setD(int value) { + swigfaissJNI.Index_d_set(swigCPtr, this, value); + } + + public int getD() { + return swigfaissJNI.Index_d_get(swigCPtr, this); + } + + public void setNtotal(long value) { + swigfaissJNI.Index_ntotal_set(swigCPtr, this, value); + } + + public long getNtotal() { + return swigfaissJNI.Index_ntotal_get(swigCPtr, this); +} + + public void setVerbose(boolean value) { + swigfaissJNI.Index_verbose_set(swigCPtr, this, value); + } + + public boolean getVerbose() { + return swigfaissJNI.Index_verbose_get(swigCPtr, this); + } + + public void setIs_trained(boolean value) { + swigfaissJNI.Index_is_trained_set(swigCPtr, this, value); + } + + public boolean getIs_trained() { + return swigfaissJNI.Index_is_trained_get(swigCPtr, this); + } + + public void setMetric_type(MetricType value) { + swigfaissJNI.Index_metric_type_set(swigCPtr, this, value.swigValue()); + } + + public MetricType getMetric_type() { + return MetricType.swigToEnum(swigfaissJNI.Index_metric_type_get(swigCPtr, this)); + } + + public void setMetric_arg(float value) { + swigfaissJNI.Index_metric_arg_set(swigCPtr, this, value); + } + + public float getMetric_arg() { + return swigfaissJNI.Index_metric_arg_get(swigCPtr, this); + } + + public void train(long n, SWIGTYPE_p_float x) { + swigfaissJNI.Index_train(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x)); + } + + public void add(long n, SWIGTYPE_p_float x) { + swigfaissJNI.Index_add(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x)); + } + + public void add_with_ids(long n, SWIGTYPE_p_float x, LongVector xids) { + swigfaissJNI.Index_add_with_ids(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x), SWIGTYPE_p_long_long.getCPtr(xids.data()), xids); + } + + public void search(long n, SWIGTYPE_p_float x, long k, SWIGTYPE_p_float distances, LongVector labels) { + swigfaissJNI.Index_search(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x), k, SWIGTYPE_p_float.getCPtr(distances), SWIGTYPE_p_long_long.getCPtr(labels.data()), labels); + } + + public void range_search(long n, SWIGTYPE_p_float x, float radius, RangeSearchResult result) { + swigfaissJNI.Index_range_search(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x), radius, RangeSearchResult.getCPtr(result), result); + } + + public void assign(long n, SWIGTYPE_p_float x, LongVector labels, long k) { + swigfaissJNI.Index_assign__SWIG_0(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x), SWIGTYPE_p_long_long.getCPtr(labels.data()), labels, k); + } + + public void assign(long n, SWIGTYPE_p_float x, LongVector labels) { + swigfaissJNI.Index_assign__SWIG_1(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x), SWIGTYPE_p_long_long.getCPtr(labels.data()), labels); + } + + public void reset() { + swigfaissJNI.Index_reset(swigCPtr, this); + } + + public long remove_ids(IDSelector sel) { + return swigfaissJNI.Index_remove_ids(swigCPtr, this, IDSelector.getCPtr(sel), sel); + } + + public void reconstruct(long key, SWIGTYPE_p_float recons) { + swigfaissJNI.Index_reconstruct(swigCPtr, this, key, SWIGTYPE_p_float.getCPtr(recons)); + } + + public void reconstruct_n(long i0, long ni, SWIGTYPE_p_float recons) { + swigfaissJNI.Index_reconstruct_n(swigCPtr, this, i0, ni, SWIGTYPE_p_float.getCPtr(recons)); + } + + public void search_and_reconstruct(long n, SWIGTYPE_p_float x, long k, SWIGTYPE_p_float distances, LongVector labels, SWIGTYPE_p_float recons) { + swigfaissJNI.Index_search_and_reconstruct(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x), k, SWIGTYPE_p_float.getCPtr(distances), SWIGTYPE_p_long_long.getCPtr(labels.data()), labels, SWIGTYPE_p_float.getCPtr(recons)); + } + + public void compute_residual(SWIGTYPE_p_float x, SWIGTYPE_p_float residual, long key) { + swigfaissJNI.Index_compute_residual(swigCPtr, this, SWIGTYPE_p_float.getCPtr(x), SWIGTYPE_p_float.getCPtr(residual), key); + } + + public void compute_residual_n(long n, SWIGTYPE_p_float xs, SWIGTYPE_p_float residuals, LongVector keys) { + swigfaissJNI.Index_compute_residual_n(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(xs), SWIGTYPE_p_float.getCPtr(residuals), SWIGTYPE_p_long_long.getCPtr(keys.data()), keys); + } + + public DistanceComputer get_distance_computer() { + long cPtr = swigfaissJNI.Index_get_distance_computer(swigCPtr, this); + return (cPtr == 0) ? null : new DistanceComputer(cPtr, false); + } + + public long sa_code_size() { + return swigfaissJNI.Index_sa_code_size(swigCPtr, this); + } + + public void sa_encode(long n, SWIGTYPE_p_float x, SWIGTYPE_p_unsigned_char bytes) { + swigfaissJNI.Index_sa_encode(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x), SWIGTYPE_p_unsigned_char.getCPtr(bytes)); + } + + public void sa_decode(long n, SWIGTYPE_p_unsigned_char bytes, SWIGTYPE_p_float x) { + swigfaissJNI.Index_sa_decode(swigCPtr, this, n, SWIGTYPE_p_unsigned_char.getCPtr(bytes), SWIGTYPE_p_float.getCPtr(x)); + } + + public IndexIVF toIVF() { + long cPtr = swigfaissJNI.Index_toIVF(swigCPtr, this); + return (cPtr == 0) ? null : new IndexIVF(cPtr, false); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/Index2Layer.java b/ann/src/main/java/com/twitter/ann/faiss/swig/Index2Layer.java new file mode 100644 index 0000000000..6045797ad5 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/Index2Layer.java @@ -0,0 +1,114 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class Index2Layer extends IndexFlatCodes { + private transient long swigCPtr; + + protected Index2Layer(long cPtr, boolean cMemoryOwn) { + super(swigfaissJNI.Index2Layer_SWIGUpcast(cPtr), cMemoryOwn); + swigCPtr = cPtr; + } + + protected static long getCPtr(Index2Layer obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_Index2Layer(swigCPtr); + } + swigCPtr = 0; + } + super.delete(); + } + + public void setQ1(Level1Quantizer value) { + swigfaissJNI.Index2Layer_q1_set(swigCPtr, this, Level1Quantizer.getCPtr(value), value); + } + + public Level1Quantizer getQ1() { + long cPtr = swigfaissJNI.Index2Layer_q1_get(swigCPtr, this); + return (cPtr == 0) ? null : new Level1Quantizer(cPtr, false); + } + + public void setPq(ProductQuantizer value) { + swigfaissJNI.Index2Layer_pq_set(swigCPtr, this, ProductQuantizer.getCPtr(value), value); + } + + public ProductQuantizer getPq() { + long cPtr = swigfaissJNI.Index2Layer_pq_get(swigCPtr, this); + return (cPtr == 0) ? null : new ProductQuantizer(cPtr, false); + } + + public void setCode_size_1(long value) { + swigfaissJNI.Index2Layer_code_size_1_set(swigCPtr, this, value); + } + + public long getCode_size_1() { + return swigfaissJNI.Index2Layer_code_size_1_get(swigCPtr, this); + } + + public void setCode_size_2(long value) { + swigfaissJNI.Index2Layer_code_size_2_set(swigCPtr, this, value); + } + + public long getCode_size_2() { + return swigfaissJNI.Index2Layer_code_size_2_get(swigCPtr, this); + } + + public Index2Layer(Index quantizer, long nlist, int M, int nbit, MetricType metric) { + this(swigfaissJNI.new_Index2Layer__SWIG_0(Index.getCPtr(quantizer), quantizer, nlist, M, nbit, metric.swigValue()), true); + } + + public Index2Layer(Index quantizer, long nlist, int M, int nbit) { + this(swigfaissJNI.new_Index2Layer__SWIG_1(Index.getCPtr(quantizer), quantizer, nlist, M, nbit), true); + } + + public Index2Layer(Index quantizer, long nlist, int M) { + this(swigfaissJNI.new_Index2Layer__SWIG_2(Index.getCPtr(quantizer), quantizer, nlist, M), true); + } + + public Index2Layer() { + this(swigfaissJNI.new_Index2Layer__SWIG_3(), true); + } + + public void train(long n, SWIGTYPE_p_float x) { + swigfaissJNI.Index2Layer_train(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x)); + } + + public void search(long n, SWIGTYPE_p_float x, long k, SWIGTYPE_p_float distances, LongVector labels) { + swigfaissJNI.Index2Layer_search(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x), k, SWIGTYPE_p_float.getCPtr(distances), SWIGTYPE_p_long_long.getCPtr(labels.data()), labels); + } + + public DistanceComputer get_distance_computer() { + long cPtr = swigfaissJNI.Index2Layer_get_distance_computer(swigCPtr, this); + return (cPtr == 0) ? null : new DistanceComputer(cPtr, false); + } + + public void transfer_to_IVFPQ(IndexIVFPQ other) { + swigfaissJNI.Index2Layer_transfer_to_IVFPQ(swigCPtr, this, IndexIVFPQ.getCPtr(other), other); + } + + public void sa_encode(long n, SWIGTYPE_p_float x, SWIGTYPE_p_unsigned_char bytes) { + swigfaissJNI.Index2Layer_sa_encode(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x), SWIGTYPE_p_unsigned_char.getCPtr(bytes)); + } + + public void sa_decode(long n, SWIGTYPE_p_unsigned_char bytes, SWIGTYPE_p_float x) { + swigfaissJNI.Index2Layer_sa_decode(swigCPtr, this, n, SWIGTYPE_p_unsigned_char.getCPtr(bytes), SWIGTYPE_p_float.getCPtr(x)); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/IndexBinary.java b/ann/src/main/java/com/twitter/ann/faiss/swig/IndexBinary.java new file mode 100644 index 0000000000..c60ea69abd --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/IndexBinary.java @@ -0,0 +1,139 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class IndexBinary { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected IndexBinary(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(IndexBinary obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_IndexBinary(swigCPtr); + } + swigCPtr = 0; + } + } + + public void setD(int value) { + swigfaissJNI.IndexBinary_d_set(swigCPtr, this, value); + } + + public int getD() { + return swigfaissJNI.IndexBinary_d_get(swigCPtr, this); + } + + public void setCode_size(int value) { + swigfaissJNI.IndexBinary_code_size_set(swigCPtr, this, value); + } + + public int getCode_size() { + return swigfaissJNI.IndexBinary_code_size_get(swigCPtr, this); + } + + public void setNtotal(long value) { + swigfaissJNI.IndexBinary_ntotal_set(swigCPtr, this, value); + } + + public long getNtotal() { + return swigfaissJNI.IndexBinary_ntotal_get(swigCPtr, this); +} + + public void setVerbose(boolean value) { + swigfaissJNI.IndexBinary_verbose_set(swigCPtr, this, value); + } + + public boolean getVerbose() { + return swigfaissJNI.IndexBinary_verbose_get(swigCPtr, this); + } + + public void setIs_trained(boolean value) { + swigfaissJNI.IndexBinary_is_trained_set(swigCPtr, this, value); + } + + public boolean getIs_trained() { + return swigfaissJNI.IndexBinary_is_trained_get(swigCPtr, this); + } + + public void setMetric_type(MetricType value) { + swigfaissJNI.IndexBinary_metric_type_set(swigCPtr, this, value.swigValue()); + } + + public MetricType getMetric_type() { + return MetricType.swigToEnum(swigfaissJNI.IndexBinary_metric_type_get(swigCPtr, this)); + } + + public void train(long n, SWIGTYPE_p_unsigned_char x) { + swigfaissJNI.IndexBinary_train(swigCPtr, this, n, SWIGTYPE_p_unsigned_char.getCPtr(x)); + } + + public void add(long n, SWIGTYPE_p_unsigned_char x) { + swigfaissJNI.IndexBinary_add(swigCPtr, this, n, SWIGTYPE_p_unsigned_char.getCPtr(x)); + } + + public void add_with_ids(long n, SWIGTYPE_p_unsigned_char x, LongVector xids) { + swigfaissJNI.IndexBinary_add_with_ids(swigCPtr, this, n, SWIGTYPE_p_unsigned_char.getCPtr(x), SWIGTYPE_p_long_long.getCPtr(xids.data()), xids); + } + + public void search(long n, SWIGTYPE_p_unsigned_char x, long k, SWIGTYPE_p_int distances, LongVector labels) { + swigfaissJNI.IndexBinary_search(swigCPtr, this, n, SWIGTYPE_p_unsigned_char.getCPtr(x), k, SWIGTYPE_p_int.getCPtr(distances), SWIGTYPE_p_long_long.getCPtr(labels.data()), labels); + } + + public void range_search(long n, SWIGTYPE_p_unsigned_char x, int radius, RangeSearchResult result) { + swigfaissJNI.IndexBinary_range_search(swigCPtr, this, n, SWIGTYPE_p_unsigned_char.getCPtr(x), radius, RangeSearchResult.getCPtr(result), result); + } + + public void assign(long n, SWIGTYPE_p_unsigned_char x, LongVector labels, long k) { + swigfaissJNI.IndexBinary_assign__SWIG_0(swigCPtr, this, n, SWIGTYPE_p_unsigned_char.getCPtr(x), SWIGTYPE_p_long_long.getCPtr(labels.data()), labels, k); + } + + public void assign(long n, SWIGTYPE_p_unsigned_char x, LongVector labels) { + swigfaissJNI.IndexBinary_assign__SWIG_1(swigCPtr, this, n, SWIGTYPE_p_unsigned_char.getCPtr(x), SWIGTYPE_p_long_long.getCPtr(labels.data()), labels); + } + + public void reset() { + swigfaissJNI.IndexBinary_reset(swigCPtr, this); + } + + public long remove_ids(IDSelector sel) { + return swigfaissJNI.IndexBinary_remove_ids(swigCPtr, this, IDSelector.getCPtr(sel), sel); + } + + public void reconstruct(long key, SWIGTYPE_p_unsigned_char recons) { + swigfaissJNI.IndexBinary_reconstruct(swigCPtr, this, key, SWIGTYPE_p_unsigned_char.getCPtr(recons)); + } + + public void reconstruct_n(long i0, long ni, SWIGTYPE_p_unsigned_char recons) { + swigfaissJNI.IndexBinary_reconstruct_n(swigCPtr, this, i0, ni, SWIGTYPE_p_unsigned_char.getCPtr(recons)); + } + + public void search_and_reconstruct(long n, SWIGTYPE_p_unsigned_char x, long k, SWIGTYPE_p_int distances, LongVector labels, SWIGTYPE_p_unsigned_char recons) { + swigfaissJNI.IndexBinary_search_and_reconstruct(swigCPtr, this, n, SWIGTYPE_p_unsigned_char.getCPtr(x), k, SWIGTYPE_p_int.getCPtr(distances), SWIGTYPE_p_long_long.getCPtr(labels.data()), labels, SWIGTYPE_p_unsigned_char.getCPtr(recons)); + } + + public void display() { + swigfaissJNI.IndexBinary_display(swigCPtr, this); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/IndexBinaryFlat.java b/ann/src/main/java/com/twitter/ann/faiss/swig/IndexBinaryFlat.java new file mode 100644 index 0000000000..84be3becd2 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/IndexBinaryFlat.java @@ -0,0 +1,96 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class IndexBinaryFlat extends IndexBinary { + private transient long swigCPtr; + + protected IndexBinaryFlat(long cPtr, boolean cMemoryOwn) { + super(swigfaissJNI.IndexBinaryFlat_SWIGUpcast(cPtr), cMemoryOwn); + swigCPtr = cPtr; + } + + protected static long getCPtr(IndexBinaryFlat obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_IndexBinaryFlat(swigCPtr); + } + swigCPtr = 0; + } + super.delete(); + } + + public void setXb(ByteVector value) { + swigfaissJNI.IndexBinaryFlat_xb_set(swigCPtr, this, ByteVector.getCPtr(value), value); + } + + public ByteVector getXb() { + long cPtr = swigfaissJNI.IndexBinaryFlat_xb_get(swigCPtr, this); + return (cPtr == 0) ? null : new ByteVector(cPtr, false); + } + + public void setUse_heap(boolean value) { + swigfaissJNI.IndexBinaryFlat_use_heap_set(swigCPtr, this, value); + } + + public boolean getUse_heap() { + return swigfaissJNI.IndexBinaryFlat_use_heap_get(swigCPtr, this); + } + + public void setQuery_batch_size(long value) { + swigfaissJNI.IndexBinaryFlat_query_batch_size_set(swigCPtr, this, value); + } + + public long getQuery_batch_size() { + return swigfaissJNI.IndexBinaryFlat_query_batch_size_get(swigCPtr, this); + } + + public IndexBinaryFlat(long d) { + this(swigfaissJNI.new_IndexBinaryFlat__SWIG_0(d), true); + } + + public void add(long n, SWIGTYPE_p_unsigned_char x) { + swigfaissJNI.IndexBinaryFlat_add(swigCPtr, this, n, SWIGTYPE_p_unsigned_char.getCPtr(x)); + } + + public void reset() { + swigfaissJNI.IndexBinaryFlat_reset(swigCPtr, this); + } + + public void search(long n, SWIGTYPE_p_unsigned_char x, long k, SWIGTYPE_p_int distances, LongVector labels) { + swigfaissJNI.IndexBinaryFlat_search(swigCPtr, this, n, SWIGTYPE_p_unsigned_char.getCPtr(x), k, SWIGTYPE_p_int.getCPtr(distances), SWIGTYPE_p_long_long.getCPtr(labels.data()), labels); + } + + public void range_search(long n, SWIGTYPE_p_unsigned_char x, int radius, RangeSearchResult result) { + swigfaissJNI.IndexBinaryFlat_range_search(swigCPtr, this, n, SWIGTYPE_p_unsigned_char.getCPtr(x), radius, RangeSearchResult.getCPtr(result), result); + } + + public void reconstruct(long key, SWIGTYPE_p_unsigned_char recons) { + swigfaissJNI.IndexBinaryFlat_reconstruct(swigCPtr, this, key, SWIGTYPE_p_unsigned_char.getCPtr(recons)); + } + + public long remove_ids(IDSelector sel) { + return swigfaissJNI.IndexBinaryFlat_remove_ids(swigCPtr, this, IDSelector.getCPtr(sel), sel); + } + + public IndexBinaryFlat() { + this(swigfaissJNI.new_IndexBinaryFlat__SWIG_1(), true); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/IndexBinaryFromFloat.java b/ann/src/main/java/com/twitter/ann/faiss/swig/IndexBinaryFromFloat.java new file mode 100644 index 0000000000..c55ac683ba --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/IndexBinaryFromFloat.java @@ -0,0 +1,80 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class IndexBinaryFromFloat extends IndexBinary { + private transient long swigCPtr; + + protected IndexBinaryFromFloat(long cPtr, boolean cMemoryOwn) { + super(swigfaissJNI.IndexBinaryFromFloat_SWIGUpcast(cPtr), cMemoryOwn); + swigCPtr = cPtr; + } + + protected static long getCPtr(IndexBinaryFromFloat obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_IndexBinaryFromFloat(swigCPtr); + } + swigCPtr = 0; + } + super.delete(); + } + + public void setIndex(Index value) { + swigfaissJNI.IndexBinaryFromFloat_index_set(swigCPtr, this, Index.getCPtr(value), value); + } + + public Index getIndex() { + long cPtr = swigfaissJNI.IndexBinaryFromFloat_index_get(swigCPtr, this); + return (cPtr == 0) ? null : new Index(cPtr, false); + } + + public void setOwn_fields(boolean value) { + swigfaissJNI.IndexBinaryFromFloat_own_fields_set(swigCPtr, this, value); + } + + public boolean getOwn_fields() { + return swigfaissJNI.IndexBinaryFromFloat_own_fields_get(swigCPtr, this); + } + + public IndexBinaryFromFloat() { + this(swigfaissJNI.new_IndexBinaryFromFloat__SWIG_0(), true); + } + + public IndexBinaryFromFloat(Index index) { + this(swigfaissJNI.new_IndexBinaryFromFloat__SWIG_1(Index.getCPtr(index), index), true); + } + + public void add(long n, SWIGTYPE_p_unsigned_char x) { + swigfaissJNI.IndexBinaryFromFloat_add(swigCPtr, this, n, SWIGTYPE_p_unsigned_char.getCPtr(x)); + } + + public void reset() { + swigfaissJNI.IndexBinaryFromFloat_reset(swigCPtr, this); + } + + public void search(long n, SWIGTYPE_p_unsigned_char x, long k, SWIGTYPE_p_int distances, LongVector labels) { + swigfaissJNI.IndexBinaryFromFloat_search(swigCPtr, this, n, SWIGTYPE_p_unsigned_char.getCPtr(x), k, SWIGTYPE_p_int.getCPtr(distances), SWIGTYPE_p_long_long.getCPtr(labels.data()), labels); + } + + public void train(long n, SWIGTYPE_p_unsigned_char x) { + swigfaissJNI.IndexBinaryFromFloat_train(swigCPtr, this, n, SWIGTYPE_p_unsigned_char.getCPtr(x)); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/IndexBinaryHNSW.java b/ann/src/main/java/com/twitter/ann/faiss/swig/IndexBinaryHNSW.java new file mode 100644 index 0000000000..f103061369 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/IndexBinaryHNSW.java @@ -0,0 +1,110 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class IndexBinaryHNSW extends IndexBinary { + private transient long swigCPtr; + + protected IndexBinaryHNSW(long cPtr, boolean cMemoryOwn) { + super(swigfaissJNI.IndexBinaryHNSW_SWIGUpcast(cPtr), cMemoryOwn); + swigCPtr = cPtr; + } + + protected static long getCPtr(IndexBinaryHNSW obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_IndexBinaryHNSW(swigCPtr); + } + swigCPtr = 0; + } + super.delete(); + } + + public void setHnsw(HNSW value) { + swigfaissJNI.IndexBinaryHNSW_hnsw_set(swigCPtr, this, HNSW.getCPtr(value), value); + } + + public HNSW getHnsw() { + long cPtr = swigfaissJNI.IndexBinaryHNSW_hnsw_get(swigCPtr, this); + return (cPtr == 0) ? null : new HNSW(cPtr, false); + } + + public void setOwn_fields(boolean value) { + swigfaissJNI.IndexBinaryHNSW_own_fields_set(swigCPtr, this, value); + } + + public boolean getOwn_fields() { + return swigfaissJNI.IndexBinaryHNSW_own_fields_get(swigCPtr, this); + } + + public void setStorage(IndexBinary value) { + swigfaissJNI.IndexBinaryHNSW_storage_set(swigCPtr, this, IndexBinary.getCPtr(value), value); + } + + public IndexBinary getStorage() { + long cPtr = swigfaissJNI.IndexBinaryHNSW_storage_get(swigCPtr, this); + return (cPtr == 0) ? null : new IndexBinary(cPtr, false); + } + + public IndexBinaryHNSW() { + this(swigfaissJNI.new_IndexBinaryHNSW__SWIG_0(), true); + } + + public IndexBinaryHNSW(int d, int M) { + this(swigfaissJNI.new_IndexBinaryHNSW__SWIG_1(d, M), true); + } + + public IndexBinaryHNSW(int d) { + this(swigfaissJNI.new_IndexBinaryHNSW__SWIG_2(d), true); + } + + public IndexBinaryHNSW(IndexBinary storage, int M) { + this(swigfaissJNI.new_IndexBinaryHNSW__SWIG_3(IndexBinary.getCPtr(storage), storage, M), true); + } + + public IndexBinaryHNSW(IndexBinary storage) { + this(swigfaissJNI.new_IndexBinaryHNSW__SWIG_4(IndexBinary.getCPtr(storage), storage), true); + } + + public DistanceComputer get_distance_computer() { + long cPtr = swigfaissJNI.IndexBinaryHNSW_get_distance_computer(swigCPtr, this); + return (cPtr == 0) ? null : new DistanceComputer(cPtr, false); + } + + public void add(long n, SWIGTYPE_p_unsigned_char x) { + swigfaissJNI.IndexBinaryHNSW_add(swigCPtr, this, n, SWIGTYPE_p_unsigned_char.getCPtr(x)); + } + + public void train(long n, SWIGTYPE_p_unsigned_char x) { + swigfaissJNI.IndexBinaryHNSW_train(swigCPtr, this, n, SWIGTYPE_p_unsigned_char.getCPtr(x)); + } + + public void search(long n, SWIGTYPE_p_unsigned_char x, long k, SWIGTYPE_p_int distances, LongVector labels) { + swigfaissJNI.IndexBinaryHNSW_search(swigCPtr, this, n, SWIGTYPE_p_unsigned_char.getCPtr(x), k, SWIGTYPE_p_int.getCPtr(distances), SWIGTYPE_p_long_long.getCPtr(labels.data()), labels); + } + + public void reconstruct(long key, SWIGTYPE_p_unsigned_char recons) { + swigfaissJNI.IndexBinaryHNSW_reconstruct(swigCPtr, this, key, SWIGTYPE_p_unsigned_char.getCPtr(recons)); + } + + public void reset() { + swigfaissJNI.IndexBinaryHNSW_reset(swigCPtr, this); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/IndexBinaryIVF.java b/ann/src/main/java/com/twitter/ann/faiss/swig/IndexBinaryIVF.java new file mode 100644 index 0000000000..0da7053a45 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/IndexBinaryIVF.java @@ -0,0 +1,237 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class IndexBinaryIVF extends IndexBinary { + private transient long swigCPtr; + + protected IndexBinaryIVF(long cPtr, boolean cMemoryOwn) { + super(swigfaissJNI.IndexBinaryIVF_SWIGUpcast(cPtr), cMemoryOwn); + swigCPtr = cPtr; + } + + protected static long getCPtr(IndexBinaryIVF obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_IndexBinaryIVF(swigCPtr); + } + swigCPtr = 0; + } + super.delete(); + } + + public void setInvlists(InvertedLists value) { + swigfaissJNI.IndexBinaryIVF_invlists_set(swigCPtr, this, InvertedLists.getCPtr(value), value); + } + + public InvertedLists getInvlists() { + long cPtr = swigfaissJNI.IndexBinaryIVF_invlists_get(swigCPtr, this); + return (cPtr == 0) ? null : new InvertedLists(cPtr, false); + } + + public void setOwn_invlists(boolean value) { + swigfaissJNI.IndexBinaryIVF_own_invlists_set(swigCPtr, this, value); + } + + public boolean getOwn_invlists() { + return swigfaissJNI.IndexBinaryIVF_own_invlists_get(swigCPtr, this); + } + + public void setNprobe(long value) { + swigfaissJNI.IndexBinaryIVF_nprobe_set(swigCPtr, this, value); + } + + public long getNprobe() { + return swigfaissJNI.IndexBinaryIVF_nprobe_get(swigCPtr, this); + } + + public void setMax_codes(long value) { + swigfaissJNI.IndexBinaryIVF_max_codes_set(swigCPtr, this, value); + } + + public long getMax_codes() { + return swigfaissJNI.IndexBinaryIVF_max_codes_get(swigCPtr, this); + } + + public void setUse_heap(boolean value) { + swigfaissJNI.IndexBinaryIVF_use_heap_set(swigCPtr, this, value); + } + + public boolean getUse_heap() { + return swigfaissJNI.IndexBinaryIVF_use_heap_get(swigCPtr, this); + } + + public void setDirect_map(SWIGTYPE_p_DirectMap value) { + swigfaissJNI.IndexBinaryIVF_direct_map_set(swigCPtr, this, SWIGTYPE_p_DirectMap.getCPtr(value)); + } + + public SWIGTYPE_p_DirectMap getDirect_map() { + return new SWIGTYPE_p_DirectMap(swigfaissJNI.IndexBinaryIVF_direct_map_get(swigCPtr, this), true); + } + + public void setQuantizer(IndexBinary value) { + swigfaissJNI.IndexBinaryIVF_quantizer_set(swigCPtr, this, IndexBinary.getCPtr(value), value); + } + + public IndexBinary getQuantizer() { + long cPtr = swigfaissJNI.IndexBinaryIVF_quantizer_get(swigCPtr, this); + return (cPtr == 0) ? null : new IndexBinary(cPtr, false); + } + + public void setNlist(long value) { + swigfaissJNI.IndexBinaryIVF_nlist_set(swigCPtr, this, value); + } + + public long getNlist() { + return swigfaissJNI.IndexBinaryIVF_nlist_get(swigCPtr, this); + } + + public void setOwn_fields(boolean value) { + swigfaissJNI.IndexBinaryIVF_own_fields_set(swigCPtr, this, value); + } + + public boolean getOwn_fields() { + return swigfaissJNI.IndexBinaryIVF_own_fields_get(swigCPtr, this); + } + + public void setCp(ClusteringParameters value) { + swigfaissJNI.IndexBinaryIVF_cp_set(swigCPtr, this, ClusteringParameters.getCPtr(value), value); + } + + public ClusteringParameters getCp() { + long cPtr = swigfaissJNI.IndexBinaryIVF_cp_get(swigCPtr, this); + return (cPtr == 0) ? null : new ClusteringParameters(cPtr, false); + } + + public void setClustering_index(Index value) { + swigfaissJNI.IndexBinaryIVF_clustering_index_set(swigCPtr, this, Index.getCPtr(value), value); + } + + public Index getClustering_index() { + long cPtr = swigfaissJNI.IndexBinaryIVF_clustering_index_get(swigCPtr, this); + return (cPtr == 0) ? null : new Index(cPtr, false); + } + + public IndexBinaryIVF(IndexBinary quantizer, long d, long nlist) { + this(swigfaissJNI.new_IndexBinaryIVF__SWIG_0(IndexBinary.getCPtr(quantizer), quantizer, d, nlist), true); + } + + public IndexBinaryIVF() { + this(swigfaissJNI.new_IndexBinaryIVF__SWIG_1(), true); + } + + public void reset() { + swigfaissJNI.IndexBinaryIVF_reset(swigCPtr, this); + } + + public void train(long n, SWIGTYPE_p_unsigned_char x) { + swigfaissJNI.IndexBinaryIVF_train(swigCPtr, this, n, SWIGTYPE_p_unsigned_char.getCPtr(x)); + } + + public void add(long n, SWIGTYPE_p_unsigned_char x) { + swigfaissJNI.IndexBinaryIVF_add(swigCPtr, this, n, SWIGTYPE_p_unsigned_char.getCPtr(x)); + } + + public void add_with_ids(long n, SWIGTYPE_p_unsigned_char x, LongVector xids) { + swigfaissJNI.IndexBinaryIVF_add_with_ids(swigCPtr, this, n, SWIGTYPE_p_unsigned_char.getCPtr(x), SWIGTYPE_p_long_long.getCPtr(xids.data()), xids); + } + + public void add_core(long n, SWIGTYPE_p_unsigned_char x, LongVector xids, LongVector precomputed_idx) { + swigfaissJNI.IndexBinaryIVF_add_core(swigCPtr, this, n, SWIGTYPE_p_unsigned_char.getCPtr(x), SWIGTYPE_p_long_long.getCPtr(xids.data()), xids, SWIGTYPE_p_long_long.getCPtr(precomputed_idx.data()), precomputed_idx); + } + + public void search_preassigned(long n, SWIGTYPE_p_unsigned_char x, long k, LongVector assign, SWIGTYPE_p_int centroid_dis, SWIGTYPE_p_int distances, LongVector labels, boolean store_pairs, IVFSearchParameters params) { + swigfaissJNI.IndexBinaryIVF_search_preassigned__SWIG_0(swigCPtr, this, n, SWIGTYPE_p_unsigned_char.getCPtr(x), k, SWIGTYPE_p_long_long.getCPtr(assign.data()), assign, SWIGTYPE_p_int.getCPtr(centroid_dis), SWIGTYPE_p_int.getCPtr(distances), SWIGTYPE_p_long_long.getCPtr(labels.data()), labels, store_pairs, IVFSearchParameters.getCPtr(params), params); + } + + public void search_preassigned(long n, SWIGTYPE_p_unsigned_char x, long k, LongVector assign, SWIGTYPE_p_int centroid_dis, SWIGTYPE_p_int distances, LongVector labels, boolean store_pairs) { + swigfaissJNI.IndexBinaryIVF_search_preassigned__SWIG_1(swigCPtr, this, n, SWIGTYPE_p_unsigned_char.getCPtr(x), k, SWIGTYPE_p_long_long.getCPtr(assign.data()), assign, SWIGTYPE_p_int.getCPtr(centroid_dis), SWIGTYPE_p_int.getCPtr(distances), SWIGTYPE_p_long_long.getCPtr(labels.data()), labels, store_pairs); + } + + public SWIGTYPE_p_faiss__BinaryInvertedListScanner get_InvertedListScanner(boolean store_pairs) { + long cPtr = swigfaissJNI.IndexBinaryIVF_get_InvertedListScanner__SWIG_0(swigCPtr, this, store_pairs); + return (cPtr == 0) ? null : new SWIGTYPE_p_faiss__BinaryInvertedListScanner(cPtr, false); + } + + public SWIGTYPE_p_faiss__BinaryInvertedListScanner get_InvertedListScanner() { + long cPtr = swigfaissJNI.IndexBinaryIVF_get_InvertedListScanner__SWIG_1(swigCPtr, this); + return (cPtr == 0) ? null : new SWIGTYPE_p_faiss__BinaryInvertedListScanner(cPtr, false); + } + + public void search(long n, SWIGTYPE_p_unsigned_char x, long k, SWIGTYPE_p_int distances, LongVector labels) { + swigfaissJNI.IndexBinaryIVF_search(swigCPtr, this, n, SWIGTYPE_p_unsigned_char.getCPtr(x), k, SWIGTYPE_p_int.getCPtr(distances), SWIGTYPE_p_long_long.getCPtr(labels.data()), labels); + } + + public void range_search(long n, SWIGTYPE_p_unsigned_char x, int radius, RangeSearchResult result) { + swigfaissJNI.IndexBinaryIVF_range_search(swigCPtr, this, n, SWIGTYPE_p_unsigned_char.getCPtr(x), radius, RangeSearchResult.getCPtr(result), result); + } + + public void range_search_preassigned(long n, SWIGTYPE_p_unsigned_char x, int radius, LongVector assign, SWIGTYPE_p_int centroid_dis, RangeSearchResult result) { + swigfaissJNI.IndexBinaryIVF_range_search_preassigned(swigCPtr, this, n, SWIGTYPE_p_unsigned_char.getCPtr(x), radius, SWIGTYPE_p_long_long.getCPtr(assign.data()), assign, SWIGTYPE_p_int.getCPtr(centroid_dis), RangeSearchResult.getCPtr(result), result); + } + + public void reconstruct(long key, SWIGTYPE_p_unsigned_char recons) { + swigfaissJNI.IndexBinaryIVF_reconstruct(swigCPtr, this, key, SWIGTYPE_p_unsigned_char.getCPtr(recons)); + } + + public void reconstruct_n(long i0, long ni, SWIGTYPE_p_unsigned_char recons) { + swigfaissJNI.IndexBinaryIVF_reconstruct_n(swigCPtr, this, i0, ni, SWIGTYPE_p_unsigned_char.getCPtr(recons)); + } + + public void search_and_reconstruct(long n, SWIGTYPE_p_unsigned_char x, long k, SWIGTYPE_p_int distances, LongVector labels, SWIGTYPE_p_unsigned_char recons) { + swigfaissJNI.IndexBinaryIVF_search_and_reconstruct(swigCPtr, this, n, SWIGTYPE_p_unsigned_char.getCPtr(x), k, SWIGTYPE_p_int.getCPtr(distances), SWIGTYPE_p_long_long.getCPtr(labels.data()), labels, SWIGTYPE_p_unsigned_char.getCPtr(recons)); + } + + public void reconstruct_from_offset(long list_no, long offset, SWIGTYPE_p_unsigned_char recons) { + swigfaissJNI.IndexBinaryIVF_reconstruct_from_offset(swigCPtr, this, list_no, offset, SWIGTYPE_p_unsigned_char.getCPtr(recons)); + } + + public long remove_ids(IDSelector sel) { + return swigfaissJNI.IndexBinaryIVF_remove_ids(swigCPtr, this, IDSelector.getCPtr(sel), sel); + } + + public void merge_from(IndexBinaryIVF other, long add_id) { + swigfaissJNI.IndexBinaryIVF_merge_from(swigCPtr, this, IndexBinaryIVF.getCPtr(other), other, add_id); + } + + public long get_list_size(long list_no) { + return swigfaissJNI.IndexBinaryIVF_get_list_size(swigCPtr, this, list_no); + } + + public void make_direct_map(boolean new_maintain_direct_map) { + swigfaissJNI.IndexBinaryIVF_make_direct_map__SWIG_0(swigCPtr, this, new_maintain_direct_map); + } + + public void make_direct_map() { + swigfaissJNI.IndexBinaryIVF_make_direct_map__SWIG_1(swigCPtr, this); + } + + public void set_direct_map_type(SWIGTYPE_p_DirectMap__Type type) { + swigfaissJNI.IndexBinaryIVF_set_direct_map_type(swigCPtr, this, SWIGTYPE_p_DirectMap__Type.getCPtr(type)); + } + + public void replace_invlists(InvertedLists il, boolean own) { + swigfaissJNI.IndexBinaryIVF_replace_invlists__SWIG_0(swigCPtr, this, InvertedLists.getCPtr(il), il, own); + } + + public void replace_invlists(InvertedLists il) { + swigfaissJNI.IndexBinaryIVF_replace_invlists__SWIG_1(swigCPtr, this, InvertedLists.getCPtr(il), il); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/IndexFlat.java b/ann/src/main/java/com/twitter/ann/faiss/swig/IndexFlat.java new file mode 100644 index 0000000000..2408455f20 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/IndexFlat.java @@ -0,0 +1,85 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class IndexFlat extends IndexFlatCodes { + private transient long swigCPtr; + + protected IndexFlat(long cPtr, boolean cMemoryOwn) { + super(swigfaissJNI.IndexFlat_SWIGUpcast(cPtr), cMemoryOwn); + swigCPtr = cPtr; + } + + protected static long getCPtr(IndexFlat obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_IndexFlat(swigCPtr); + } + swigCPtr = 0; + } + super.delete(); + } + + public IndexFlat(long d, MetricType metric) { + this(swigfaissJNI.new_IndexFlat__SWIG_0(d, metric.swigValue()), true); + } + + public IndexFlat(long d) { + this(swigfaissJNI.new_IndexFlat__SWIG_1(d), true); + } + + public void search(long n, SWIGTYPE_p_float x, long k, SWIGTYPE_p_float distances, LongVector labels) { + swigfaissJNI.IndexFlat_search(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x), k, SWIGTYPE_p_float.getCPtr(distances), SWIGTYPE_p_long_long.getCPtr(labels.data()), labels); + } + + public void range_search(long n, SWIGTYPE_p_float x, float radius, RangeSearchResult result) { + swigfaissJNI.IndexFlat_range_search(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x), radius, RangeSearchResult.getCPtr(result), result); + } + + public void reconstruct(long key, SWIGTYPE_p_float recons) { + swigfaissJNI.IndexFlat_reconstruct(swigCPtr, this, key, SWIGTYPE_p_float.getCPtr(recons)); + } + + public void compute_distance_subset(long n, SWIGTYPE_p_float x, long k, SWIGTYPE_p_float distances, LongVector labels) { + swigfaissJNI.IndexFlat_compute_distance_subset(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x), k, SWIGTYPE_p_float.getCPtr(distances), SWIGTYPE_p_long_long.getCPtr(labels.data()), labels); + } + + public SWIGTYPE_p_float get_xb() { + long cPtr = swigfaissJNI.IndexFlat_get_xb__SWIG_0(swigCPtr, this); + return (cPtr == 0) ? null : new SWIGTYPE_p_float(cPtr, false); + } + + public IndexFlat() { + this(swigfaissJNI.new_IndexFlat__SWIG_2(), true); + } + + public DistanceComputer get_distance_computer() { + long cPtr = swigfaissJNI.IndexFlat_get_distance_computer(swigCPtr, this); + return (cPtr == 0) ? null : new DistanceComputer(cPtr, false); + } + + public void sa_encode(long n, SWIGTYPE_p_float x, SWIGTYPE_p_unsigned_char bytes) { + swigfaissJNI.IndexFlat_sa_encode(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x), SWIGTYPE_p_unsigned_char.getCPtr(bytes)); + } + + public void sa_decode(long n, SWIGTYPE_p_unsigned_char bytes, SWIGTYPE_p_float x) { + swigfaissJNI.IndexFlat_sa_decode(swigCPtr, this, n, SWIGTYPE_p_unsigned_char.getCPtr(bytes), SWIGTYPE_p_float.getCPtr(x)); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/IndexFlat1D.java b/ann/src/main/java/com/twitter/ann/faiss/swig/IndexFlat1D.java new file mode 100644 index 0000000000..8104181b48 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/IndexFlat1D.java @@ -0,0 +1,80 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class IndexFlat1D extends IndexFlatL2 { + private transient long swigCPtr; + + protected IndexFlat1D(long cPtr, boolean cMemoryOwn) { + super(swigfaissJNI.IndexFlat1D_SWIGUpcast(cPtr), cMemoryOwn); + swigCPtr = cPtr; + } + + protected static long getCPtr(IndexFlat1D obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_IndexFlat1D(swigCPtr); + } + swigCPtr = 0; + } + super.delete(); + } + + public void setContinuous_update(boolean value) { + swigfaissJNI.IndexFlat1D_continuous_update_set(swigCPtr, this, value); + } + + public boolean getContinuous_update() { + return swigfaissJNI.IndexFlat1D_continuous_update_get(swigCPtr, this); + } + + public void setPerm(SWIGTYPE_p_std__vectorT_int64_t_t value) { + swigfaissJNI.IndexFlat1D_perm_set(swigCPtr, this, SWIGTYPE_p_std__vectorT_int64_t_t.getCPtr(value)); + } + + public SWIGTYPE_p_std__vectorT_int64_t_t getPerm() { + long cPtr = swigfaissJNI.IndexFlat1D_perm_get(swigCPtr, this); + return (cPtr == 0) ? null : new SWIGTYPE_p_std__vectorT_int64_t_t(cPtr, false); + } + + public IndexFlat1D(boolean continuous_update) { + this(swigfaissJNI.new_IndexFlat1D__SWIG_0(continuous_update), true); + } + + public IndexFlat1D() { + this(swigfaissJNI.new_IndexFlat1D__SWIG_1(), true); + } + + public void update_permutation() { + swigfaissJNI.IndexFlat1D_update_permutation(swigCPtr, this); + } + + public void add(long n, SWIGTYPE_p_float x) { + swigfaissJNI.IndexFlat1D_add(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x)); + } + + public void reset() { + swigfaissJNI.IndexFlat1D_reset(swigCPtr, this); + } + + public void search(long n, SWIGTYPE_p_float x, long k, SWIGTYPE_p_float distances, LongVector labels) { + swigfaissJNI.IndexFlat1D_search(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x), k, SWIGTYPE_p_float.getCPtr(distances), SWIGTYPE_p_long_long.getCPtr(labels.data()), labels); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/IndexFlatCodes.java b/ann/src/main/java/com/twitter/ann/faiss/swig/IndexFlatCodes.java new file mode 100644 index 0000000000..62448b5426 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/IndexFlatCodes.java @@ -0,0 +1,80 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class IndexFlatCodes extends Index { + private transient long swigCPtr; + + protected IndexFlatCodes(long cPtr, boolean cMemoryOwn) { + super(swigfaissJNI.IndexFlatCodes_SWIGUpcast(cPtr), cMemoryOwn); + swigCPtr = cPtr; + } + + protected static long getCPtr(IndexFlatCodes obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_IndexFlatCodes(swigCPtr); + } + swigCPtr = 0; + } + super.delete(); + } + + public void setCode_size(long value) { + swigfaissJNI.IndexFlatCodes_code_size_set(swigCPtr, this, value); + } + + public long getCode_size() { + return swigfaissJNI.IndexFlatCodes_code_size_get(swigCPtr, this); + } + + public void setCodes(ByteVector value) { + swigfaissJNI.IndexFlatCodes_codes_set(swigCPtr, this, ByteVector.getCPtr(value), value); + } + + public ByteVector getCodes() { + long cPtr = swigfaissJNI.IndexFlatCodes_codes_get(swigCPtr, this); + return (cPtr == 0) ? null : new ByteVector(cPtr, false); + } + + public void add(long n, SWIGTYPE_p_float x) { + swigfaissJNI.IndexFlatCodes_add(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x)); + } + + public void reset() { + swigfaissJNI.IndexFlatCodes_reset(swigCPtr, this); + } + + public void reconstruct_n(long i0, long ni, SWIGTYPE_p_float recons) { + swigfaissJNI.IndexFlatCodes_reconstruct_n(swigCPtr, this, i0, ni, SWIGTYPE_p_float.getCPtr(recons)); + } + + public void reconstruct(long key, SWIGTYPE_p_float recons) { + swigfaissJNI.IndexFlatCodes_reconstruct(swigCPtr, this, key, SWIGTYPE_p_float.getCPtr(recons)); + } + + public long sa_code_size() { + return swigfaissJNI.IndexFlatCodes_sa_code_size(swigCPtr, this); + } + + public long remove_ids(IDSelector sel) { + return swigfaissJNI.IndexFlatCodes_remove_ids(swigCPtr, this, IDSelector.getCPtr(sel), sel); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/IndexFlatIP.java b/ann/src/main/java/com/twitter/ann/faiss/swig/IndexFlatIP.java new file mode 100644 index 0000000000..d1cb9c9ff2 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/IndexFlatIP.java @@ -0,0 +1,47 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class IndexFlatIP extends IndexFlat { + private transient long swigCPtr; + + protected IndexFlatIP(long cPtr, boolean cMemoryOwn) { + super(swigfaissJNI.IndexFlatIP_SWIGUpcast(cPtr), cMemoryOwn); + swigCPtr = cPtr; + } + + protected static long getCPtr(IndexFlatIP obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_IndexFlatIP(swigCPtr); + } + swigCPtr = 0; + } + super.delete(); + } + + public IndexFlatIP(long d) { + this(swigfaissJNI.new_IndexFlatIP__SWIG_0(d), true); + } + + public IndexFlatIP() { + this(swigfaissJNI.new_IndexFlatIP__SWIG_1(), true); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/IndexFlatL2.java b/ann/src/main/java/com/twitter/ann/faiss/swig/IndexFlatL2.java new file mode 100644 index 0000000000..9bd6ae092c --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/IndexFlatL2.java @@ -0,0 +1,47 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class IndexFlatL2 extends IndexFlat { + private transient long swigCPtr; + + protected IndexFlatL2(long cPtr, boolean cMemoryOwn) { + super(swigfaissJNI.IndexFlatL2_SWIGUpcast(cPtr), cMemoryOwn); + swigCPtr = cPtr; + } + + protected static long getCPtr(IndexFlatL2 obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_IndexFlatL2(swigCPtr); + } + swigCPtr = 0; + } + super.delete(); + } + + public IndexFlatL2(long d) { + this(swigfaissJNI.new_IndexFlatL2__SWIG_0(d), true); + } + + public IndexFlatL2() { + this(swigfaissJNI.new_IndexFlatL2__SWIG_1(), true); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/IndexHNSW.java b/ann/src/main/java/com/twitter/ann/faiss/swig/IndexHNSW.java new file mode 100644 index 0000000000..b5be7d13ca --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/IndexHNSW.java @@ -0,0 +1,150 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class IndexHNSW extends Index { + private transient long swigCPtr; + + protected IndexHNSW(long cPtr, boolean cMemoryOwn) { + super(swigfaissJNI.IndexHNSW_SWIGUpcast(cPtr), cMemoryOwn); + swigCPtr = cPtr; + } + + protected static long getCPtr(IndexHNSW obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_IndexHNSW(swigCPtr); + } + swigCPtr = 0; + } + super.delete(); + } + + public void setHnsw(HNSW value) { + swigfaissJNI.IndexHNSW_hnsw_set(swigCPtr, this, HNSW.getCPtr(value), value); + } + + public HNSW getHnsw() { + long cPtr = swigfaissJNI.IndexHNSW_hnsw_get(swigCPtr, this); + return (cPtr == 0) ? null : new HNSW(cPtr, false); + } + + public void setOwn_fields(boolean value) { + swigfaissJNI.IndexHNSW_own_fields_set(swigCPtr, this, value); + } + + public boolean getOwn_fields() { + return swigfaissJNI.IndexHNSW_own_fields_get(swigCPtr, this); + } + + public void setStorage(Index value) { + swigfaissJNI.IndexHNSW_storage_set(swigCPtr, this, Index.getCPtr(value), value); + } + + public Index getStorage() { + long cPtr = swigfaissJNI.IndexHNSW_storage_get(swigCPtr, this); + return (cPtr == 0) ? null : new Index(cPtr, false); + } + + public void setReconstruct_from_neighbors(ReconstructFromNeighbors value) { + swigfaissJNI.IndexHNSW_reconstruct_from_neighbors_set(swigCPtr, this, ReconstructFromNeighbors.getCPtr(value), value); + } + + public ReconstructFromNeighbors getReconstruct_from_neighbors() { + long cPtr = swigfaissJNI.IndexHNSW_reconstruct_from_neighbors_get(swigCPtr, this); + return (cPtr == 0) ? null : new ReconstructFromNeighbors(cPtr, false); + } + + public IndexHNSW(int d, int M, MetricType metric) { + this(swigfaissJNI.new_IndexHNSW__SWIG_0(d, M, metric.swigValue()), true); + } + + public IndexHNSW(int d, int M) { + this(swigfaissJNI.new_IndexHNSW__SWIG_1(d, M), true); + } + + public IndexHNSW(int d) { + this(swigfaissJNI.new_IndexHNSW__SWIG_2(d), true); + } + + public IndexHNSW() { + this(swigfaissJNI.new_IndexHNSW__SWIG_3(), true); + } + + public IndexHNSW(Index storage, int M) { + this(swigfaissJNI.new_IndexHNSW__SWIG_4(Index.getCPtr(storage), storage, M), true); + } + + public IndexHNSW(Index storage) { + this(swigfaissJNI.new_IndexHNSW__SWIG_5(Index.getCPtr(storage), storage), true); + } + + public void add(long n, SWIGTYPE_p_float x) { + swigfaissJNI.IndexHNSW_add(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x)); + } + + public void train(long n, SWIGTYPE_p_float x) { + swigfaissJNI.IndexHNSW_train(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x)); + } + + public void search(long n, SWIGTYPE_p_float x, long k, SWIGTYPE_p_float distances, LongVector labels) { + swigfaissJNI.IndexHNSW_search(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x), k, SWIGTYPE_p_float.getCPtr(distances), SWIGTYPE_p_long_long.getCPtr(labels.data()), labels); + } + + public void reconstruct(long key, SWIGTYPE_p_float recons) { + swigfaissJNI.IndexHNSW_reconstruct(swigCPtr, this, key, SWIGTYPE_p_float.getCPtr(recons)); + } + + public void reset() { + swigfaissJNI.IndexHNSW_reset(swigCPtr, this); + } + + public void shrink_level_0_neighbors(int size) { + swigfaissJNI.IndexHNSW_shrink_level_0_neighbors(swigCPtr, this, size); + } + + public void search_level_0(long n, SWIGTYPE_p_float x, long k, SWIGTYPE_p_int nearest, SWIGTYPE_p_float nearest_d, SWIGTYPE_p_float distances, LongVector labels, int nprobe, int search_type) { + swigfaissJNI.IndexHNSW_search_level_0__SWIG_0(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x), k, SWIGTYPE_p_int.getCPtr(nearest), SWIGTYPE_p_float.getCPtr(nearest_d), SWIGTYPE_p_float.getCPtr(distances), SWIGTYPE_p_long_long.getCPtr(labels.data()), labels, nprobe, search_type); + } + + public void search_level_0(long n, SWIGTYPE_p_float x, long k, SWIGTYPE_p_int nearest, SWIGTYPE_p_float nearest_d, SWIGTYPE_p_float distances, LongVector labels, int nprobe) { + swigfaissJNI.IndexHNSW_search_level_0__SWIG_1(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x), k, SWIGTYPE_p_int.getCPtr(nearest), SWIGTYPE_p_float.getCPtr(nearest_d), SWIGTYPE_p_float.getCPtr(distances), SWIGTYPE_p_long_long.getCPtr(labels.data()), labels, nprobe); + } + + public void search_level_0(long n, SWIGTYPE_p_float x, long k, SWIGTYPE_p_int nearest, SWIGTYPE_p_float nearest_d, SWIGTYPE_p_float distances, LongVector labels) { + swigfaissJNI.IndexHNSW_search_level_0__SWIG_2(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x), k, SWIGTYPE_p_int.getCPtr(nearest), SWIGTYPE_p_float.getCPtr(nearest_d), SWIGTYPE_p_float.getCPtr(distances), SWIGTYPE_p_long_long.getCPtr(labels.data()), labels); + } + + public void init_level_0_from_knngraph(int k, SWIGTYPE_p_float D, LongVector I) { + swigfaissJNI.IndexHNSW_init_level_0_from_knngraph(swigCPtr, this, k, SWIGTYPE_p_float.getCPtr(D), SWIGTYPE_p_long_long.getCPtr(I.data()), I); + } + + public void init_level_0_from_entry_points(int npt, SWIGTYPE_p_int points, SWIGTYPE_p_int nearests) { + swigfaissJNI.IndexHNSW_init_level_0_from_entry_points(swigCPtr, this, npt, SWIGTYPE_p_int.getCPtr(points), SWIGTYPE_p_int.getCPtr(nearests)); + } + + public void reorder_links() { + swigfaissJNI.IndexHNSW_reorder_links(swigCPtr, this); + } + + public void link_singletons() { + swigfaissJNI.IndexHNSW_link_singletons(swigCPtr, this); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/IndexHNSW2Level.java b/ann/src/main/java/com/twitter/ann/faiss/swig/IndexHNSW2Level.java new file mode 100644 index 0000000000..9f1544ca12 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/IndexHNSW2Level.java @@ -0,0 +1,55 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class IndexHNSW2Level extends IndexHNSW { + private transient long swigCPtr; + + protected IndexHNSW2Level(long cPtr, boolean cMemoryOwn) { + super(swigfaissJNI.IndexHNSW2Level_SWIGUpcast(cPtr), cMemoryOwn); + swigCPtr = cPtr; + } + + protected static long getCPtr(IndexHNSW2Level obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_IndexHNSW2Level(swigCPtr); + } + swigCPtr = 0; + } + super.delete(); + } + + public IndexHNSW2Level() { + this(swigfaissJNI.new_IndexHNSW2Level__SWIG_0(), true); + } + + public IndexHNSW2Level(Index quantizer, long nlist, int m_pq, int M) { + this(swigfaissJNI.new_IndexHNSW2Level__SWIG_1(Index.getCPtr(quantizer), quantizer, nlist, m_pq, M), true); + } + + public void flip_to_ivf() { + swigfaissJNI.IndexHNSW2Level_flip_to_ivf(swigCPtr, this); + } + + public void search(long n, SWIGTYPE_p_float x, long k, SWIGTYPE_p_float distances, LongVector labels) { + swigfaissJNI.IndexHNSW2Level_search(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x), k, SWIGTYPE_p_float.getCPtr(distances), SWIGTYPE_p_long_long.getCPtr(labels.data()), labels); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/IndexHNSWFlat.java b/ann/src/main/java/com/twitter/ann/faiss/swig/IndexHNSWFlat.java new file mode 100644 index 0000000000..c632b4fb6c --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/IndexHNSWFlat.java @@ -0,0 +1,51 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class IndexHNSWFlat extends IndexHNSW { + private transient long swigCPtr; + + protected IndexHNSWFlat(long cPtr, boolean cMemoryOwn) { + super(swigfaissJNI.IndexHNSWFlat_SWIGUpcast(cPtr), cMemoryOwn); + swigCPtr = cPtr; + } + + protected static long getCPtr(IndexHNSWFlat obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_IndexHNSWFlat(swigCPtr); + } + swigCPtr = 0; + } + super.delete(); + } + + public IndexHNSWFlat() { + this(swigfaissJNI.new_IndexHNSWFlat__SWIG_0(), true); + } + + public IndexHNSWFlat(int d, int M, MetricType metric) { + this(swigfaissJNI.new_IndexHNSWFlat__SWIG_1(d, M, metric.swigValue()), true); + } + + public IndexHNSWFlat(int d, int M) { + this(swigfaissJNI.new_IndexHNSWFlat__SWIG_2(d, M), true); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/IndexHNSWPQ.java b/ann/src/main/java/com/twitter/ann/faiss/swig/IndexHNSWPQ.java new file mode 100644 index 0000000000..cc761d9c83 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/IndexHNSWPQ.java @@ -0,0 +1,51 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class IndexHNSWPQ extends IndexHNSW { + private transient long swigCPtr; + + protected IndexHNSWPQ(long cPtr, boolean cMemoryOwn) { + super(swigfaissJNI.IndexHNSWPQ_SWIGUpcast(cPtr), cMemoryOwn); + swigCPtr = cPtr; + } + + protected static long getCPtr(IndexHNSWPQ obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_IndexHNSWPQ(swigCPtr); + } + swigCPtr = 0; + } + super.delete(); + } + + public IndexHNSWPQ() { + this(swigfaissJNI.new_IndexHNSWPQ__SWIG_0(), true); + } + + public IndexHNSWPQ(int d, int pq_m, int M) { + this(swigfaissJNI.new_IndexHNSWPQ__SWIG_1(d, pq_m, M), true); + } + + public void train(long n, SWIGTYPE_p_float x) { + swigfaissJNI.IndexHNSWPQ_train(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x)); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/IndexHNSWSQ.java b/ann/src/main/java/com/twitter/ann/faiss/swig/IndexHNSWSQ.java new file mode 100644 index 0000000000..ee08aa9bb5 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/IndexHNSWSQ.java @@ -0,0 +1,51 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class IndexHNSWSQ extends IndexHNSW { + private transient long swigCPtr; + + protected IndexHNSWSQ(long cPtr, boolean cMemoryOwn) { + super(swigfaissJNI.IndexHNSWSQ_SWIGUpcast(cPtr), cMemoryOwn); + swigCPtr = cPtr; + } + + protected static long getCPtr(IndexHNSWSQ obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_IndexHNSWSQ(swigCPtr); + } + swigCPtr = 0; + } + super.delete(); + } + + public IndexHNSWSQ() { + this(swigfaissJNI.new_IndexHNSWSQ__SWIG_0(), true); + } + + public IndexHNSWSQ(int d, SWIGTYPE_p_ScalarQuantizer__QuantizerType qtype, int M, MetricType metric) { + this(swigfaissJNI.new_IndexHNSWSQ__SWIG_1(d, SWIGTYPE_p_ScalarQuantizer__QuantizerType.getCPtr(qtype), M, metric.swigValue()), true); + } + + public IndexHNSWSQ(int d, SWIGTYPE_p_ScalarQuantizer__QuantizerType qtype, int M) { + this(swigfaissJNI.new_IndexHNSWSQ__SWIG_2(d, SWIGTYPE_p_ScalarQuantizer__QuantizerType.getCPtr(qtype), M), true); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/IndexIDMap.java b/ann/src/main/java/com/twitter/ann/faiss/swig/IndexIDMap.java new file mode 100644 index 0000000000..72c5574f94 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/IndexIDMap.java @@ -0,0 +1,101 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class IndexIDMap extends Index { + private transient long swigCPtr; + + protected IndexIDMap(long cPtr, boolean cMemoryOwn) { + super(swigfaissJNI.IndexIDMap_SWIGUpcast(cPtr), cMemoryOwn); + swigCPtr = cPtr; + } + + protected static long getCPtr(IndexIDMap obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_IndexIDMap(swigCPtr); + } + swigCPtr = 0; + } + super.delete(); + } + + public void setIndex(Index value) { + swigfaissJNI.IndexIDMap_index_set(swigCPtr, this, Index.getCPtr(value), value); + } + + public Index getIndex() { + long cPtr = swigfaissJNI.IndexIDMap_index_get(swigCPtr, this); + return (cPtr == 0) ? null : new Index(cPtr, false); + } + + public void setOwn_fields(boolean value) { + swigfaissJNI.IndexIDMap_own_fields_set(swigCPtr, this, value); + } + + public boolean getOwn_fields() { + return swigfaissJNI.IndexIDMap_own_fields_get(swigCPtr, this); + } + + public void setId_map(SWIGTYPE_p_std__vectorT_int64_t_t value) { + swigfaissJNI.IndexIDMap_id_map_set(swigCPtr, this, SWIGTYPE_p_std__vectorT_int64_t_t.getCPtr(value)); + } + + public SWIGTYPE_p_std__vectorT_int64_t_t getId_map() { + long cPtr = swigfaissJNI.IndexIDMap_id_map_get(swigCPtr, this); + return (cPtr == 0) ? null : new SWIGTYPE_p_std__vectorT_int64_t_t(cPtr, false); + } + + public IndexIDMap(Index index) { + this(swigfaissJNI.new_IndexIDMap__SWIG_0(Index.getCPtr(index), index), true); + } + + public void add_with_ids(long n, SWIGTYPE_p_float x, LongVector xids) { + swigfaissJNI.IndexIDMap_add_with_ids(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x), SWIGTYPE_p_long_long.getCPtr(xids.data()), xids); + } + + public void add(long n, SWIGTYPE_p_float x) { + swigfaissJNI.IndexIDMap_add(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x)); + } + + public void search(long n, SWIGTYPE_p_float x, long k, SWIGTYPE_p_float distances, LongVector labels) { + swigfaissJNI.IndexIDMap_search(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x), k, SWIGTYPE_p_float.getCPtr(distances), SWIGTYPE_p_long_long.getCPtr(labels.data()), labels); + } + + public void train(long n, SWIGTYPE_p_float x) { + swigfaissJNI.IndexIDMap_train(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x)); + } + + public void reset() { + swigfaissJNI.IndexIDMap_reset(swigCPtr, this); + } + + public long remove_ids(IDSelector sel) { + return swigfaissJNI.IndexIDMap_remove_ids(swigCPtr, this, IDSelector.getCPtr(sel), sel); + } + + public void range_search(long n, SWIGTYPE_p_float x, float radius, RangeSearchResult result) { + swigfaissJNI.IndexIDMap_range_search(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x), radius, RangeSearchResult.getCPtr(result), result); + } + + public IndexIDMap() { + this(swigfaissJNI.new_IndexIDMap__SWIG_1(), true); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/IndexIVF.java b/ann/src/main/java/com/twitter/ann/faiss/swig/IndexIVF.java new file mode 100644 index 0000000000..04c22067a6 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/IndexIVF.java @@ -0,0 +1,250 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class IndexIVF extends Index { + private transient long swigCPtr; + + protected IndexIVF(long cPtr, boolean cMemoryOwn) { + super(swigfaissJNI.IndexIVF_SWIGUpcast(cPtr), cMemoryOwn); + swigCPtr = cPtr; + } + + protected static long getCPtr(IndexIVF obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_IndexIVF(swigCPtr); + } + swigCPtr = 0; + } + super.delete(); + } + + public void setInvlists(InvertedLists value) { + swigfaissJNI.IndexIVF_invlists_set(swigCPtr, this, InvertedLists.getCPtr(value), value); + } + + public InvertedLists getInvlists() { + long cPtr = swigfaissJNI.IndexIVF_invlists_get(swigCPtr, this); + return (cPtr == 0) ? null : new InvertedLists(cPtr, false); + } + + public void setOwn_invlists(boolean value) { + swigfaissJNI.IndexIVF_own_invlists_set(swigCPtr, this, value); + } + + public boolean getOwn_invlists() { + return swigfaissJNI.IndexIVF_own_invlists_get(swigCPtr, this); + } + + public void setCode_size(long value) { + swigfaissJNI.IndexIVF_code_size_set(swigCPtr, this, value); + } + + public long getCode_size() { + return swigfaissJNI.IndexIVF_code_size_get(swigCPtr, this); + } + + public void setNprobe(long value) { + swigfaissJNI.IndexIVF_nprobe_set(swigCPtr, this, value); + } + + public long getNprobe() { + return swigfaissJNI.IndexIVF_nprobe_get(swigCPtr, this); + } + + public void setMax_codes(long value) { + swigfaissJNI.IndexIVF_max_codes_set(swigCPtr, this, value); + } + + public long getMax_codes() { + return swigfaissJNI.IndexIVF_max_codes_get(swigCPtr, this); + } + + public void setParallel_mode(int value) { + swigfaissJNI.IndexIVF_parallel_mode_set(swigCPtr, this, value); + } + + public int getParallel_mode() { + return swigfaissJNI.IndexIVF_parallel_mode_get(swigCPtr, this); + } + + public int getPARALLEL_MODE_NO_HEAP_INIT() { + return swigfaissJNI.IndexIVF_PARALLEL_MODE_NO_HEAP_INIT_get(swigCPtr, this); + } + + public void setDirect_map(SWIGTYPE_p_DirectMap value) { + swigfaissJNI.IndexIVF_direct_map_set(swigCPtr, this, SWIGTYPE_p_DirectMap.getCPtr(value)); + } + + public SWIGTYPE_p_DirectMap getDirect_map() { + return new SWIGTYPE_p_DirectMap(swigfaissJNI.IndexIVF_direct_map_get(swigCPtr, this), true); + } + + public void reset() { + swigfaissJNI.IndexIVF_reset(swigCPtr, this); + } + + public void train(long n, SWIGTYPE_p_float x) { + swigfaissJNI.IndexIVF_train(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x)); + } + + public void add(long n, SWIGTYPE_p_float x) { + swigfaissJNI.IndexIVF_add(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x)); + } + + public void add_with_ids(long n, SWIGTYPE_p_float x, LongVector xids) { + swigfaissJNI.IndexIVF_add_with_ids(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x), SWIGTYPE_p_long_long.getCPtr(xids.data()), xids); + } + + public void add_core(long n, SWIGTYPE_p_float x, LongVector xids, LongVector precomputed_idx) { + swigfaissJNI.IndexIVF_add_core(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x), SWIGTYPE_p_long_long.getCPtr(xids.data()), xids, SWIGTYPE_p_long_long.getCPtr(precomputed_idx.data()), precomputed_idx); + } + + public void encode_vectors(long n, SWIGTYPE_p_float x, LongVector list_nos, SWIGTYPE_p_unsigned_char codes, boolean include_listno) { + swigfaissJNI.IndexIVF_encode_vectors__SWIG_0(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x), SWIGTYPE_p_long_long.getCPtr(list_nos.data()), list_nos, SWIGTYPE_p_unsigned_char.getCPtr(codes), include_listno); + } + + public void encode_vectors(long n, SWIGTYPE_p_float x, LongVector list_nos, SWIGTYPE_p_unsigned_char codes) { + swigfaissJNI.IndexIVF_encode_vectors__SWIG_1(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x), SWIGTYPE_p_long_long.getCPtr(list_nos.data()), list_nos, SWIGTYPE_p_unsigned_char.getCPtr(codes)); + } + + public void add_sa_codes(long n, SWIGTYPE_p_unsigned_char codes, LongVector xids) { + swigfaissJNI.IndexIVF_add_sa_codes(swigCPtr, this, n, SWIGTYPE_p_unsigned_char.getCPtr(codes), SWIGTYPE_p_long_long.getCPtr(xids.data()), xids); + } + + public void train_residual(long n, SWIGTYPE_p_float x) { + swigfaissJNI.IndexIVF_train_residual(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x)); + } + + public void search_preassigned(long n, SWIGTYPE_p_float x, long k, LongVector assign, SWIGTYPE_p_float centroid_dis, SWIGTYPE_p_float distances, LongVector labels, boolean store_pairs, IVFSearchParameters params, IndexIVFStats stats) { + swigfaissJNI.IndexIVF_search_preassigned__SWIG_0(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x), k, SWIGTYPE_p_long_long.getCPtr(assign.data()), assign, SWIGTYPE_p_float.getCPtr(centroid_dis), SWIGTYPE_p_float.getCPtr(distances), SWIGTYPE_p_long_long.getCPtr(labels.data()), labels, store_pairs, IVFSearchParameters.getCPtr(params), params, IndexIVFStats.getCPtr(stats), stats); + } + + public void search_preassigned(long n, SWIGTYPE_p_float x, long k, LongVector assign, SWIGTYPE_p_float centroid_dis, SWIGTYPE_p_float distances, LongVector labels, boolean store_pairs, IVFSearchParameters params) { + swigfaissJNI.IndexIVF_search_preassigned__SWIG_1(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x), k, SWIGTYPE_p_long_long.getCPtr(assign.data()), assign, SWIGTYPE_p_float.getCPtr(centroid_dis), SWIGTYPE_p_float.getCPtr(distances), SWIGTYPE_p_long_long.getCPtr(labels.data()), labels, store_pairs, IVFSearchParameters.getCPtr(params), params); + } + + public void search_preassigned(long n, SWIGTYPE_p_float x, long k, LongVector assign, SWIGTYPE_p_float centroid_dis, SWIGTYPE_p_float distances, LongVector labels, boolean store_pairs) { + swigfaissJNI.IndexIVF_search_preassigned__SWIG_2(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x), k, SWIGTYPE_p_long_long.getCPtr(assign.data()), assign, SWIGTYPE_p_float.getCPtr(centroid_dis), SWIGTYPE_p_float.getCPtr(distances), SWIGTYPE_p_long_long.getCPtr(labels.data()), labels, store_pairs); + } + + public void search(long n, SWIGTYPE_p_float x, long k, SWIGTYPE_p_float distances, LongVector labels) { + swigfaissJNI.IndexIVF_search(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x), k, SWIGTYPE_p_float.getCPtr(distances), SWIGTYPE_p_long_long.getCPtr(labels.data()), labels); + } + + public void range_search(long n, SWIGTYPE_p_float x, float radius, RangeSearchResult result) { + swigfaissJNI.IndexIVF_range_search(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x), radius, RangeSearchResult.getCPtr(result), result); + } + + public void range_search_preassigned(long nx, SWIGTYPE_p_float x, float radius, LongVector keys, SWIGTYPE_p_float coarse_dis, RangeSearchResult result, boolean store_pairs, IVFSearchParameters params, IndexIVFStats stats) { + swigfaissJNI.IndexIVF_range_search_preassigned__SWIG_0(swigCPtr, this, nx, SWIGTYPE_p_float.getCPtr(x), radius, SWIGTYPE_p_long_long.getCPtr(keys.data()), keys, SWIGTYPE_p_float.getCPtr(coarse_dis), RangeSearchResult.getCPtr(result), result, store_pairs, IVFSearchParameters.getCPtr(params), params, IndexIVFStats.getCPtr(stats), stats); + } + + public void range_search_preassigned(long nx, SWIGTYPE_p_float x, float radius, LongVector keys, SWIGTYPE_p_float coarse_dis, RangeSearchResult result, boolean store_pairs, IVFSearchParameters params) { + swigfaissJNI.IndexIVF_range_search_preassigned__SWIG_1(swigCPtr, this, nx, SWIGTYPE_p_float.getCPtr(x), radius, SWIGTYPE_p_long_long.getCPtr(keys.data()), keys, SWIGTYPE_p_float.getCPtr(coarse_dis), RangeSearchResult.getCPtr(result), result, store_pairs, IVFSearchParameters.getCPtr(params), params); + } + + public void range_search_preassigned(long nx, SWIGTYPE_p_float x, float radius, LongVector keys, SWIGTYPE_p_float coarse_dis, RangeSearchResult result, boolean store_pairs) { + swigfaissJNI.IndexIVF_range_search_preassigned__SWIG_2(swigCPtr, this, nx, SWIGTYPE_p_float.getCPtr(x), radius, SWIGTYPE_p_long_long.getCPtr(keys.data()), keys, SWIGTYPE_p_float.getCPtr(coarse_dis), RangeSearchResult.getCPtr(result), result, store_pairs); + } + + public void range_search_preassigned(long nx, SWIGTYPE_p_float x, float radius, LongVector keys, SWIGTYPE_p_float coarse_dis, RangeSearchResult result) { + swigfaissJNI.IndexIVF_range_search_preassigned__SWIG_3(swigCPtr, this, nx, SWIGTYPE_p_float.getCPtr(x), radius, SWIGTYPE_p_long_long.getCPtr(keys.data()), keys, SWIGTYPE_p_float.getCPtr(coarse_dis), RangeSearchResult.getCPtr(result), result); + } + + public SWIGTYPE_p_faiss__InvertedListScanner get_InvertedListScanner(boolean store_pairs) { + long cPtr = swigfaissJNI.IndexIVF_get_InvertedListScanner__SWIG_0(swigCPtr, this, store_pairs); + return (cPtr == 0) ? null : new SWIGTYPE_p_faiss__InvertedListScanner(cPtr, false); + } + + public SWIGTYPE_p_faiss__InvertedListScanner get_InvertedListScanner() { + long cPtr = swigfaissJNI.IndexIVF_get_InvertedListScanner__SWIG_1(swigCPtr, this); + return (cPtr == 0) ? null : new SWIGTYPE_p_faiss__InvertedListScanner(cPtr, false); + } + + public void reconstruct(long key, SWIGTYPE_p_float recons) { + swigfaissJNI.IndexIVF_reconstruct(swigCPtr, this, key, SWIGTYPE_p_float.getCPtr(recons)); + } + + public void update_vectors(int nv, LongVector idx, SWIGTYPE_p_float v) { + swigfaissJNI.IndexIVF_update_vectors(swigCPtr, this, nv, SWIGTYPE_p_long_long.getCPtr(idx.data()), idx, SWIGTYPE_p_float.getCPtr(v)); + } + + public void reconstruct_n(long i0, long ni, SWIGTYPE_p_float recons) { + swigfaissJNI.IndexIVF_reconstruct_n(swigCPtr, this, i0, ni, SWIGTYPE_p_float.getCPtr(recons)); + } + + public void search_and_reconstruct(long n, SWIGTYPE_p_float x, long k, SWIGTYPE_p_float distances, LongVector labels, SWIGTYPE_p_float recons) { + swigfaissJNI.IndexIVF_search_and_reconstruct(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x), k, SWIGTYPE_p_float.getCPtr(distances), SWIGTYPE_p_long_long.getCPtr(labels.data()), labels, SWIGTYPE_p_float.getCPtr(recons)); + } + + public void reconstruct_from_offset(long list_no, long offset, SWIGTYPE_p_float recons) { + swigfaissJNI.IndexIVF_reconstruct_from_offset(swigCPtr, this, list_no, offset, SWIGTYPE_p_float.getCPtr(recons)); + } + + public long remove_ids(IDSelector sel) { + return swigfaissJNI.IndexIVF_remove_ids(swigCPtr, this, IDSelector.getCPtr(sel), sel); + } + + public void check_compatible_for_merge(IndexIVF other) { + swigfaissJNI.IndexIVF_check_compatible_for_merge(swigCPtr, this, IndexIVF.getCPtr(other), other); + } + + public void merge_from(IndexIVF other, long add_id) { + swigfaissJNI.IndexIVF_merge_from(swigCPtr, this, IndexIVF.getCPtr(other), other, add_id); + } + + public void copy_subset_to(IndexIVF other, int subset_type, long a1, long a2) { + swigfaissJNI.IndexIVF_copy_subset_to(swigCPtr, this, IndexIVF.getCPtr(other), other, subset_type, a1, a2); + } + + public long get_list_size(long list_no) { + return swigfaissJNI.IndexIVF_get_list_size(swigCPtr, this, list_no); + } + + public void make_direct_map(boolean new_maintain_direct_map) { + swigfaissJNI.IndexIVF_make_direct_map__SWIG_0(swigCPtr, this, new_maintain_direct_map); + } + + public void make_direct_map() { + swigfaissJNI.IndexIVF_make_direct_map__SWIG_1(swigCPtr, this); + } + + public void set_direct_map_type(SWIGTYPE_p_DirectMap__Type type) { + swigfaissJNI.IndexIVF_set_direct_map_type(swigCPtr, this, SWIGTYPE_p_DirectMap__Type.getCPtr(type)); + } + + public void replace_invlists(InvertedLists il, boolean own) { + swigfaissJNI.IndexIVF_replace_invlists__SWIG_0(swigCPtr, this, InvertedLists.getCPtr(il), il, own); + } + + public void replace_invlists(InvertedLists il) { + swigfaissJNI.IndexIVF_replace_invlists__SWIG_1(swigCPtr, this, InvertedLists.getCPtr(il), il); + } + + public long sa_code_size() { + return swigfaissJNI.IndexIVF_sa_code_size(swigCPtr, this); + } + + public void sa_encode(long n, SWIGTYPE_p_float x, SWIGTYPE_p_unsigned_char bytes) { + swigfaissJNI.IndexIVF_sa_encode(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x), SWIGTYPE_p_unsigned_char.getCPtr(bytes)); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/IndexIVFFlat.java b/ann/src/main/java/com/twitter/ann/faiss/swig/IndexIVFFlat.java new file mode 100644 index 0000000000..f2e4120267 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/IndexIVFFlat.java @@ -0,0 +1,76 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class IndexIVFFlat extends IndexIVF { + private transient long swigCPtr; + + protected IndexIVFFlat(long cPtr, boolean cMemoryOwn) { + super(swigfaissJNI.IndexIVFFlat_SWIGUpcast(cPtr), cMemoryOwn); + swigCPtr = cPtr; + } + + protected static long getCPtr(IndexIVFFlat obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_IndexIVFFlat(swigCPtr); + } + swigCPtr = 0; + } + super.delete(); + } + + public IndexIVFFlat(Index quantizer, long d, long nlist_, MetricType arg3) { + this(swigfaissJNI.new_IndexIVFFlat__SWIG_0(Index.getCPtr(quantizer), quantizer, d, nlist_, arg3.swigValue()), true); + } + + public IndexIVFFlat(Index quantizer, long d, long nlist_) { + this(swigfaissJNI.new_IndexIVFFlat__SWIG_1(Index.getCPtr(quantizer), quantizer, d, nlist_), true); + } + + public void add_core(long n, SWIGTYPE_p_float x, LongVector xids, LongVector precomputed_idx) { + swigfaissJNI.IndexIVFFlat_add_core(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x), SWIGTYPE_p_long_long.getCPtr(xids.data()), xids, SWIGTYPE_p_long_long.getCPtr(precomputed_idx.data()), precomputed_idx); + } + + public void encode_vectors(long n, SWIGTYPE_p_float x, LongVector list_nos, SWIGTYPE_p_unsigned_char codes, boolean include_listnos) { + swigfaissJNI.IndexIVFFlat_encode_vectors__SWIG_0(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x), SWIGTYPE_p_long_long.getCPtr(list_nos.data()), list_nos, SWIGTYPE_p_unsigned_char.getCPtr(codes), include_listnos); + } + + public void encode_vectors(long n, SWIGTYPE_p_float x, LongVector list_nos, SWIGTYPE_p_unsigned_char codes) { + swigfaissJNI.IndexIVFFlat_encode_vectors__SWIG_1(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x), SWIGTYPE_p_long_long.getCPtr(list_nos.data()), list_nos, SWIGTYPE_p_unsigned_char.getCPtr(codes)); + } + + public SWIGTYPE_p_faiss__InvertedListScanner get_InvertedListScanner(boolean store_pairs) { + long cPtr = swigfaissJNI.IndexIVFFlat_get_InvertedListScanner(swigCPtr, this, store_pairs); + return (cPtr == 0) ? null : new SWIGTYPE_p_faiss__InvertedListScanner(cPtr, false); + } + + public void reconstruct_from_offset(long list_no, long offset, SWIGTYPE_p_float recons) { + swigfaissJNI.IndexIVFFlat_reconstruct_from_offset(swigCPtr, this, list_no, offset, SWIGTYPE_p_float.getCPtr(recons)); + } + + public void sa_decode(long n, SWIGTYPE_p_unsigned_char bytes, SWIGTYPE_p_float x) { + swigfaissJNI.IndexIVFFlat_sa_decode(swigCPtr, this, n, SWIGTYPE_p_unsigned_char.getCPtr(bytes), SWIGTYPE_p_float.getCPtr(x)); + } + + public IndexIVFFlat() { + this(swigfaissJNI.new_IndexIVFFlat__SWIG_2(), true); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/IndexIVFFlatDedup.java b/ann/src/main/java/com/twitter/ann/faiss/swig/IndexIVFFlatDedup.java new file mode 100644 index 0000000000..7efd005375 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/IndexIVFFlatDedup.java @@ -0,0 +1,95 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class IndexIVFFlatDedup extends IndexIVFFlat { + private transient long swigCPtr; + + protected IndexIVFFlatDedup(long cPtr, boolean cMemoryOwn) { + super(swigfaissJNI.IndexIVFFlatDedup_SWIGUpcast(cPtr), cMemoryOwn); + swigCPtr = cPtr; + } + + protected static long getCPtr(IndexIVFFlatDedup obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_IndexIVFFlatDedup(swigCPtr); + } + swigCPtr = 0; + } + super.delete(); + } + + public void setInstances(SWIGTYPE_p_std__unordered_multimapT_int64_t_int64_t_t value) { + swigfaissJNI.IndexIVFFlatDedup_instances_set(swigCPtr, this, SWIGTYPE_p_std__unordered_multimapT_int64_t_int64_t_t.getCPtr(value)); + } + + public SWIGTYPE_p_std__unordered_multimapT_int64_t_int64_t_t getInstances() { + return new SWIGTYPE_p_std__unordered_multimapT_int64_t_int64_t_t(swigfaissJNI.IndexIVFFlatDedup_instances_get(swigCPtr, this), true); + } + + public IndexIVFFlatDedup(Index quantizer, long d, long nlist_, MetricType arg3) { + this(swigfaissJNI.new_IndexIVFFlatDedup__SWIG_0(Index.getCPtr(quantizer), quantizer, d, nlist_, arg3.swigValue()), true); + } + + public IndexIVFFlatDedup(Index quantizer, long d, long nlist_) { + this(swigfaissJNI.new_IndexIVFFlatDedup__SWIG_1(Index.getCPtr(quantizer), quantizer, d, nlist_), true); + } + + public void train(long n, SWIGTYPE_p_float x) { + swigfaissJNI.IndexIVFFlatDedup_train(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x)); + } + + public void add_with_ids(long n, SWIGTYPE_p_float x, LongVector xids) { + swigfaissJNI.IndexIVFFlatDedup_add_with_ids(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x), SWIGTYPE_p_long_long.getCPtr(xids.data()), xids); + } + + public void search_preassigned(long n, SWIGTYPE_p_float x, long k, LongVector assign, SWIGTYPE_p_float centroid_dis, SWIGTYPE_p_float distances, LongVector labels, boolean store_pairs, IVFSearchParameters params, IndexIVFStats stats) { + swigfaissJNI.IndexIVFFlatDedup_search_preassigned__SWIG_0(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x), k, SWIGTYPE_p_long_long.getCPtr(assign.data()), assign, SWIGTYPE_p_float.getCPtr(centroid_dis), SWIGTYPE_p_float.getCPtr(distances), SWIGTYPE_p_long_long.getCPtr(labels.data()), labels, store_pairs, IVFSearchParameters.getCPtr(params), params, IndexIVFStats.getCPtr(stats), stats); + } + + public void search_preassigned(long n, SWIGTYPE_p_float x, long k, LongVector assign, SWIGTYPE_p_float centroid_dis, SWIGTYPE_p_float distances, LongVector labels, boolean store_pairs, IVFSearchParameters params) { + swigfaissJNI.IndexIVFFlatDedup_search_preassigned__SWIG_1(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x), k, SWIGTYPE_p_long_long.getCPtr(assign.data()), assign, SWIGTYPE_p_float.getCPtr(centroid_dis), SWIGTYPE_p_float.getCPtr(distances), SWIGTYPE_p_long_long.getCPtr(labels.data()), labels, store_pairs, IVFSearchParameters.getCPtr(params), params); + } + + public void search_preassigned(long n, SWIGTYPE_p_float x, long k, LongVector assign, SWIGTYPE_p_float centroid_dis, SWIGTYPE_p_float distances, LongVector labels, boolean store_pairs) { + swigfaissJNI.IndexIVFFlatDedup_search_preassigned__SWIG_2(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x), k, SWIGTYPE_p_long_long.getCPtr(assign.data()), assign, SWIGTYPE_p_float.getCPtr(centroid_dis), SWIGTYPE_p_float.getCPtr(distances), SWIGTYPE_p_long_long.getCPtr(labels.data()), labels, store_pairs); + } + + public long remove_ids(IDSelector sel) { + return swigfaissJNI.IndexIVFFlatDedup_remove_ids(swigCPtr, this, IDSelector.getCPtr(sel), sel); + } + + public void range_search(long n, SWIGTYPE_p_float x, float radius, RangeSearchResult result) { + swigfaissJNI.IndexIVFFlatDedup_range_search(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x), radius, RangeSearchResult.getCPtr(result), result); + } + + public void update_vectors(int nv, LongVector idx, SWIGTYPE_p_float v) { + swigfaissJNI.IndexIVFFlatDedup_update_vectors(swigCPtr, this, nv, SWIGTYPE_p_long_long.getCPtr(idx.data()), idx, SWIGTYPE_p_float.getCPtr(v)); + } + + public void reconstruct_from_offset(long list_no, long offset, SWIGTYPE_p_float recons) { + swigfaissJNI.IndexIVFFlatDedup_reconstruct_from_offset(swigCPtr, this, list_no, offset, SWIGTYPE_p_float.getCPtr(recons)); + } + + public IndexIVFFlatDedup() { + this(swigfaissJNI.new_IndexIVFFlatDedup__SWIG_2(), true); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/IndexIVFPQ.java b/ann/src/main/java/com/twitter/ann/faiss/swig/IndexIVFPQ.java new file mode 100644 index 0000000000..ba5514a0f6 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/IndexIVFPQ.java @@ -0,0 +1,182 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class IndexIVFPQ extends IndexIVF { + private transient long swigCPtr; + + protected IndexIVFPQ(long cPtr, boolean cMemoryOwn) { + super(swigfaissJNI.IndexIVFPQ_SWIGUpcast(cPtr), cMemoryOwn); + swigCPtr = cPtr; + } + + protected static long getCPtr(IndexIVFPQ obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_IndexIVFPQ(swigCPtr); + } + swigCPtr = 0; + } + super.delete(); + } + + public void setBy_residual(boolean value) { + swigfaissJNI.IndexIVFPQ_by_residual_set(swigCPtr, this, value); + } + + public boolean getBy_residual() { + return swigfaissJNI.IndexIVFPQ_by_residual_get(swigCPtr, this); + } + + public void setPq(ProductQuantizer value) { + swigfaissJNI.IndexIVFPQ_pq_set(swigCPtr, this, ProductQuantizer.getCPtr(value), value); + } + + public ProductQuantizer getPq() { + long cPtr = swigfaissJNI.IndexIVFPQ_pq_get(swigCPtr, this); + return (cPtr == 0) ? null : new ProductQuantizer(cPtr, false); + } + + public void setDo_polysemous_training(boolean value) { + swigfaissJNI.IndexIVFPQ_do_polysemous_training_set(swigCPtr, this, value); + } + + public boolean getDo_polysemous_training() { + return swigfaissJNI.IndexIVFPQ_do_polysemous_training_get(swigCPtr, this); + } + + public void setPolysemous_training(PolysemousTraining value) { + swigfaissJNI.IndexIVFPQ_polysemous_training_set(swigCPtr, this, PolysemousTraining.getCPtr(value), value); + } + + public PolysemousTraining getPolysemous_training() { + long cPtr = swigfaissJNI.IndexIVFPQ_polysemous_training_get(swigCPtr, this); + return (cPtr == 0) ? null : new PolysemousTraining(cPtr, false); + } + + public void setScan_table_threshold(long value) { + swigfaissJNI.IndexIVFPQ_scan_table_threshold_set(swigCPtr, this, value); + } + + public long getScan_table_threshold() { + return swigfaissJNI.IndexIVFPQ_scan_table_threshold_get(swigCPtr, this); + } + + public void setPolysemous_ht(int value) { + swigfaissJNI.IndexIVFPQ_polysemous_ht_set(swigCPtr, this, value); + } + + public int getPolysemous_ht() { + return swigfaissJNI.IndexIVFPQ_polysemous_ht_get(swigCPtr, this); + } + + public void setUse_precomputed_table(int value) { + swigfaissJNI.IndexIVFPQ_use_precomputed_table_set(swigCPtr, this, value); + } + + public int getUse_precomputed_table() { + return swigfaissJNI.IndexIVFPQ_use_precomputed_table_get(swigCPtr, this); + } + + public void setPrecomputed_table(SWIGTYPE_p_AlignedTableT_float_t value) { + swigfaissJNI.IndexIVFPQ_precomputed_table_set(swigCPtr, this, SWIGTYPE_p_AlignedTableT_float_t.getCPtr(value)); + } + + public SWIGTYPE_p_AlignedTableT_float_t getPrecomputed_table() { + return new SWIGTYPE_p_AlignedTableT_float_t(swigfaissJNI.IndexIVFPQ_precomputed_table_get(swigCPtr, this), true); + } + + public IndexIVFPQ(Index quantizer, long d, long nlist, long M, long nbits_per_idx, MetricType metric) { + this(swigfaissJNI.new_IndexIVFPQ__SWIG_0(Index.getCPtr(quantizer), quantizer, d, nlist, M, nbits_per_idx, metric.swigValue()), true); + } + + public IndexIVFPQ(Index quantizer, long d, long nlist, long M, long nbits_per_idx) { + this(swigfaissJNI.new_IndexIVFPQ__SWIG_1(Index.getCPtr(quantizer), quantizer, d, nlist, M, nbits_per_idx), true); + } + + public void encode_vectors(long n, SWIGTYPE_p_float x, LongVector list_nos, SWIGTYPE_p_unsigned_char codes, boolean include_listnos) { + swigfaissJNI.IndexIVFPQ_encode_vectors__SWIG_0(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x), SWIGTYPE_p_long_long.getCPtr(list_nos.data()), list_nos, SWIGTYPE_p_unsigned_char.getCPtr(codes), include_listnos); + } + + public void encode_vectors(long n, SWIGTYPE_p_float x, LongVector list_nos, SWIGTYPE_p_unsigned_char codes) { + swigfaissJNI.IndexIVFPQ_encode_vectors__SWIG_1(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x), SWIGTYPE_p_long_long.getCPtr(list_nos.data()), list_nos, SWIGTYPE_p_unsigned_char.getCPtr(codes)); + } + + public void sa_decode(long n, SWIGTYPE_p_unsigned_char bytes, SWIGTYPE_p_float x) { + swigfaissJNI.IndexIVFPQ_sa_decode(swigCPtr, this, n, SWIGTYPE_p_unsigned_char.getCPtr(bytes), SWIGTYPE_p_float.getCPtr(x)); + } + + public void add_core(long n, SWIGTYPE_p_float x, LongVector xids, LongVector precomputed_idx) { + swigfaissJNI.IndexIVFPQ_add_core(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x), SWIGTYPE_p_long_long.getCPtr(xids.data()), xids, SWIGTYPE_p_long_long.getCPtr(precomputed_idx.data()), precomputed_idx); + } + + public void add_core_o(long n, SWIGTYPE_p_float x, LongVector xids, SWIGTYPE_p_float residuals_2, LongVector precomputed_idx) { + swigfaissJNI.IndexIVFPQ_add_core_o__SWIG_0(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x), SWIGTYPE_p_long_long.getCPtr(xids.data()), xids, SWIGTYPE_p_float.getCPtr(residuals_2), SWIGTYPE_p_long_long.getCPtr(precomputed_idx.data()), precomputed_idx); + } + + public void add_core_o(long n, SWIGTYPE_p_float x, LongVector xids, SWIGTYPE_p_float residuals_2) { + swigfaissJNI.IndexIVFPQ_add_core_o__SWIG_1(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x), SWIGTYPE_p_long_long.getCPtr(xids.data()), xids, SWIGTYPE_p_float.getCPtr(residuals_2)); + } + + public void train_residual(long n, SWIGTYPE_p_float x) { + swigfaissJNI.IndexIVFPQ_train_residual(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x)); + } + + public void train_residual_o(long n, SWIGTYPE_p_float x, SWIGTYPE_p_float residuals_2) { + swigfaissJNI.IndexIVFPQ_train_residual_o(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x), SWIGTYPE_p_float.getCPtr(residuals_2)); + } + + public void reconstruct_from_offset(long list_no, long offset, SWIGTYPE_p_float recons) { + swigfaissJNI.IndexIVFPQ_reconstruct_from_offset(swigCPtr, this, list_no, offset, SWIGTYPE_p_float.getCPtr(recons)); + } + + public long find_duplicates(LongVector ids, SWIGTYPE_p_unsigned_long lims) { + return swigfaissJNI.IndexIVFPQ_find_duplicates(swigCPtr, this, SWIGTYPE_p_long_long.getCPtr(ids.data()), ids, SWIGTYPE_p_unsigned_long.getCPtr(lims)); + } + + public void encode(long key, SWIGTYPE_p_float x, SWIGTYPE_p_unsigned_char code) { + swigfaissJNI.IndexIVFPQ_encode(swigCPtr, this, key, SWIGTYPE_p_float.getCPtr(x), SWIGTYPE_p_unsigned_char.getCPtr(code)); + } + + public void encode_multiple(long n, LongVector keys, SWIGTYPE_p_float x, SWIGTYPE_p_unsigned_char codes, boolean compute_keys) { + swigfaissJNI.IndexIVFPQ_encode_multiple__SWIG_0(swigCPtr, this, n, SWIGTYPE_p_long_long.getCPtr(keys.data()), keys, SWIGTYPE_p_float.getCPtr(x), SWIGTYPE_p_unsigned_char.getCPtr(codes), compute_keys); + } + + public void encode_multiple(long n, LongVector keys, SWIGTYPE_p_float x, SWIGTYPE_p_unsigned_char codes) { + swigfaissJNI.IndexIVFPQ_encode_multiple__SWIG_1(swigCPtr, this, n, SWIGTYPE_p_long_long.getCPtr(keys.data()), keys, SWIGTYPE_p_float.getCPtr(x), SWIGTYPE_p_unsigned_char.getCPtr(codes)); + } + + public void decode_multiple(long n, LongVector keys, SWIGTYPE_p_unsigned_char xcodes, SWIGTYPE_p_float x) { + swigfaissJNI.IndexIVFPQ_decode_multiple(swigCPtr, this, n, SWIGTYPE_p_long_long.getCPtr(keys.data()), keys, SWIGTYPE_p_unsigned_char.getCPtr(xcodes), SWIGTYPE_p_float.getCPtr(x)); + } + + public SWIGTYPE_p_faiss__InvertedListScanner get_InvertedListScanner(boolean store_pairs) { + long cPtr = swigfaissJNI.IndexIVFPQ_get_InvertedListScanner(swigCPtr, this, store_pairs); + return (cPtr == 0) ? null : new SWIGTYPE_p_faiss__InvertedListScanner(cPtr, false); + } + + public void precompute_table() { + swigfaissJNI.IndexIVFPQ_precompute_table(swigCPtr, this); + } + + public IndexIVFPQ() { + this(swigfaissJNI.new_IndexIVFPQ__SWIG_2(), true); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/IndexIVFPQStats.java b/ann/src/main/java/com/twitter/ann/faiss/swig/IndexIVFPQStats.java new file mode 100644 index 0000000000..81c8293633 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/IndexIVFPQStats.java @@ -0,0 +1,79 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class IndexIVFPQStats { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected IndexIVFPQStats(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(IndexIVFPQStats obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_IndexIVFPQStats(swigCPtr); + } + swigCPtr = 0; + } + } + + public void setNrefine(long value) { + swigfaissJNI.IndexIVFPQStats_nrefine_set(swigCPtr, this, value); + } + + public long getNrefine() { + return swigfaissJNI.IndexIVFPQStats_nrefine_get(swigCPtr, this); + } + + public void setN_hamming_pass(long value) { + swigfaissJNI.IndexIVFPQStats_n_hamming_pass_set(swigCPtr, this, value); + } + + public long getN_hamming_pass() { + return swigfaissJNI.IndexIVFPQStats_n_hamming_pass_get(swigCPtr, this); + } + + public void setSearch_cycles(long value) { + swigfaissJNI.IndexIVFPQStats_search_cycles_set(swigCPtr, this, value); + } + + public long getSearch_cycles() { + return swigfaissJNI.IndexIVFPQStats_search_cycles_get(swigCPtr, this); + } + + public void setRefine_cycles(long value) { + swigfaissJNI.IndexIVFPQStats_refine_cycles_set(swigCPtr, this, value); + } + + public long getRefine_cycles() { + return swigfaissJNI.IndexIVFPQStats_refine_cycles_get(swigCPtr, this); + } + + public IndexIVFPQStats() { + this(swigfaissJNI.new_IndexIVFPQStats(), true); + } + + public void reset() { + swigfaissJNI.IndexIVFPQStats_reset(swigCPtr, this); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/IndexIVFScalarQuantizer.java b/ann/src/main/java/com/twitter/ann/faiss/swig/IndexIVFScalarQuantizer.java new file mode 100644 index 0000000000..8e72059dec --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/IndexIVFScalarQuantizer.java @@ -0,0 +1,100 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class IndexIVFScalarQuantizer extends IndexIVF { + private transient long swigCPtr; + + protected IndexIVFScalarQuantizer(long cPtr, boolean cMemoryOwn) { + super(swigfaissJNI.IndexIVFScalarQuantizer_SWIGUpcast(cPtr), cMemoryOwn); + swigCPtr = cPtr; + } + + protected static long getCPtr(IndexIVFScalarQuantizer obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_IndexIVFScalarQuantizer(swigCPtr); + } + swigCPtr = 0; + } + super.delete(); + } + + public void setSq(SWIGTYPE_p_ScalarQuantizer value) { + swigfaissJNI.IndexIVFScalarQuantizer_sq_set(swigCPtr, this, SWIGTYPE_p_ScalarQuantizer.getCPtr(value)); + } + + public SWIGTYPE_p_ScalarQuantizer getSq() { + return new SWIGTYPE_p_ScalarQuantizer(swigfaissJNI.IndexIVFScalarQuantizer_sq_get(swigCPtr, this), true); + } + + public void setBy_residual(boolean value) { + swigfaissJNI.IndexIVFScalarQuantizer_by_residual_set(swigCPtr, this, value); + } + + public boolean getBy_residual() { + return swigfaissJNI.IndexIVFScalarQuantizer_by_residual_get(swigCPtr, this); + } + + public IndexIVFScalarQuantizer(Index quantizer, long d, long nlist, SWIGTYPE_p_ScalarQuantizer__QuantizerType qtype, MetricType metric, boolean encode_residual) { + this(swigfaissJNI.new_IndexIVFScalarQuantizer__SWIG_0(Index.getCPtr(quantizer), quantizer, d, nlist, SWIGTYPE_p_ScalarQuantizer__QuantizerType.getCPtr(qtype), metric.swigValue(), encode_residual), true); + } + + public IndexIVFScalarQuantizer(Index quantizer, long d, long nlist, SWIGTYPE_p_ScalarQuantizer__QuantizerType qtype, MetricType metric) { + this(swigfaissJNI.new_IndexIVFScalarQuantizer__SWIG_1(Index.getCPtr(quantizer), quantizer, d, nlist, SWIGTYPE_p_ScalarQuantizer__QuantizerType.getCPtr(qtype), metric.swigValue()), true); + } + + public IndexIVFScalarQuantizer(Index quantizer, long d, long nlist, SWIGTYPE_p_ScalarQuantizer__QuantizerType qtype) { + this(swigfaissJNI.new_IndexIVFScalarQuantizer__SWIG_2(Index.getCPtr(quantizer), quantizer, d, nlist, SWIGTYPE_p_ScalarQuantizer__QuantizerType.getCPtr(qtype)), true); + } + + public IndexIVFScalarQuantizer() { + this(swigfaissJNI.new_IndexIVFScalarQuantizer__SWIG_3(), true); + } + + public void train_residual(long n, SWIGTYPE_p_float x) { + swigfaissJNI.IndexIVFScalarQuantizer_train_residual(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x)); + } + + public void encode_vectors(long n, SWIGTYPE_p_float x, LongVector list_nos, SWIGTYPE_p_unsigned_char codes, boolean include_listnos) { + swigfaissJNI.IndexIVFScalarQuantizer_encode_vectors__SWIG_0(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x), SWIGTYPE_p_long_long.getCPtr(list_nos.data()), list_nos, SWIGTYPE_p_unsigned_char.getCPtr(codes), include_listnos); + } + + public void encode_vectors(long n, SWIGTYPE_p_float x, LongVector list_nos, SWIGTYPE_p_unsigned_char codes) { + swigfaissJNI.IndexIVFScalarQuantizer_encode_vectors__SWIG_1(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x), SWIGTYPE_p_long_long.getCPtr(list_nos.data()), list_nos, SWIGTYPE_p_unsigned_char.getCPtr(codes)); + } + + public void add_core(long n, SWIGTYPE_p_float x, LongVector xids, LongVector precomputed_idx) { + swigfaissJNI.IndexIVFScalarQuantizer_add_core(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x), SWIGTYPE_p_long_long.getCPtr(xids.data()), xids, SWIGTYPE_p_long_long.getCPtr(precomputed_idx.data()), precomputed_idx); + } + + public SWIGTYPE_p_faiss__InvertedListScanner get_InvertedListScanner(boolean store_pairs) { + long cPtr = swigfaissJNI.IndexIVFScalarQuantizer_get_InvertedListScanner(swigCPtr, this, store_pairs); + return (cPtr == 0) ? null : new SWIGTYPE_p_faiss__InvertedListScanner(cPtr, false); + } + + public void reconstruct_from_offset(long list_no, long offset, SWIGTYPE_p_float recons) { + swigfaissJNI.IndexIVFScalarQuantizer_reconstruct_from_offset(swigCPtr, this, list_no, offset, SWIGTYPE_p_float.getCPtr(recons)); + } + + public void sa_decode(long n, SWIGTYPE_p_unsigned_char bytes, SWIGTYPE_p_float x) { + swigfaissJNI.IndexIVFScalarQuantizer_sa_decode(swigCPtr, this, n, SWIGTYPE_p_unsigned_char.getCPtr(bytes), SWIGTYPE_p_float.getCPtr(x)); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/IndexIVFStats.java b/ann/src/main/java/com/twitter/ann/faiss/swig/IndexIVFStats.java new file mode 100644 index 0000000000..b70bb07aa3 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/IndexIVFStats.java @@ -0,0 +1,99 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class IndexIVFStats { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected IndexIVFStats(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(IndexIVFStats obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_IndexIVFStats(swigCPtr); + } + swigCPtr = 0; + } + } + + public void setNq(long value) { + swigfaissJNI.IndexIVFStats_nq_set(swigCPtr, this, value); + } + + public long getNq() { + return swigfaissJNI.IndexIVFStats_nq_get(swigCPtr, this); + } + + public void setNlist(long value) { + swigfaissJNI.IndexIVFStats_nlist_set(swigCPtr, this, value); + } + + public long getNlist() { + return swigfaissJNI.IndexIVFStats_nlist_get(swigCPtr, this); + } + + public void setNdis(long value) { + swigfaissJNI.IndexIVFStats_ndis_set(swigCPtr, this, value); + } + + public long getNdis() { + return swigfaissJNI.IndexIVFStats_ndis_get(swigCPtr, this); + } + + public void setNheap_updates(long value) { + swigfaissJNI.IndexIVFStats_nheap_updates_set(swigCPtr, this, value); + } + + public long getNheap_updates() { + return swigfaissJNI.IndexIVFStats_nheap_updates_get(swigCPtr, this); + } + + public void setQuantization_time(double value) { + swigfaissJNI.IndexIVFStats_quantization_time_set(swigCPtr, this, value); + } + + public double getQuantization_time() { + return swigfaissJNI.IndexIVFStats_quantization_time_get(swigCPtr, this); + } + + public void setSearch_time(double value) { + swigfaissJNI.IndexIVFStats_search_time_set(swigCPtr, this, value); + } + + public double getSearch_time() { + return swigfaissJNI.IndexIVFStats_search_time_get(swigCPtr, this); + } + + public IndexIVFStats() { + this(swigfaissJNI.new_IndexIVFStats(), true); + } + + public void reset() { + swigfaissJNI.IndexIVFStats_reset(swigCPtr, this); + } + + public void add(IndexIVFStats other) { + swigfaissJNI.IndexIVFStats_add(swigCPtr, this, IndexIVFStats.getCPtr(other), other); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/IndexLSH.java b/ann/src/main/java/com/twitter/ann/faiss/swig/IndexLSH.java new file mode 100644 index 0000000000..77c1cb855a --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/IndexLSH.java @@ -0,0 +1,122 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class IndexLSH extends IndexFlatCodes { + private transient long swigCPtr; + + protected IndexLSH(long cPtr, boolean cMemoryOwn) { + super(swigfaissJNI.IndexLSH_SWIGUpcast(cPtr), cMemoryOwn); + swigCPtr = cPtr; + } + + protected static long getCPtr(IndexLSH obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_IndexLSH(swigCPtr); + } + swigCPtr = 0; + } + super.delete(); + } + + public void setNbits(int value) { + swigfaissJNI.IndexLSH_nbits_set(swigCPtr, this, value); + } + + public int getNbits() { + return swigfaissJNI.IndexLSH_nbits_get(swigCPtr, this); + } + + public void setRotate_data(boolean value) { + swigfaissJNI.IndexLSH_rotate_data_set(swigCPtr, this, value); + } + + public boolean getRotate_data() { + return swigfaissJNI.IndexLSH_rotate_data_get(swigCPtr, this); + } + + public void setTrain_thresholds(boolean value) { + swigfaissJNI.IndexLSH_train_thresholds_set(swigCPtr, this, value); + } + + public boolean getTrain_thresholds() { + return swigfaissJNI.IndexLSH_train_thresholds_get(swigCPtr, this); + } + + public void setRrot(RandomRotationMatrix value) { + swigfaissJNI.IndexLSH_rrot_set(swigCPtr, this, RandomRotationMatrix.getCPtr(value), value); + } + + public RandomRotationMatrix getRrot() { + long cPtr = swigfaissJNI.IndexLSH_rrot_get(swigCPtr, this); + return (cPtr == 0) ? null : new RandomRotationMatrix(cPtr, false); + } + + public void setThresholds(FloatVector value) { + swigfaissJNI.IndexLSH_thresholds_set(swigCPtr, this, FloatVector.getCPtr(value), value); + } + + public FloatVector getThresholds() { + long cPtr = swigfaissJNI.IndexLSH_thresholds_get(swigCPtr, this); + return (cPtr == 0) ? null : new FloatVector(cPtr, false); + } + + public IndexLSH(long d, int nbits, boolean rotate_data, boolean train_thresholds) { + this(swigfaissJNI.new_IndexLSH__SWIG_0(d, nbits, rotate_data, train_thresholds), true); + } + + public IndexLSH(long d, int nbits, boolean rotate_data) { + this(swigfaissJNI.new_IndexLSH__SWIG_1(d, nbits, rotate_data), true); + } + + public IndexLSH(long d, int nbits) { + this(swigfaissJNI.new_IndexLSH__SWIG_2(d, nbits), true); + } + + public SWIGTYPE_p_float apply_preprocess(long n, SWIGTYPE_p_float x) { + long cPtr = swigfaissJNI.IndexLSH_apply_preprocess(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x)); + return (cPtr == 0) ? null : new SWIGTYPE_p_float(cPtr, false); + } + + public void train(long n, SWIGTYPE_p_float x) { + swigfaissJNI.IndexLSH_train(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x)); + } + + public void search(long n, SWIGTYPE_p_float x, long k, SWIGTYPE_p_float distances, LongVector labels) { + swigfaissJNI.IndexLSH_search(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x), k, SWIGTYPE_p_float.getCPtr(distances), SWIGTYPE_p_long_long.getCPtr(labels.data()), labels); + } + + public void transfer_thresholds(LinearTransform vt) { + swigfaissJNI.IndexLSH_transfer_thresholds(swigCPtr, this, LinearTransform.getCPtr(vt), vt); + } + + public IndexLSH() { + this(swigfaissJNI.new_IndexLSH__SWIG_3(), true); + } + + public void sa_encode(long n, SWIGTYPE_p_float x, SWIGTYPE_p_unsigned_char bytes) { + swigfaissJNI.IndexLSH_sa_encode(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x), SWIGTYPE_p_unsigned_char.getCPtr(bytes)); + } + + public void sa_decode(long n, SWIGTYPE_p_unsigned_char bytes, SWIGTYPE_p_float x) { + swigfaissJNI.IndexLSH_sa_decode(swigCPtr, this, n, SWIGTYPE_p_unsigned_char.getCPtr(bytes), SWIGTYPE_p_float.getCPtr(x)); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/IndexPQ.java b/ann/src/main/java/com/twitter/ann/faiss/swig/IndexPQ.java new file mode 100644 index 0000000000..b0e874cbc0 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/IndexPQ.java @@ -0,0 +1,182 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class IndexPQ extends IndexFlatCodes { + private transient long swigCPtr; + + protected IndexPQ(long cPtr, boolean cMemoryOwn) { + super(swigfaissJNI.IndexPQ_SWIGUpcast(cPtr), cMemoryOwn); + swigCPtr = cPtr; + } + + protected static long getCPtr(IndexPQ obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_IndexPQ(swigCPtr); + } + swigCPtr = 0; + } + super.delete(); + } + + public void setPq(ProductQuantizer value) { + swigfaissJNI.IndexPQ_pq_set(swigCPtr, this, ProductQuantizer.getCPtr(value), value); + } + + public ProductQuantizer getPq() { + long cPtr = swigfaissJNI.IndexPQ_pq_get(swigCPtr, this); + return (cPtr == 0) ? null : new ProductQuantizer(cPtr, false); + } + + public IndexPQ(int d, long M, long nbits, MetricType metric) { + this(swigfaissJNI.new_IndexPQ__SWIG_0(d, M, nbits, metric.swigValue()), true); + } + + public IndexPQ(int d, long M, long nbits) { + this(swigfaissJNI.new_IndexPQ__SWIG_1(d, M, nbits), true); + } + + public IndexPQ() { + this(swigfaissJNI.new_IndexPQ__SWIG_2(), true); + } + + public void train(long n, SWIGTYPE_p_float x) { + swigfaissJNI.IndexPQ_train(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x)); + } + + public void search(long n, SWIGTYPE_p_float x, long k, SWIGTYPE_p_float distances, LongVector labels) { + swigfaissJNI.IndexPQ_search(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x), k, SWIGTYPE_p_float.getCPtr(distances), SWIGTYPE_p_long_long.getCPtr(labels.data()), labels); + } + + public void sa_encode(long n, SWIGTYPE_p_float x, SWIGTYPE_p_unsigned_char bytes) { + swigfaissJNI.IndexPQ_sa_encode(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x), SWIGTYPE_p_unsigned_char.getCPtr(bytes)); + } + + public void sa_decode(long n, SWIGTYPE_p_unsigned_char bytes, SWIGTYPE_p_float x) { + swigfaissJNI.IndexPQ_sa_decode(swigCPtr, this, n, SWIGTYPE_p_unsigned_char.getCPtr(bytes), SWIGTYPE_p_float.getCPtr(x)); + } + + public DistanceComputer get_distance_computer() { + long cPtr = swigfaissJNI.IndexPQ_get_distance_computer(swigCPtr, this); + return (cPtr == 0) ? null : new DistanceComputer(cPtr, false); + } + + public void setDo_polysemous_training(boolean value) { + swigfaissJNI.IndexPQ_do_polysemous_training_set(swigCPtr, this, value); + } + + public boolean getDo_polysemous_training() { + return swigfaissJNI.IndexPQ_do_polysemous_training_get(swigCPtr, this); + } + + public void setPolysemous_training(PolysemousTraining value) { + swigfaissJNI.IndexPQ_polysemous_training_set(swigCPtr, this, PolysemousTraining.getCPtr(value), value); + } + + public PolysemousTraining getPolysemous_training() { + long cPtr = swigfaissJNI.IndexPQ_polysemous_training_get(swigCPtr, this); + return (cPtr == 0) ? null : new PolysemousTraining(cPtr, false); + } + + public void setSearch_type(IndexPQ.Search_type_t value) { + swigfaissJNI.IndexPQ_search_type_set(swigCPtr, this, value.swigValue()); + } + + public IndexPQ.Search_type_t getSearch_type() { + return IndexPQ.Search_type_t.swigToEnum(swigfaissJNI.IndexPQ_search_type_get(swigCPtr, this)); + } + + public void setEncode_signs(boolean value) { + swigfaissJNI.IndexPQ_encode_signs_set(swigCPtr, this, value); + } + + public boolean getEncode_signs() { + return swigfaissJNI.IndexPQ_encode_signs_get(swigCPtr, this); + } + + public void setPolysemous_ht(int value) { + swigfaissJNI.IndexPQ_polysemous_ht_set(swigCPtr, this, value); + } + + public int getPolysemous_ht() { + return swigfaissJNI.IndexPQ_polysemous_ht_get(swigCPtr, this); + } + + public void search_core_polysemous(long n, SWIGTYPE_p_float x, long k, SWIGTYPE_p_float distances, LongVector labels) { + swigfaissJNI.IndexPQ_search_core_polysemous(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x), k, SWIGTYPE_p_float.getCPtr(distances), SWIGTYPE_p_long_long.getCPtr(labels.data()), labels); + } + + public void hamming_distance_histogram(long n, SWIGTYPE_p_float x, long nb, SWIGTYPE_p_float xb, LongVector dist_histogram) { + swigfaissJNI.IndexPQ_hamming_distance_histogram(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x), nb, SWIGTYPE_p_float.getCPtr(xb), SWIGTYPE_p_long_long.getCPtr(dist_histogram.data()), dist_histogram); + } + + public void hamming_distance_table(long n, SWIGTYPE_p_float x, SWIGTYPE_p_int dis) { + swigfaissJNI.IndexPQ_hamming_distance_table(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x), SWIGTYPE_p_int.getCPtr(dis)); + } + + public final static class Search_type_t { + public final static IndexPQ.Search_type_t ST_PQ = new IndexPQ.Search_type_t("ST_PQ"); + public final static IndexPQ.Search_type_t ST_HE = new IndexPQ.Search_type_t("ST_HE"); + public final static IndexPQ.Search_type_t ST_generalized_HE = new IndexPQ.Search_type_t("ST_generalized_HE"); + public final static IndexPQ.Search_type_t ST_SDC = new IndexPQ.Search_type_t("ST_SDC"); + public final static IndexPQ.Search_type_t ST_polysemous = new IndexPQ.Search_type_t("ST_polysemous"); + public final static IndexPQ.Search_type_t ST_polysemous_generalize = new IndexPQ.Search_type_t("ST_polysemous_generalize"); + + public final int swigValue() { + return swigValue; + } + + public String toString() { + return swigName; + } + + public static Search_type_t swigToEnum(int swigValue) { + if (swigValue < swigValues.length && swigValue >= 0 && swigValues[swigValue].swigValue == swigValue) + return swigValues[swigValue]; + for (int i = 0; i < swigValues.length; i++) + if (swigValues[i].swigValue == swigValue) + return swigValues[i]; + throw new IllegalArgumentException("No enum " + Search_type_t.class + " with value " + swigValue); + } + + private Search_type_t(String swigName) { + this.swigName = swigName; + this.swigValue = swigNext++; + } + + private Search_type_t(String swigName, int swigValue) { + this.swigName = swigName; + this.swigValue = swigValue; + swigNext = swigValue+1; + } + + private Search_type_t(String swigName, Search_type_t swigEnum) { + this.swigName = swigName; + this.swigValue = swigEnum.swigValue; + swigNext = this.swigValue+1; + } + + private static Search_type_t[] swigValues = { ST_PQ, ST_HE, ST_generalized_HE, ST_SDC, ST_polysemous, ST_polysemous_generalize }; + private static int swigNext = 0; + private final int swigValue; + private final String swigName; + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/IndexPQStats.java b/ann/src/main/java/com/twitter/ann/faiss/swig/IndexPQStats.java new file mode 100644 index 0000000000..c5e0b9d2ba --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/IndexPQStats.java @@ -0,0 +1,71 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class IndexPQStats { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected IndexPQStats(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(IndexPQStats obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_IndexPQStats(swigCPtr); + } + swigCPtr = 0; + } + } + + public void setNq(long value) { + swigfaissJNI.IndexPQStats_nq_set(swigCPtr, this, value); + } + + public long getNq() { + return swigfaissJNI.IndexPQStats_nq_get(swigCPtr, this); + } + + public void setNcode(long value) { + swigfaissJNI.IndexPQStats_ncode_set(swigCPtr, this, value); + } + + public long getNcode() { + return swigfaissJNI.IndexPQStats_ncode_get(swigCPtr, this); + } + + public void setN_hamming_pass(long value) { + swigfaissJNI.IndexPQStats_n_hamming_pass_set(swigCPtr, this, value); + } + + public long getN_hamming_pass() { + return swigfaissJNI.IndexPQStats_n_hamming_pass_get(swigCPtr, this); + } + + public IndexPQStats() { + this(swigfaissJNI.new_IndexPQStats(), true); + } + + public void reset() { + swigfaissJNI.IndexPQStats_reset(swigCPtr, this); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/IndexRefine.java b/ann/src/main/java/com/twitter/ann/faiss/swig/IndexRefine.java new file mode 100644 index 0000000000..f0e1269d8f --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/IndexRefine.java @@ -0,0 +1,121 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class IndexRefine extends Index { + private transient long swigCPtr; + + protected IndexRefine(long cPtr, boolean cMemoryOwn) { + super(swigfaissJNI.IndexRefine_SWIGUpcast(cPtr), cMemoryOwn); + swigCPtr = cPtr; + } + + protected static long getCPtr(IndexRefine obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_IndexRefine(swigCPtr); + } + swigCPtr = 0; + } + super.delete(); + } + + public void setBase_index(Index value) { + swigfaissJNI.IndexRefine_base_index_set(swigCPtr, this, Index.getCPtr(value), value); + } + + public Index getBase_index() { + long cPtr = swigfaissJNI.IndexRefine_base_index_get(swigCPtr, this); + return (cPtr == 0) ? null : new Index(cPtr, false); + } + + public void setRefine_index(Index value) { + swigfaissJNI.IndexRefine_refine_index_set(swigCPtr, this, Index.getCPtr(value), value); + } + + public Index getRefine_index() { + long cPtr = swigfaissJNI.IndexRefine_refine_index_get(swigCPtr, this); + return (cPtr == 0) ? null : new Index(cPtr, false); + } + + public void setOwn_fields(boolean value) { + swigfaissJNI.IndexRefine_own_fields_set(swigCPtr, this, value); + } + + public boolean getOwn_fields() { + return swigfaissJNI.IndexRefine_own_fields_get(swigCPtr, this); + } + + public void setOwn_refine_index(boolean value) { + swigfaissJNI.IndexRefine_own_refine_index_set(swigCPtr, this, value); + } + + public boolean getOwn_refine_index() { + return swigfaissJNI.IndexRefine_own_refine_index_get(swigCPtr, this); + } + + public void setK_factor(float value) { + swigfaissJNI.IndexRefine_k_factor_set(swigCPtr, this, value); + } + + public float getK_factor() { + return swigfaissJNI.IndexRefine_k_factor_get(swigCPtr, this); + } + + public IndexRefine(Index base_index, Index refine_index) { + this(swigfaissJNI.new_IndexRefine__SWIG_0(Index.getCPtr(base_index), base_index, Index.getCPtr(refine_index), refine_index), true); + } + + public IndexRefine() { + this(swigfaissJNI.new_IndexRefine__SWIG_1(), true); + } + + public void train(long n, SWIGTYPE_p_float x) { + swigfaissJNI.IndexRefine_train(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x)); + } + + public void add(long n, SWIGTYPE_p_float x) { + swigfaissJNI.IndexRefine_add(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x)); + } + + public void reset() { + swigfaissJNI.IndexRefine_reset(swigCPtr, this); + } + + public void search(long n, SWIGTYPE_p_float x, long k, SWIGTYPE_p_float distances, LongVector labels) { + swigfaissJNI.IndexRefine_search(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x), k, SWIGTYPE_p_float.getCPtr(distances), SWIGTYPE_p_long_long.getCPtr(labels.data()), labels); + } + + public void reconstruct(long key, SWIGTYPE_p_float recons) { + swigfaissJNI.IndexRefine_reconstruct(swigCPtr, this, key, SWIGTYPE_p_float.getCPtr(recons)); + } + + public long sa_code_size() { + return swigfaissJNI.IndexRefine_sa_code_size(swigCPtr, this); + } + + public void sa_encode(long n, SWIGTYPE_p_float x, SWIGTYPE_p_unsigned_char bytes) { + swigfaissJNI.IndexRefine_sa_encode(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x), SWIGTYPE_p_unsigned_char.getCPtr(bytes)); + } + + public void sa_decode(long n, SWIGTYPE_p_unsigned_char bytes, SWIGTYPE_p_float x) { + swigfaissJNI.IndexRefine_sa_decode(swigCPtr, this, n, SWIGTYPE_p_unsigned_char.getCPtr(bytes), SWIGTYPE_p_float.getCPtr(x)); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/IndexRefineFlat.java b/ann/src/main/java/com/twitter/ann/faiss/swig/IndexRefineFlat.java new file mode 100644 index 0000000000..5b4b43b034 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/IndexRefineFlat.java @@ -0,0 +1,55 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class IndexRefineFlat extends IndexRefine { + private transient long swigCPtr; + + protected IndexRefineFlat(long cPtr, boolean cMemoryOwn) { + super(swigfaissJNI.IndexRefineFlat_SWIGUpcast(cPtr), cMemoryOwn); + swigCPtr = cPtr; + } + + protected static long getCPtr(IndexRefineFlat obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_IndexRefineFlat(swigCPtr); + } + swigCPtr = 0; + } + super.delete(); + } + + public IndexRefineFlat(Index base_index) { + this(swigfaissJNI.new_IndexRefineFlat__SWIG_0(Index.getCPtr(base_index), base_index), true); + } + + public IndexRefineFlat(Index base_index, SWIGTYPE_p_float xb) { + this(swigfaissJNI.new_IndexRefineFlat__SWIG_1(Index.getCPtr(base_index), base_index, SWIGTYPE_p_float.getCPtr(xb)), true); + } + + public IndexRefineFlat() { + this(swigfaissJNI.new_IndexRefineFlat__SWIG_2(), true); + } + + public void search(long n, SWIGTYPE_p_float x, long k, SWIGTYPE_p_float distances, LongVector labels) { + swigfaissJNI.IndexRefineFlat_search(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x), k, SWIGTYPE_p_float.getCPtr(distances), SWIGTYPE_p_long_long.getCPtr(labels.data()), labels); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/IndexScalarQuantizer.java b/ann/src/main/java/com/twitter/ann/faiss/swig/IndexScalarQuantizer.java new file mode 100644 index 0000000000..0d7e862e56 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/IndexScalarQuantizer.java @@ -0,0 +1,80 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class IndexScalarQuantizer extends IndexFlatCodes { + private transient long swigCPtr; + + protected IndexScalarQuantizer(long cPtr, boolean cMemoryOwn) { + super(swigfaissJNI.IndexScalarQuantizer_SWIGUpcast(cPtr), cMemoryOwn); + swigCPtr = cPtr; + } + + protected static long getCPtr(IndexScalarQuantizer obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_IndexScalarQuantizer(swigCPtr); + } + swigCPtr = 0; + } + super.delete(); + } + + public void setSq(SWIGTYPE_p_ScalarQuantizer value) { + swigfaissJNI.IndexScalarQuantizer_sq_set(swigCPtr, this, SWIGTYPE_p_ScalarQuantizer.getCPtr(value)); + } + + public SWIGTYPE_p_ScalarQuantizer getSq() { + return new SWIGTYPE_p_ScalarQuantizer(swigfaissJNI.IndexScalarQuantizer_sq_get(swigCPtr, this), true); + } + + public IndexScalarQuantizer(int d, SWIGTYPE_p_ScalarQuantizer__QuantizerType qtype, MetricType metric) { + this(swigfaissJNI.new_IndexScalarQuantizer__SWIG_0(d, SWIGTYPE_p_ScalarQuantizer__QuantizerType.getCPtr(qtype), metric.swigValue()), true); + } + + public IndexScalarQuantizer(int d, SWIGTYPE_p_ScalarQuantizer__QuantizerType qtype) { + this(swigfaissJNI.new_IndexScalarQuantizer__SWIG_1(d, SWIGTYPE_p_ScalarQuantizer__QuantizerType.getCPtr(qtype)), true); + } + + public IndexScalarQuantizer() { + this(swigfaissJNI.new_IndexScalarQuantizer__SWIG_2(), true); + } + + public void train(long n, SWIGTYPE_p_float x) { + swigfaissJNI.IndexScalarQuantizer_train(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x)); + } + + public void search(long n, SWIGTYPE_p_float x, long k, SWIGTYPE_p_float distances, LongVector labels) { + swigfaissJNI.IndexScalarQuantizer_search(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x), k, SWIGTYPE_p_float.getCPtr(distances), SWIGTYPE_p_long_long.getCPtr(labels.data()), labels); + } + + public DistanceComputer get_distance_computer() { + long cPtr = swigfaissJNI.IndexScalarQuantizer_get_distance_computer(swigCPtr, this); + return (cPtr == 0) ? null : new DistanceComputer(cPtr, false); + } + + public void sa_encode(long n, SWIGTYPE_p_float x, SWIGTYPE_p_unsigned_char bytes) { + swigfaissJNI.IndexScalarQuantizer_sa_encode(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x), SWIGTYPE_p_unsigned_char.getCPtr(bytes)); + } + + public void sa_decode(long n, SWIGTYPE_p_unsigned_char bytes, SWIGTYPE_p_float x) { + swigfaissJNI.IndexScalarQuantizer_sa_decode(swigCPtr, this, n, SWIGTYPE_p_unsigned_char.getCPtr(bytes), SWIGTYPE_p_float.getCPtr(x)); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/IndexShards.java b/ann/src/main/java/com/twitter/ann/faiss/swig/IndexShards.java new file mode 100644 index 0000000000..a86d128f99 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/IndexShards.java @@ -0,0 +1,99 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class IndexShards { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected IndexShards(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(IndexShards obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_IndexShards(swigCPtr); + } + swigCPtr = 0; + } + } + + public IndexShards(boolean threaded, boolean successive_ids) { + this(swigfaissJNI.new_IndexShards__SWIG_0(threaded, successive_ids), true); + } + + public IndexShards(boolean threaded) { + this(swigfaissJNI.new_IndexShards__SWIG_1(threaded), true); + } + + public IndexShards() { + this(swigfaissJNI.new_IndexShards__SWIG_2(), true); + } + + public IndexShards(int d, boolean threaded, boolean successive_ids) { + this(swigfaissJNI.new_IndexShards__SWIG_3(d, threaded, successive_ids), true); + } + + public IndexShards(int d, boolean threaded) { + this(swigfaissJNI.new_IndexShards__SWIG_4(d, threaded), true); + } + + public IndexShards(int d) { + this(swigfaissJNI.new_IndexShards__SWIG_5(d), true); + } + + public void add_shard(Index index) { + swigfaissJNI.IndexShards_add_shard(swigCPtr, this, Index.getCPtr(index), index); + } + + public void remove_shard(Index index) { + swigfaissJNI.IndexShards_remove_shard(swigCPtr, this, Index.getCPtr(index), index); + } + + public void add(long n, SWIGTYPE_p_float x) { + swigfaissJNI.IndexShards_add(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x)); + } + + public void add_with_ids(long n, SWIGTYPE_p_float x, LongVector xids) { + swigfaissJNI.IndexShards_add_with_ids(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x), SWIGTYPE_p_long_long.getCPtr(xids.data()), xids); + } + + public void search(long n, SWIGTYPE_p_float x, long k, SWIGTYPE_p_float distances, LongVector labels) { + swigfaissJNI.IndexShards_search(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x), k, SWIGTYPE_p_float.getCPtr(distances), SWIGTYPE_p_long_long.getCPtr(labels.data()), labels); + } + + public void train(long n, SWIGTYPE_p_float x) { + swigfaissJNI.IndexShards_train(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x)); + } + + public void setSuccessive_ids(boolean value) { + swigfaissJNI.IndexShards_successive_ids_set(swigCPtr, this, value); + } + + public boolean getSuccessive_ids() { + return swigfaissJNI.IndexShards_successive_ids_get(swigCPtr, this); + } + + public void syncWithSubIndexes() { + swigfaissJNI.IndexShards_syncWithSubIndexes(swigCPtr, this); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/IndexSplitVectors.java b/ann/src/main/java/com/twitter/ann/faiss/swig/IndexSplitVectors.java new file mode 100644 index 0000000000..701d899196 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/IndexSplitVectors.java @@ -0,0 +1,96 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class IndexSplitVectors extends Index { + private transient long swigCPtr; + + protected IndexSplitVectors(long cPtr, boolean cMemoryOwn) { + super(swigfaissJNI.IndexSplitVectors_SWIGUpcast(cPtr), cMemoryOwn); + swigCPtr = cPtr; + } + + protected static long getCPtr(IndexSplitVectors obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_IndexSplitVectors(swigCPtr); + } + swigCPtr = 0; + } + super.delete(); + } + + public void setOwn_fields(boolean value) { + swigfaissJNI.IndexSplitVectors_own_fields_set(swigCPtr, this, value); + } + + public boolean getOwn_fields() { + return swigfaissJNI.IndexSplitVectors_own_fields_get(swigCPtr, this); + } + + public void setThreaded(boolean value) { + swigfaissJNI.IndexSplitVectors_threaded_set(swigCPtr, this, value); + } + + public boolean getThreaded() { + return swigfaissJNI.IndexSplitVectors_threaded_get(swigCPtr, this); + } + + public void setSub_indexes(SWIGTYPE_p_std__vectorT_faiss__Index_p_t value) { + swigfaissJNI.IndexSplitVectors_sub_indexes_set(swigCPtr, this, SWIGTYPE_p_std__vectorT_faiss__Index_p_t.getCPtr(value)); + } + + public SWIGTYPE_p_std__vectorT_faiss__Index_p_t getSub_indexes() { + long cPtr = swigfaissJNI.IndexSplitVectors_sub_indexes_get(swigCPtr, this); + return (cPtr == 0) ? null : new SWIGTYPE_p_std__vectorT_faiss__Index_p_t(cPtr, false); + } + + public void setSum_d(long value) { + swigfaissJNI.IndexSplitVectors_sum_d_set(swigCPtr, this, value); + } + + public long getSum_d() { + return swigfaissJNI.IndexSplitVectors_sum_d_get(swigCPtr, this); +} + + public void add_sub_index(Index arg0) { + swigfaissJNI.IndexSplitVectors_add_sub_index(swigCPtr, this, Index.getCPtr(arg0), arg0); + } + + public void sync_with_sub_indexes() { + swigfaissJNI.IndexSplitVectors_sync_with_sub_indexes(swigCPtr, this); + } + + public void add(long n, SWIGTYPE_p_float x) { + swigfaissJNI.IndexSplitVectors_add(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x)); + } + + public void search(long n, SWIGTYPE_p_float x, long k, SWIGTYPE_p_float distances, LongVector labels) { + swigfaissJNI.IndexSplitVectors_search(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x), k, SWIGTYPE_p_float.getCPtr(distances), SWIGTYPE_p_long_long.getCPtr(labels.data()), labels); + } + + public void train(long n, SWIGTYPE_p_float x) { + swigfaissJNI.IndexSplitVectors_train(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x)); + } + + public void reset() { + swigfaissJNI.IndexSplitVectors_reset(swigCPtr, this); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/IntVector.java b/ann/src/main/java/com/twitter/ann/faiss/swig/IntVector.java new file mode 100644 index 0000000000..712ea37182 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/IntVector.java @@ -0,0 +1,76 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class IntVector { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected IntVector(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(IntVector obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_IntVector(swigCPtr); + } + swigCPtr = 0; + } + } + + public IntVector() { + this(swigfaissJNI.new_IntVector(), true); + } + + public void push_back(int arg0) { + swigfaissJNI.IntVector_push_back(swigCPtr, this, arg0); + } + + public void clear() { + swigfaissJNI.IntVector_clear(swigCPtr, this); + } + + public SWIGTYPE_p_int data() { + long cPtr = swigfaissJNI.IntVector_data(swigCPtr, this); + return (cPtr == 0) ? null : new SWIGTYPE_p_int(cPtr, false); + } + + public long size() { + return swigfaissJNI.IntVector_size(swigCPtr, this); + } + + public int at(long n) { + return swigfaissJNI.IntVector_at(swigCPtr, this, n); + } + + public void resize(long n) { + swigfaissJNI.IntVector_resize(swigCPtr, this, n); + } + + public void reserve(long n) { + swigfaissJNI.IntVector_reserve(swigCPtr, this, n); + } + + public void swap(IntVector other) { + swigfaissJNI.IntVector_swap(swigCPtr, this, IntVector.getCPtr(other), other); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/InterruptCallback.java b/ann/src/main/java/com/twitter/ann/faiss/swig/InterruptCallback.java new file mode 100644 index 0000000000..3213696de7 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/InterruptCallback.java @@ -0,0 +1,59 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class InterruptCallback { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected InterruptCallback(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(InterruptCallback obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_InterruptCallback(swigCPtr); + } + swigCPtr = 0; + } + } + + public boolean want_interrupt() { + return swigfaissJNI.InterruptCallback_want_interrupt(swigCPtr, this); + } + + public static void clear_instance() { + swigfaissJNI.InterruptCallback_clear_instance(); + } + + public static void check() { + swigfaissJNI.InterruptCallback_check(); + } + + public static boolean is_interrupted() { + return swigfaissJNI.InterruptCallback_is_interrupted(); + } + + public static long get_period_hint(long flops) { + return swigfaissJNI.InterruptCallback_get_period_hint(flops); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/IntersectionCriterion.java b/ann/src/main/java/com/twitter/ann/faiss/swig/IntersectionCriterion.java new file mode 100644 index 0000000000..80f44e1d2e --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/IntersectionCriterion.java @@ -0,0 +1,55 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class IntersectionCriterion extends AutoTuneCriterion { + private transient long swigCPtr; + + protected IntersectionCriterion(long cPtr, boolean cMemoryOwn) { + super(swigfaissJNI.IntersectionCriterion_SWIGUpcast(cPtr), cMemoryOwn); + swigCPtr = cPtr; + } + + protected static long getCPtr(IntersectionCriterion obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_IntersectionCriterion(swigCPtr); + } + swigCPtr = 0; + } + super.delete(); + } + + public void setR(long value) { + swigfaissJNI.IntersectionCriterion_R_set(swigCPtr, this, value); + } + + public long getR() { + return swigfaissJNI.IntersectionCriterion_R_get(swigCPtr, this); +} + + public IntersectionCriterion(long nq, long R) { + this(swigfaissJNI.new_IntersectionCriterion(nq, R), true); + } + + public double evaluate(SWIGTYPE_p_float D, LongVector I) { + return swigfaissJNI.IntersectionCriterion_evaluate(swigCPtr, this, SWIGTYPE_p_float.getCPtr(D), SWIGTYPE_p_long_long.getCPtr(I.data()), I); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/InvertedLists.java b/ann/src/main/java/com/twitter/ann/faiss/swig/InvertedLists.java new file mode 100644 index 0000000000..d7311cc75c --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/InvertedLists.java @@ -0,0 +1,262 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class InvertedLists { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected InvertedLists(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(InvertedLists obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_InvertedLists(swigCPtr); + } + swigCPtr = 0; + } + } + + public void setNlist(long value) { + swigfaissJNI.InvertedLists_nlist_set(swigCPtr, this, value); + } + + public long getNlist() { + return swigfaissJNI.InvertedLists_nlist_get(swigCPtr, this); + } + + public void setCode_size(long value) { + swigfaissJNI.InvertedLists_code_size_set(swigCPtr, this, value); + } + + public long getCode_size() { + return swigfaissJNI.InvertedLists_code_size_get(swigCPtr, this); + } + + public long list_size(long list_no) { + return swigfaissJNI.InvertedLists_list_size(swigCPtr, this, list_no); + } + + public SWIGTYPE_p_unsigned_char get_codes(long list_no) { + long cPtr = swigfaissJNI.InvertedLists_get_codes(swigCPtr, this, list_no); + return (cPtr == 0) ? null : new SWIGTYPE_p_unsigned_char(cPtr, false); + } + + public LongVector get_ids(long list_no) { + return new LongVector(swigfaissJNI.InvertedLists_get_ids(swigCPtr, this, list_no), false); +} + + public void release_codes(long list_no, SWIGTYPE_p_unsigned_char codes) { + swigfaissJNI.InvertedLists_release_codes(swigCPtr, this, list_no, SWIGTYPE_p_unsigned_char.getCPtr(codes)); + } + + public void release_ids(long list_no, LongVector ids) { + swigfaissJNI.InvertedLists_release_ids(swigCPtr, this, list_no, SWIGTYPE_p_long_long.getCPtr(ids.data()), ids); + } + + public long get_single_id(long list_no, long offset) { + return swigfaissJNI.InvertedLists_get_single_id(swigCPtr, this, list_no, offset); +} + + public SWIGTYPE_p_unsigned_char get_single_code(long list_no, long offset) { + long cPtr = swigfaissJNI.InvertedLists_get_single_code(swigCPtr, this, list_no, offset); + return (cPtr == 0) ? null : new SWIGTYPE_p_unsigned_char(cPtr, false); + } + + public void prefetch_lists(LongVector list_nos, int nlist) { + swigfaissJNI.InvertedLists_prefetch_lists(swigCPtr, this, SWIGTYPE_p_long_long.getCPtr(list_nos.data()), list_nos, nlist); + } + + public long add_entry(long list_no, long theid, SWIGTYPE_p_unsigned_char code) { + return swigfaissJNI.InvertedLists_add_entry(swigCPtr, this, list_no, theid, SWIGTYPE_p_unsigned_char.getCPtr(code)); + } + + public long add_entries(long list_no, long n_entry, LongVector ids, SWIGTYPE_p_unsigned_char code) { + return swigfaissJNI.InvertedLists_add_entries(swigCPtr, this, list_no, n_entry, SWIGTYPE_p_long_long.getCPtr(ids.data()), ids, SWIGTYPE_p_unsigned_char.getCPtr(code)); + } + + public void update_entry(long list_no, long offset, long id, SWIGTYPE_p_unsigned_char code) { + swigfaissJNI.InvertedLists_update_entry(swigCPtr, this, list_no, offset, id, SWIGTYPE_p_unsigned_char.getCPtr(code)); + } + + public void update_entries(long list_no, long offset, long n_entry, LongVector ids, SWIGTYPE_p_unsigned_char code) { + swigfaissJNI.InvertedLists_update_entries(swigCPtr, this, list_no, offset, n_entry, SWIGTYPE_p_long_long.getCPtr(ids.data()), ids, SWIGTYPE_p_unsigned_char.getCPtr(code)); + } + + public void resize(long list_no, long new_size) { + swigfaissJNI.InvertedLists_resize(swigCPtr, this, list_no, new_size); + } + + public void reset() { + swigfaissJNI.InvertedLists_reset(swigCPtr, this); + } + + public void merge_from(InvertedLists oivf, long add_id) { + swigfaissJNI.InvertedLists_merge_from(swigCPtr, this, InvertedLists.getCPtr(oivf), oivf, add_id); + } + + public double imbalance_factor() { + return swigfaissJNI.InvertedLists_imbalance_factor(swigCPtr, this); + } + + public void print_stats() { + swigfaissJNI.InvertedLists_print_stats(swigCPtr, this); + } + + public long compute_ntotal() { + return swigfaissJNI.InvertedLists_compute_ntotal(swigCPtr, this); + } + + static public class ScopedIds { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected ScopedIds(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(ScopedIds obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_InvertedLists_ScopedIds(swigCPtr); + } + swigCPtr = 0; + } + } + + public void setIl(InvertedLists value) { + swigfaissJNI.InvertedLists_ScopedIds_il_set(swigCPtr, this, InvertedLists.getCPtr(value), value); + } + + public InvertedLists getIl() { + long cPtr = swigfaissJNI.InvertedLists_ScopedIds_il_get(swigCPtr, this); + return (cPtr == 0) ? null : new InvertedLists(cPtr, false); + } + + public void setIds(LongVector value) { + swigfaissJNI.InvertedLists_ScopedIds_ids_set(swigCPtr, this, SWIGTYPE_p_long_long.getCPtr(value.data()), value); + } + + public LongVector getIds() { + return new LongVector(swigfaissJNI.InvertedLists_ScopedIds_ids_get(swigCPtr, this), false); + } + + public void setList_no(long value) { + swigfaissJNI.InvertedLists_ScopedIds_list_no_set(swigCPtr, this, value); + } + + public long getList_no() { + return swigfaissJNI.InvertedLists_ScopedIds_list_no_get(swigCPtr, this); + } + + public ScopedIds(InvertedLists il, long list_no) { + this(swigfaissJNI.new_InvertedLists_ScopedIds(InvertedLists.getCPtr(il), il, list_no), true); + } + + public LongVector get() { + return new LongVector(swigfaissJNI.InvertedLists_ScopedIds_get(swigCPtr, this), false); + } + + } + + static public class ScopedCodes { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected ScopedCodes(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(ScopedCodes obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_InvertedLists_ScopedCodes(swigCPtr); + } + swigCPtr = 0; + } + } + + public void setIl(InvertedLists value) { + swigfaissJNI.InvertedLists_ScopedCodes_il_set(swigCPtr, this, InvertedLists.getCPtr(value), value); + } + + public InvertedLists getIl() { + long cPtr = swigfaissJNI.InvertedLists_ScopedCodes_il_get(swigCPtr, this); + return (cPtr == 0) ? null : new InvertedLists(cPtr, false); + } + + public void setCodes(SWIGTYPE_p_unsigned_char value) { + swigfaissJNI.InvertedLists_ScopedCodes_codes_set(swigCPtr, this, SWIGTYPE_p_unsigned_char.getCPtr(value)); + } + + public SWIGTYPE_p_unsigned_char getCodes() { + long cPtr = swigfaissJNI.InvertedLists_ScopedCodes_codes_get(swigCPtr, this); + return (cPtr == 0) ? null : new SWIGTYPE_p_unsigned_char(cPtr, false); + } + + public void setList_no(long value) { + swigfaissJNI.InvertedLists_ScopedCodes_list_no_set(swigCPtr, this, value); + } + + public long getList_no() { + return swigfaissJNI.InvertedLists_ScopedCodes_list_no_get(swigCPtr, this); + } + + public ScopedCodes(InvertedLists il, long list_no) { + this(swigfaissJNI.new_InvertedLists_ScopedCodes__SWIG_0(InvertedLists.getCPtr(il), il, list_no), true); + } + + public ScopedCodes(InvertedLists il, long list_no, long offset) { + this(swigfaissJNI.new_InvertedLists_ScopedCodes__SWIG_1(InvertedLists.getCPtr(il), il, list_no, offset), true); + } + + public SWIGTYPE_p_unsigned_char get() { + long cPtr = swigfaissJNI.InvertedLists_ScopedCodes_get(swigCPtr, this); + return (cPtr == 0) ? null : new SWIGTYPE_p_unsigned_char(cPtr, false); + } + + } + + public final static long INVALID_CODE_SIZE = swigfaissJNI.InvertedLists_INVALID_CODE_SIZE_get(); +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/InvertedListsPtrVector.java b/ann/src/main/java/com/twitter/ann/faiss/swig/InvertedListsPtrVector.java new file mode 100644 index 0000000000..4c9a14b74f --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/InvertedListsPtrVector.java @@ -0,0 +1,77 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class InvertedListsPtrVector { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected InvertedListsPtrVector(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(InvertedListsPtrVector obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_InvertedListsPtrVector(swigCPtr); + } + swigCPtr = 0; + } + } + + public InvertedListsPtrVector() { + this(swigfaissJNI.new_InvertedListsPtrVector(), true); + } + + public void push_back(InvertedLists arg0) { + swigfaissJNI.InvertedListsPtrVector_push_back(swigCPtr, this, InvertedLists.getCPtr(arg0), arg0); + } + + public void clear() { + swigfaissJNI.InvertedListsPtrVector_clear(swigCPtr, this); + } + + public SWIGTYPE_p_p_faiss__InvertedLists data() { + long cPtr = swigfaissJNI.InvertedListsPtrVector_data(swigCPtr, this); + return (cPtr == 0) ? null : new SWIGTYPE_p_p_faiss__InvertedLists(cPtr, false); + } + + public long size() { + return swigfaissJNI.InvertedListsPtrVector_size(swigCPtr, this); + } + + public InvertedLists at(long n) { + long cPtr = swigfaissJNI.InvertedListsPtrVector_at(swigCPtr, this, n); + return (cPtr == 0) ? null : new InvertedLists(cPtr, false); + } + + public void resize(long n) { + swigfaissJNI.InvertedListsPtrVector_resize(swigCPtr, this, n); + } + + public void reserve(long n) { + swigfaissJNI.InvertedListsPtrVector_reserve(swigCPtr, this, n); + } + + public void swap(InvertedListsPtrVector other) { + swigfaissJNI.InvertedListsPtrVector_swap(swigCPtr, this, InvertedListsPtrVector.getCPtr(other), other); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/Level1Quantizer.java b/ann/src/main/java/com/twitter/ann/faiss/swig/Level1Quantizer.java new file mode 100644 index 0000000000..9de2d8c443 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/Level1Quantizer.java @@ -0,0 +1,114 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class Level1Quantizer { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected Level1Quantizer(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(Level1Quantizer obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_Level1Quantizer(swigCPtr); + } + swigCPtr = 0; + } + } + + public void setQuantizer(Index value) { + swigfaissJNI.Level1Quantizer_quantizer_set(swigCPtr, this, Index.getCPtr(value), value); + } + + public Index getQuantizer() { + long cPtr = swigfaissJNI.Level1Quantizer_quantizer_get(swigCPtr, this); + return (cPtr == 0) ? null : new Index(cPtr, false); + } + + public void setNlist(long value) { + swigfaissJNI.Level1Quantizer_nlist_set(swigCPtr, this, value); + } + + public long getNlist() { + return swigfaissJNI.Level1Quantizer_nlist_get(swigCPtr, this); + } + + public void setQuantizer_trains_alone(char value) { + swigfaissJNI.Level1Quantizer_quantizer_trains_alone_set(swigCPtr, this, value); + } + + public char getQuantizer_trains_alone() { + return swigfaissJNI.Level1Quantizer_quantizer_trains_alone_get(swigCPtr, this); + } + + public void setOwn_fields(boolean value) { + swigfaissJNI.Level1Quantizer_own_fields_set(swigCPtr, this, value); + } + + public boolean getOwn_fields() { + return swigfaissJNI.Level1Quantizer_own_fields_get(swigCPtr, this); + } + + public void setCp(ClusteringParameters value) { + swigfaissJNI.Level1Quantizer_cp_set(swigCPtr, this, ClusteringParameters.getCPtr(value), value); + } + + public ClusteringParameters getCp() { + long cPtr = swigfaissJNI.Level1Quantizer_cp_get(swigCPtr, this); + return (cPtr == 0) ? null : new ClusteringParameters(cPtr, false); + } + + public void setClustering_index(Index value) { + swigfaissJNI.Level1Quantizer_clustering_index_set(swigCPtr, this, Index.getCPtr(value), value); + } + + public Index getClustering_index() { + long cPtr = swigfaissJNI.Level1Quantizer_clustering_index_get(swigCPtr, this); + return (cPtr == 0) ? null : new Index(cPtr, false); + } + + public void train_q1(long n, SWIGTYPE_p_float x, boolean verbose, MetricType metric_type) { + swigfaissJNI.Level1Quantizer_train_q1(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x), verbose, metric_type.swigValue()); + } + + public long coarse_code_size() { + return swigfaissJNI.Level1Quantizer_coarse_code_size(swigCPtr, this); + } + + public void encode_listno(long list_no, SWIGTYPE_p_unsigned_char code) { + swigfaissJNI.Level1Quantizer_encode_listno(swigCPtr, this, list_no, SWIGTYPE_p_unsigned_char.getCPtr(code)); + } + + public long decode_listno(SWIGTYPE_p_unsigned_char code) { + return swigfaissJNI.Level1Quantizer_decode_listno(swigCPtr, this, SWIGTYPE_p_unsigned_char.getCPtr(code)); +} + + public Level1Quantizer(Index quantizer, long nlist) { + this(swigfaissJNI.new_Level1Quantizer__SWIG_0(Index.getCPtr(quantizer), quantizer, nlist), true); + } + + public Level1Quantizer() { + this(swigfaissJNI.new_Level1Quantizer__SWIG_1(), true); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/LinearTransform.java b/ann/src/main/java/com/twitter/ann/faiss/swig/LinearTransform.java new file mode 100644 index 0000000000..6353fb0c2e --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/LinearTransform.java @@ -0,0 +1,117 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class LinearTransform extends VectorTransform { + private transient long swigCPtr; + + protected LinearTransform(long cPtr, boolean cMemoryOwn) { + super(swigfaissJNI.LinearTransform_SWIGUpcast(cPtr), cMemoryOwn); + swigCPtr = cPtr; + } + + protected static long getCPtr(LinearTransform obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_LinearTransform(swigCPtr); + } + swigCPtr = 0; + } + super.delete(); + } + + public void setHave_bias(boolean value) { + swigfaissJNI.LinearTransform_have_bias_set(swigCPtr, this, value); + } + + public boolean getHave_bias() { + return swigfaissJNI.LinearTransform_have_bias_get(swigCPtr, this); + } + + public void setIs_orthonormal(boolean value) { + swigfaissJNI.LinearTransform_is_orthonormal_set(swigCPtr, this, value); + } + + public boolean getIs_orthonormal() { + return swigfaissJNI.LinearTransform_is_orthonormal_get(swigCPtr, this); + } + + public void setA(FloatVector value) { + swigfaissJNI.LinearTransform_A_set(swigCPtr, this, FloatVector.getCPtr(value), value); + } + + public FloatVector getA() { + long cPtr = swigfaissJNI.LinearTransform_A_get(swigCPtr, this); + return (cPtr == 0) ? null : new FloatVector(cPtr, false); + } + + public void setB(FloatVector value) { + swigfaissJNI.LinearTransform_b_set(swigCPtr, this, FloatVector.getCPtr(value), value); + } + + public FloatVector getB() { + long cPtr = swigfaissJNI.LinearTransform_b_get(swigCPtr, this); + return (cPtr == 0) ? null : new FloatVector(cPtr, false); + } + + public LinearTransform(int d_in, int d_out, boolean have_bias) { + this(swigfaissJNI.new_LinearTransform__SWIG_0(d_in, d_out, have_bias), true); + } + + public LinearTransform(int d_in, int d_out) { + this(swigfaissJNI.new_LinearTransform__SWIG_1(d_in, d_out), true); + } + + public LinearTransform(int d_in) { + this(swigfaissJNI.new_LinearTransform__SWIG_2(d_in), true); + } + + public LinearTransform() { + this(swigfaissJNI.new_LinearTransform__SWIG_3(), true); + } + + public void apply_noalloc(long n, SWIGTYPE_p_float x, SWIGTYPE_p_float xt) { + swigfaissJNI.LinearTransform_apply_noalloc(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x), SWIGTYPE_p_float.getCPtr(xt)); + } + + public void transform_transpose(long n, SWIGTYPE_p_float y, SWIGTYPE_p_float x) { + swigfaissJNI.LinearTransform_transform_transpose(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(y), SWIGTYPE_p_float.getCPtr(x)); + } + + public void reverse_transform(long n, SWIGTYPE_p_float xt, SWIGTYPE_p_float x) { + swigfaissJNI.LinearTransform_reverse_transform(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(xt), SWIGTYPE_p_float.getCPtr(x)); + } + + public void set_is_orthonormal() { + swigfaissJNI.LinearTransform_set_is_orthonormal(swigCPtr, this); + } + + public void setVerbose(boolean value) { + swigfaissJNI.LinearTransform_verbose_set(swigCPtr, this, value); + } + + public boolean getVerbose() { + return swigfaissJNI.LinearTransform_verbose_get(swigCPtr, this); + } + + public void print_if_verbose(String name, DoubleVector mat, int n, int d) { + swigfaissJNI.LinearTransform_print_if_verbose(swigCPtr, this, name, DoubleVector.getCPtr(mat), mat, n, d); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/LongVector.java b/ann/src/main/java/com/twitter/ann/faiss/swig/LongVector.java new file mode 100644 index 0000000000..89c5cdf5a0 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/LongVector.java @@ -0,0 +1,76 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class LongVector { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected LongVector(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(LongVector obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_LongVector(swigCPtr); + } + swigCPtr = 0; + } + } + + public LongVector() { + this(swigfaissJNI.new_LongVector(), true); + } + + public void push_back(long arg0) { + swigfaissJNI.LongVector_push_back(swigCPtr, this, arg0); + } + + public void clear() { + swigfaissJNI.LongVector_clear(swigCPtr, this); + } + + public SWIGTYPE_p_long_long data() { + long cPtr = swigfaissJNI.LongVector_data(swigCPtr, this); + return (cPtr == 0) ? null : new SWIGTYPE_p_long_long(cPtr, false); + } + + public long size() { + return swigfaissJNI.LongVector_size(swigCPtr, this); + } + + public long at(long n) { + return swigfaissJNI.LongVector_at(swigCPtr, this, n); + } + + public void resize(long n) { + swigfaissJNI.LongVector_resize(swigCPtr, this, n); + } + + public void reserve(long n) { + swigfaissJNI.LongVector_reserve(swigCPtr, this, n); + } + + public void swap(LongVector other) { + swigfaissJNI.LongVector_swap(swigCPtr, this, LongVector.getCPtr(other), other); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/LongVectorVector.java b/ann/src/main/java/com/twitter/ann/faiss/swig/LongVectorVector.java new file mode 100644 index 0000000000..485573bac1 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/LongVectorVector.java @@ -0,0 +1,76 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class LongVectorVector { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected LongVectorVector(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(LongVectorVector obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_LongVectorVector(swigCPtr); + } + swigCPtr = 0; + } + } + + public LongVectorVector() { + this(swigfaissJNI.new_LongVectorVector(), true); + } + + public void push_back(SWIGTYPE_p_std__vectorT_long_t arg0) { + swigfaissJNI.LongVectorVector_push_back(swigCPtr, this, SWIGTYPE_p_std__vectorT_long_t.getCPtr(arg0)); + } + + public void clear() { + swigfaissJNI.LongVectorVector_clear(swigCPtr, this); + } + + public SWIGTYPE_p_std__vectorT_long_t data() { + long cPtr = swigfaissJNI.LongVectorVector_data(swigCPtr, this); + return (cPtr == 0) ? null : new SWIGTYPE_p_std__vectorT_long_t(cPtr, false); + } + + public long size() { + return swigfaissJNI.LongVectorVector_size(swigCPtr, this); + } + + public SWIGTYPE_p_std__vectorT_long_t at(long n) { + return new SWIGTYPE_p_std__vectorT_long_t(swigfaissJNI.LongVectorVector_at(swigCPtr, this, n), true); + } + + public void resize(long n) { + swigfaissJNI.LongVectorVector_resize(swigCPtr, this, n); + } + + public void reserve(long n) { + swigfaissJNI.LongVectorVector_reserve(swigCPtr, this, n); + } + + public void swap(LongVectorVector other) { + swigfaissJNI.LongVectorVector_swap(swigCPtr, this, LongVectorVector.getCPtr(other), other); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/MapLong2Long.java b/ann/src/main/java/com/twitter/ann/faiss/swig/MapLong2Long.java new file mode 100644 index 0000000000..0ecaaf0539 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/MapLong2Long.java @@ -0,0 +1,63 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class MapLong2Long { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected MapLong2Long(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(MapLong2Long obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_MapLong2Long(swigCPtr); + } + swigCPtr = 0; + } + } + + public void setMap(SWIGTYPE_p_std__unordered_mapT_long_long_t value) { + swigfaissJNI.MapLong2Long_map_set(swigCPtr, this, SWIGTYPE_p_std__unordered_mapT_long_long_t.getCPtr(value)); + } + + public SWIGTYPE_p_std__unordered_mapT_long_long_t getMap() { + return new SWIGTYPE_p_std__unordered_mapT_long_long_t(swigfaissJNI.MapLong2Long_map_get(swigCPtr, this), true); + } + + public void add(long n, SWIGTYPE_p_long keys, SWIGTYPE_p_long vals) { + swigfaissJNI.MapLong2Long_add(swigCPtr, this, n, SWIGTYPE_p_long.getCPtr(keys), SWIGTYPE_p_long.getCPtr(vals)); + } + + public int search(int key) { + return swigfaissJNI.MapLong2Long_search(swigCPtr, this, key); + } + + public void search_multiple(long n, SWIGTYPE_p_long keys, SWIGTYPE_p_long vals) { + swigfaissJNI.MapLong2Long_search_multiple(swigCPtr, this, n, SWIGTYPE_p_long.getCPtr(keys), SWIGTYPE_p_long.getCPtr(vals)); + } + + public MapLong2Long() { + this(swigfaissJNI.new_MapLong2Long(), true); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/MaskedInvertedLists.java b/ann/src/main/java/com/twitter/ann/faiss/swig/MaskedInvertedLists.java new file mode 100644 index 0000000000..70dbb1a72c --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/MaskedInvertedLists.java @@ -0,0 +1,95 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class MaskedInvertedLists extends ReadOnlyInvertedLists { + private transient long swigCPtr; + + protected MaskedInvertedLists(long cPtr, boolean cMemoryOwn) { + super(swigfaissJNI.MaskedInvertedLists_SWIGUpcast(cPtr), cMemoryOwn); + swigCPtr = cPtr; + } + + protected static long getCPtr(MaskedInvertedLists obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_MaskedInvertedLists(swigCPtr); + } + swigCPtr = 0; + } + super.delete(); + } + + public void setIl0(InvertedLists value) { + swigfaissJNI.MaskedInvertedLists_il0_set(swigCPtr, this, InvertedLists.getCPtr(value), value); + } + + public InvertedLists getIl0() { + long cPtr = swigfaissJNI.MaskedInvertedLists_il0_get(swigCPtr, this); + return (cPtr == 0) ? null : new InvertedLists(cPtr, false); + } + + public void setIl1(InvertedLists value) { + swigfaissJNI.MaskedInvertedLists_il1_set(swigCPtr, this, InvertedLists.getCPtr(value), value); + } + + public InvertedLists getIl1() { + long cPtr = swigfaissJNI.MaskedInvertedLists_il1_get(swigCPtr, this); + return (cPtr == 0) ? null : new InvertedLists(cPtr, false); + } + + public MaskedInvertedLists(InvertedLists il0, InvertedLists il1) { + this(swigfaissJNI.new_MaskedInvertedLists(InvertedLists.getCPtr(il0), il0, InvertedLists.getCPtr(il1), il1), true); + } + + public long list_size(long list_no) { + return swigfaissJNI.MaskedInvertedLists_list_size(swigCPtr, this, list_no); + } + + public SWIGTYPE_p_unsigned_char get_codes(long list_no) { + long cPtr = swigfaissJNI.MaskedInvertedLists_get_codes(swigCPtr, this, list_no); + return (cPtr == 0) ? null : new SWIGTYPE_p_unsigned_char(cPtr, false); + } + + public LongVector get_ids(long list_no) { + return new LongVector(swigfaissJNI.MaskedInvertedLists_get_ids(swigCPtr, this, list_no), false); +} + + public void release_codes(long list_no, SWIGTYPE_p_unsigned_char codes) { + swigfaissJNI.MaskedInvertedLists_release_codes(swigCPtr, this, list_no, SWIGTYPE_p_unsigned_char.getCPtr(codes)); + } + + public void release_ids(long list_no, LongVector ids) { + swigfaissJNI.MaskedInvertedLists_release_ids(swigCPtr, this, list_no, SWIGTYPE_p_long_long.getCPtr(ids.data()), ids); + } + + public long get_single_id(long list_no, long offset) { + return swigfaissJNI.MaskedInvertedLists_get_single_id(swigCPtr, this, list_no, offset); +} + + public SWIGTYPE_p_unsigned_char get_single_code(long list_no, long offset) { + long cPtr = swigfaissJNI.MaskedInvertedLists_get_single_code(swigCPtr, this, list_no, offset); + return (cPtr == 0) ? null : new SWIGTYPE_p_unsigned_char(cPtr, false); + } + + public void prefetch_lists(LongVector list_nos, int nlist) { + swigfaissJNI.MaskedInvertedLists_prefetch_lists(swigCPtr, this, SWIGTYPE_p_long_long.getCPtr(list_nos.data()), list_nos, nlist); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/MetricType.java b/ann/src/main/java/com/twitter/ann/faiss/swig/MetricType.java new file mode 100644 index 0000000000..5382e15445 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/MetricType.java @@ -0,0 +1,60 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public final class MetricType { + public final static MetricType METRIC_INNER_PRODUCT = new MetricType("METRIC_INNER_PRODUCT", swigfaissJNI.METRIC_INNER_PRODUCT_get()); + public final static MetricType METRIC_L2 = new MetricType("METRIC_L2", swigfaissJNI.METRIC_L2_get()); + public final static MetricType METRIC_L1 = new MetricType("METRIC_L1"); + public final static MetricType METRIC_Linf = new MetricType("METRIC_Linf"); + public final static MetricType METRIC_Lp = new MetricType("METRIC_Lp"); + public final static MetricType METRIC_Canberra = new MetricType("METRIC_Canberra", swigfaissJNI.METRIC_Canberra_get()); + public final static MetricType METRIC_BrayCurtis = new MetricType("METRIC_BrayCurtis"); + public final static MetricType METRIC_JensenShannon = new MetricType("METRIC_JensenShannon"); + + public final int swigValue() { + return swigValue; + } + + public String toString() { + return swigName; + } + + public static MetricType swigToEnum(int swigValue) { + if (swigValue < swigValues.length && swigValue >= 0 && swigValues[swigValue].swigValue == swigValue) + return swigValues[swigValue]; + for (int i = 0; i < swigValues.length; i++) + if (swigValues[i].swigValue == swigValue) + return swigValues[i]; + throw new IllegalArgumentException("No enum " + MetricType.class + " with value " + swigValue); + } + + private MetricType(String swigName) { + this.swigName = swigName; + this.swigValue = swigNext++; + } + + private MetricType(String swigName, int swigValue) { + this.swigName = swigName; + this.swigValue = swigValue; + swigNext = swigValue+1; + } + + private MetricType(String swigName, MetricType swigEnum) { + this.swigName = swigName; + this.swigValue = swigEnum.swigValue; + swigNext = this.swigValue+1; + } + + private static MetricType[] swigValues = { METRIC_INNER_PRODUCT, METRIC_L2, METRIC_L1, METRIC_Linf, METRIC_Lp, METRIC_Canberra, METRIC_BrayCurtis, METRIC_JensenShannon }; + private static int swigNext = 0; + private final int swigValue; + private final String swigName; +} + diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/MultiIndexQuantizer.java b/ann/src/main/java/com/twitter/ann/faiss/swig/MultiIndexQuantizer.java new file mode 100644 index 0000000000..b0ee9c3c4e --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/MultiIndexQuantizer.java @@ -0,0 +1,76 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class MultiIndexQuantizer extends Index { + private transient long swigCPtr; + + protected MultiIndexQuantizer(long cPtr, boolean cMemoryOwn) { + super(swigfaissJNI.MultiIndexQuantizer_SWIGUpcast(cPtr), cMemoryOwn); + swigCPtr = cPtr; + } + + protected static long getCPtr(MultiIndexQuantizer obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_MultiIndexQuantizer(swigCPtr); + } + swigCPtr = 0; + } + super.delete(); + } + + public void setPq(ProductQuantizer value) { + swigfaissJNI.MultiIndexQuantizer_pq_set(swigCPtr, this, ProductQuantizer.getCPtr(value), value); + } + + public ProductQuantizer getPq() { + long cPtr = swigfaissJNI.MultiIndexQuantizer_pq_get(swigCPtr, this); + return (cPtr == 0) ? null : new ProductQuantizer(cPtr, false); + } + + public MultiIndexQuantizer(int d, long M, long nbits) { + this(swigfaissJNI.new_MultiIndexQuantizer__SWIG_0(d, M, nbits), true); + } + + public void train(long n, SWIGTYPE_p_float x) { + swigfaissJNI.MultiIndexQuantizer_train(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x)); + } + + public void search(long n, SWIGTYPE_p_float x, long k, SWIGTYPE_p_float distances, LongVector labels) { + swigfaissJNI.MultiIndexQuantizer_search(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x), k, SWIGTYPE_p_float.getCPtr(distances), SWIGTYPE_p_long_long.getCPtr(labels.data()), labels); + } + + public void add(long n, SWIGTYPE_p_float x) { + swigfaissJNI.MultiIndexQuantizer_add(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x)); + } + + public void reset() { + swigfaissJNI.MultiIndexQuantizer_reset(swigCPtr, this); + } + + public MultiIndexQuantizer() { + this(swigfaissJNI.new_MultiIndexQuantizer__SWIG_1(), true); + } + + public void reconstruct(long key, SWIGTYPE_p_float recons) { + swigfaissJNI.MultiIndexQuantizer_reconstruct(swigCPtr, this, key, SWIGTYPE_p_float.getCPtr(recons)); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/MultiIndexQuantizer2.java b/ann/src/main/java/com/twitter/ann/faiss/swig/MultiIndexQuantizer2.java new file mode 100644 index 0000000000..519c123c4a --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/MultiIndexQuantizer2.java @@ -0,0 +1,72 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class MultiIndexQuantizer2 extends MultiIndexQuantizer { + private transient long swigCPtr; + + protected MultiIndexQuantizer2(long cPtr, boolean cMemoryOwn) { + super(swigfaissJNI.MultiIndexQuantizer2_SWIGUpcast(cPtr), cMemoryOwn); + swigCPtr = cPtr; + } + + protected static long getCPtr(MultiIndexQuantizer2 obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_MultiIndexQuantizer2(swigCPtr); + } + swigCPtr = 0; + } + super.delete(); + } + + public void setAssign_indexes(SWIGTYPE_p_std__vectorT_faiss__Index_p_t value) { + swigfaissJNI.MultiIndexQuantizer2_assign_indexes_set(swigCPtr, this, SWIGTYPE_p_std__vectorT_faiss__Index_p_t.getCPtr(value)); + } + + public SWIGTYPE_p_std__vectorT_faiss__Index_p_t getAssign_indexes() { + long cPtr = swigfaissJNI.MultiIndexQuantizer2_assign_indexes_get(swigCPtr, this); + return (cPtr == 0) ? null : new SWIGTYPE_p_std__vectorT_faiss__Index_p_t(cPtr, false); + } + + public void setOwn_fields(boolean value) { + swigfaissJNI.MultiIndexQuantizer2_own_fields_set(swigCPtr, this, value); + } + + public boolean getOwn_fields() { + return swigfaissJNI.MultiIndexQuantizer2_own_fields_get(swigCPtr, this); + } + + public MultiIndexQuantizer2(int d, long M, long nbits, SWIGTYPE_p_p_faiss__Index indexes) { + this(swigfaissJNI.new_MultiIndexQuantizer2__SWIG_0(d, M, nbits, SWIGTYPE_p_p_faiss__Index.getCPtr(indexes)), true); + } + + public MultiIndexQuantizer2(int d, long nbits, Index assign_index_0, Index assign_index_1) { + this(swigfaissJNI.new_MultiIndexQuantizer2__SWIG_1(d, nbits, Index.getCPtr(assign_index_0), assign_index_0, Index.getCPtr(assign_index_1), assign_index_1), true); + } + + public void train(long n, SWIGTYPE_p_float x) { + swigfaissJNI.MultiIndexQuantizer2_train(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x)); + } + + public void search(long n, SWIGTYPE_p_float x, long k, SWIGTYPE_p_float distances, LongVector labels) { + swigfaissJNI.MultiIndexQuantizer2_search(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x), k, SWIGTYPE_p_float.getCPtr(distances), SWIGTYPE_p_long_long.getCPtr(labels.data()), labels); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/NormalizationTransform.java b/ann/src/main/java/com/twitter/ann/faiss/swig/NormalizationTransform.java new file mode 100644 index 0000000000..aaa38642c3 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/NormalizationTransform.java @@ -0,0 +1,67 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class NormalizationTransform extends VectorTransform { + private transient long swigCPtr; + + protected NormalizationTransform(long cPtr, boolean cMemoryOwn) { + super(swigfaissJNI.NormalizationTransform_SWIGUpcast(cPtr), cMemoryOwn); + swigCPtr = cPtr; + } + + protected static long getCPtr(NormalizationTransform obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_NormalizationTransform(swigCPtr); + } + swigCPtr = 0; + } + super.delete(); + } + + public void setNorm(float value) { + swigfaissJNI.NormalizationTransform_norm_set(swigCPtr, this, value); + } + + public float getNorm() { + return swigfaissJNI.NormalizationTransform_norm_get(swigCPtr, this); + } + + public NormalizationTransform(int d, float norm) { + this(swigfaissJNI.new_NormalizationTransform__SWIG_0(d, norm), true); + } + + public NormalizationTransform(int d) { + this(swigfaissJNI.new_NormalizationTransform__SWIG_1(d), true); + } + + public NormalizationTransform() { + this(swigfaissJNI.new_NormalizationTransform__SWIG_2(), true); + } + + public void apply_noalloc(long n, SWIGTYPE_p_float x, SWIGTYPE_p_float xt) { + swigfaissJNI.NormalizationTransform_apply_noalloc(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x), SWIGTYPE_p_float.getCPtr(xt)); + } + + public void reverse_transform(long n, SWIGTYPE_p_float xt, SWIGTYPE_p_float x) { + swigfaissJNI.NormalizationTransform_reverse_transform(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(xt), SWIGTYPE_p_float.getCPtr(x)); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/OPQMatrix.java b/ann/src/main/java/com/twitter/ann/faiss/swig/OPQMatrix.java new file mode 100644 index 0000000000..fafaf07d92 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/OPQMatrix.java @@ -0,0 +1,116 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class OPQMatrix extends LinearTransform { + private transient long swigCPtr; + + protected OPQMatrix(long cPtr, boolean cMemoryOwn) { + super(swigfaissJNI.OPQMatrix_SWIGUpcast(cPtr), cMemoryOwn); + swigCPtr = cPtr; + } + + protected static long getCPtr(OPQMatrix obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_OPQMatrix(swigCPtr); + } + swigCPtr = 0; + } + super.delete(); + } + + public void setM(int value) { + swigfaissJNI.OPQMatrix_M_set(swigCPtr, this, value); + } + + public int getM() { + return swigfaissJNI.OPQMatrix_M_get(swigCPtr, this); + } + + public void setNiter(int value) { + swigfaissJNI.OPQMatrix_niter_set(swigCPtr, this, value); + } + + public int getNiter() { + return swigfaissJNI.OPQMatrix_niter_get(swigCPtr, this); + } + + public void setNiter_pq(int value) { + swigfaissJNI.OPQMatrix_niter_pq_set(swigCPtr, this, value); + } + + public int getNiter_pq() { + return swigfaissJNI.OPQMatrix_niter_pq_get(swigCPtr, this); + } + + public void setNiter_pq_0(int value) { + swigfaissJNI.OPQMatrix_niter_pq_0_set(swigCPtr, this, value); + } + + public int getNiter_pq_0() { + return swigfaissJNI.OPQMatrix_niter_pq_0_get(swigCPtr, this); + } + + public void setMax_train_points(long value) { + swigfaissJNI.OPQMatrix_max_train_points_set(swigCPtr, this, value); + } + + public long getMax_train_points() { + return swigfaissJNI.OPQMatrix_max_train_points_get(swigCPtr, this); + } + + public void setVerbose(boolean value) { + swigfaissJNI.OPQMatrix_verbose_set(swigCPtr, this, value); + } + + public boolean getVerbose() { + return swigfaissJNI.OPQMatrix_verbose_get(swigCPtr, this); + } + + public void setPq(ProductQuantizer value) { + swigfaissJNI.OPQMatrix_pq_set(swigCPtr, this, ProductQuantizer.getCPtr(value), value); + } + + public ProductQuantizer getPq() { + long cPtr = swigfaissJNI.OPQMatrix_pq_get(swigCPtr, this); + return (cPtr == 0) ? null : new ProductQuantizer(cPtr, false); + } + + public OPQMatrix(int d, int M, int d2) { + this(swigfaissJNI.new_OPQMatrix__SWIG_0(d, M, d2), true); + } + + public OPQMatrix(int d, int M) { + this(swigfaissJNI.new_OPQMatrix__SWIG_1(d, M), true); + } + + public OPQMatrix(int d) { + this(swigfaissJNI.new_OPQMatrix__SWIG_2(d), true); + } + + public OPQMatrix() { + this(swigfaissJNI.new_OPQMatrix__SWIG_3(), true); + } + + public void train(long n, SWIGTYPE_p_float x) { + swigfaissJNI.OPQMatrix_train(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x)); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/OnDiskInvertedLists.java b/ann/src/main/java/com/twitter/ann/faiss/swig/OnDiskInvertedLists.java new file mode 100644 index 0000000000..94d13f5229 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/OnDiskInvertedLists.java @@ -0,0 +1,251 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class OnDiskInvertedLists extends InvertedLists { + private transient long swigCPtr; + + protected OnDiskInvertedLists(long cPtr, boolean cMemoryOwn) { + super(swigfaissJNI.OnDiskInvertedLists_SWIGUpcast(cPtr), cMemoryOwn); + swigCPtr = cPtr; + } + + protected static long getCPtr(OnDiskInvertedLists obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_OnDiskInvertedLists(swigCPtr); + } + swigCPtr = 0; + } + super.delete(); + } + + public void setLists(SWIGTYPE_p_std__vectorT_faiss__OnDiskOneList_t value) { + swigfaissJNI.OnDiskInvertedLists_lists_set(swigCPtr, this, SWIGTYPE_p_std__vectorT_faiss__OnDiskOneList_t.getCPtr(value)); + } + + public SWIGTYPE_p_std__vectorT_faiss__OnDiskOneList_t getLists() { + long cPtr = swigfaissJNI.OnDiskInvertedLists_lists_get(swigCPtr, this); + return (cPtr == 0) ? null : new SWIGTYPE_p_std__vectorT_faiss__OnDiskOneList_t(cPtr, false); + } + + static public class Slot { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected Slot(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(Slot obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_OnDiskInvertedLists_Slot(swigCPtr); + } + swigCPtr = 0; + } + } + + public void setOffset(long value) { + swigfaissJNI.OnDiskInvertedLists_Slot_offset_set(swigCPtr, this, value); + } + + public long getOffset() { + return swigfaissJNI.OnDiskInvertedLists_Slot_offset_get(swigCPtr, this); + } + + public void setCapacity(long value) { + swigfaissJNI.OnDiskInvertedLists_Slot_capacity_set(swigCPtr, this, value); + } + + public long getCapacity() { + return swigfaissJNI.OnDiskInvertedLists_Slot_capacity_get(swigCPtr, this); + } + + public Slot(long offset, long capacity) { + this(swigfaissJNI.new_OnDiskInvertedLists_Slot__SWIG_0(offset, capacity), true); + } + + public Slot() { + this(swigfaissJNI.new_OnDiskInvertedLists_Slot__SWIG_1(), true); + } + + } + + public void setSlots(SWIGTYPE_p_std__listT_faiss__OnDiskInvertedLists__Slot_t value) { + swigfaissJNI.OnDiskInvertedLists_slots_set(swigCPtr, this, SWIGTYPE_p_std__listT_faiss__OnDiskInvertedLists__Slot_t.getCPtr(value)); + } + + public SWIGTYPE_p_std__listT_faiss__OnDiskInvertedLists__Slot_t getSlots() { + long cPtr = swigfaissJNI.OnDiskInvertedLists_slots_get(swigCPtr, this); + return (cPtr == 0) ? null : new SWIGTYPE_p_std__listT_faiss__OnDiskInvertedLists__Slot_t(cPtr, false); + } + + public void setFilename(String value) { + swigfaissJNI.OnDiskInvertedLists_filename_set(swigCPtr, this, value); + } + + public String getFilename() { + return swigfaissJNI.OnDiskInvertedLists_filename_get(swigCPtr, this); + } + + public void setTotsize(long value) { + swigfaissJNI.OnDiskInvertedLists_totsize_set(swigCPtr, this, value); + } + + public long getTotsize() { + return swigfaissJNI.OnDiskInvertedLists_totsize_get(swigCPtr, this); + } + + public void setPtr(SWIGTYPE_p_unsigned_char value) { + swigfaissJNI.OnDiskInvertedLists_ptr_set(swigCPtr, this, SWIGTYPE_p_unsigned_char.getCPtr(value)); + } + + public SWIGTYPE_p_unsigned_char getPtr() { + long cPtr = swigfaissJNI.OnDiskInvertedLists_ptr_get(swigCPtr, this); + return (cPtr == 0) ? null : new SWIGTYPE_p_unsigned_char(cPtr, false); + } + + public void setRead_only(boolean value) { + swigfaissJNI.OnDiskInvertedLists_read_only_set(swigCPtr, this, value); + } + + public boolean getRead_only() { + return swigfaissJNI.OnDiskInvertedLists_read_only_get(swigCPtr, this); + } + + public OnDiskInvertedLists(long nlist, long code_size, String filename) { + this(swigfaissJNI.new_OnDiskInvertedLists__SWIG_0(nlist, code_size, filename), true); + } + + public long list_size(long list_no) { + return swigfaissJNI.OnDiskInvertedLists_list_size(swigCPtr, this, list_no); + } + + public SWIGTYPE_p_unsigned_char get_codes(long list_no) { + long cPtr = swigfaissJNI.OnDiskInvertedLists_get_codes(swigCPtr, this, list_no); + return (cPtr == 0) ? null : new SWIGTYPE_p_unsigned_char(cPtr, false); + } + + public LongVector get_ids(long list_no) { + return new LongVector(swigfaissJNI.OnDiskInvertedLists_get_ids(swigCPtr, this, list_no), false); +} + + public long add_entries(long list_no, long n_entry, LongVector ids, SWIGTYPE_p_unsigned_char code) { + return swigfaissJNI.OnDiskInvertedLists_add_entries(swigCPtr, this, list_no, n_entry, SWIGTYPE_p_long_long.getCPtr(ids.data()), ids, SWIGTYPE_p_unsigned_char.getCPtr(code)); + } + + public void update_entries(long list_no, long offset, long n_entry, LongVector ids, SWIGTYPE_p_unsigned_char code) { + swigfaissJNI.OnDiskInvertedLists_update_entries(swigCPtr, this, list_no, offset, n_entry, SWIGTYPE_p_long_long.getCPtr(ids.data()), ids, SWIGTYPE_p_unsigned_char.getCPtr(code)); + } + + public void resize(long list_no, long new_size) { + swigfaissJNI.OnDiskInvertedLists_resize(swigCPtr, this, list_no, new_size); + } + + public long merge_from(SWIGTYPE_p_p_faiss__InvertedLists ils, int n_il, boolean verbose) { + return swigfaissJNI.OnDiskInvertedLists_merge_from__SWIG_0(swigCPtr, this, SWIGTYPE_p_p_faiss__InvertedLists.getCPtr(ils), n_il, verbose); + } + + public long merge_from(SWIGTYPE_p_p_faiss__InvertedLists ils, int n_il) { + return swigfaissJNI.OnDiskInvertedLists_merge_from__SWIG_1(swigCPtr, this, SWIGTYPE_p_p_faiss__InvertedLists.getCPtr(ils), n_il); + } + + public long merge_from_1(InvertedLists il, boolean verbose) { + return swigfaissJNI.OnDiskInvertedLists_merge_from_1__SWIG_0(swigCPtr, this, InvertedLists.getCPtr(il), il, verbose); + } + + public long merge_from_1(InvertedLists il) { + return swigfaissJNI.OnDiskInvertedLists_merge_from_1__SWIG_1(swigCPtr, this, InvertedLists.getCPtr(il), il); + } + + public void crop_invlists(long l0, long l1) { + swigfaissJNI.OnDiskInvertedLists_crop_invlists(swigCPtr, this, l0, l1); + } + + public void prefetch_lists(LongVector list_nos, int nlist) { + swigfaissJNI.OnDiskInvertedLists_prefetch_lists(swigCPtr, this, SWIGTYPE_p_long_long.getCPtr(list_nos.data()), list_nos, nlist); + } + + public void setLocks(SWIGTYPE_p_faiss__LockLevels value) { + swigfaissJNI.OnDiskInvertedLists_locks_set(swigCPtr, this, SWIGTYPE_p_faiss__LockLevels.getCPtr(value)); + } + + public SWIGTYPE_p_faiss__LockLevels getLocks() { + long cPtr = swigfaissJNI.OnDiskInvertedLists_locks_get(swigCPtr, this); + return (cPtr == 0) ? null : new SWIGTYPE_p_faiss__LockLevels(cPtr, false); + } + + public void setPf(SWIGTYPE_p_faiss__OnDiskInvertedLists__OngoingPrefetch value) { + swigfaissJNI.OnDiskInvertedLists_pf_set(swigCPtr, this, SWIGTYPE_p_faiss__OnDiskInvertedLists__OngoingPrefetch.getCPtr(value)); + } + + public SWIGTYPE_p_faiss__OnDiskInvertedLists__OngoingPrefetch getPf() { + long cPtr = swigfaissJNI.OnDiskInvertedLists_pf_get(swigCPtr, this); + return (cPtr == 0) ? null : new SWIGTYPE_p_faiss__OnDiskInvertedLists__OngoingPrefetch(cPtr, false); + } + + public void setPrefetch_nthread(int value) { + swigfaissJNI.OnDiskInvertedLists_prefetch_nthread_set(swigCPtr, this, value); + } + + public int getPrefetch_nthread() { + return swigfaissJNI.OnDiskInvertedLists_prefetch_nthread_get(swigCPtr, this); + } + + public void do_mmap() { + swigfaissJNI.OnDiskInvertedLists_do_mmap(swigCPtr, this); + } + + public void update_totsize(long new_totsize) { + swigfaissJNI.OnDiskInvertedLists_update_totsize(swigCPtr, this, new_totsize); + } + + public void resize_locked(long list_no, long new_size) { + swigfaissJNI.OnDiskInvertedLists_resize_locked(swigCPtr, this, list_no, new_size); + } + + public long allocate_slot(long capacity) { + return swigfaissJNI.OnDiskInvertedLists_allocate_slot(swigCPtr, this, capacity); + } + + public void free_slot(long offset, long capacity) { + swigfaissJNI.OnDiskInvertedLists_free_slot(swigCPtr, this, offset, capacity); + } + + public void set_all_lists_sizes(SWIGTYPE_p_unsigned_long sizes) { + swigfaissJNI.OnDiskInvertedLists_set_all_lists_sizes(swigCPtr, this, SWIGTYPE_p_unsigned_long.getCPtr(sizes)); + } + + public OnDiskInvertedLists() { + this(swigfaissJNI.new_OnDiskInvertedLists__SWIG_1(), true); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/OnDiskInvertedListsIOHook.java b/ann/src/main/java/com/twitter/ann/faiss/swig/OnDiskInvertedListsIOHook.java new file mode 100644 index 0000000000..15a2138eef --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/OnDiskInvertedListsIOHook.java @@ -0,0 +1,57 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class OnDiskInvertedListsIOHook { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected OnDiskInvertedListsIOHook(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(OnDiskInvertedListsIOHook obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_OnDiskInvertedListsIOHook(swigCPtr); + } + swigCPtr = 0; + } + } + + public OnDiskInvertedListsIOHook() { + this(swigfaissJNI.new_OnDiskInvertedListsIOHook(), true); + } + + public void write(InvertedLists ils, SWIGTYPE_p_IOWriter f) { + swigfaissJNI.OnDiskInvertedListsIOHook_write(swigCPtr, this, InvertedLists.getCPtr(ils), ils, SWIGTYPE_p_IOWriter.getCPtr(f)); + } + + public InvertedLists read(SWIGTYPE_p_IOReader f, int io_flags) { + long cPtr = swigfaissJNI.OnDiskInvertedListsIOHook_read(swigCPtr, this, SWIGTYPE_p_IOReader.getCPtr(f), io_flags); + return (cPtr == 0) ? null : new InvertedLists(cPtr, false); + } + + public InvertedLists read_ArrayInvertedLists(SWIGTYPE_p_IOReader f, int io_flags, long nlist, long code_size, Uint64Vector sizes) { + long cPtr = swigfaissJNI.OnDiskInvertedListsIOHook_read_ArrayInvertedLists(swigCPtr, this, SWIGTYPE_p_IOReader.getCPtr(f), io_flags, nlist, code_size, Uint64Vector.getCPtr(sizes), sizes); + return (cPtr == 0) ? null : new InvertedLists(cPtr, false); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/OnDiskOneList.java b/ann/src/main/java/com/twitter/ann/faiss/swig/OnDiskOneList.java new file mode 100644 index 0000000000..acc2cfae75 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/OnDiskOneList.java @@ -0,0 +1,67 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class OnDiskOneList { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected OnDiskOneList(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(OnDiskOneList obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_OnDiskOneList(swigCPtr); + } + swigCPtr = 0; + } + } + + public void setSize(long value) { + swigfaissJNI.OnDiskOneList_size_set(swigCPtr, this, value); + } + + public long getSize() { + return swigfaissJNI.OnDiskOneList_size_get(swigCPtr, this); + } + + public void setCapacity(long value) { + swigfaissJNI.OnDiskOneList_capacity_set(swigCPtr, this, value); + } + + public long getCapacity() { + return swigfaissJNI.OnDiskOneList_capacity_get(swigCPtr, this); + } + + public void setOffset(long value) { + swigfaissJNI.OnDiskOneList_offset_set(swigCPtr, this, value); + } + + public long getOffset() { + return swigfaissJNI.OnDiskOneList_offset_get(swigCPtr, this); + } + + public OnDiskOneList() { + this(swigfaissJNI.new_OnDiskOneList(), true); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/OneRecallAtRCriterion.java b/ann/src/main/java/com/twitter/ann/faiss/swig/OneRecallAtRCriterion.java new file mode 100644 index 0000000000..f572142055 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/OneRecallAtRCriterion.java @@ -0,0 +1,55 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class OneRecallAtRCriterion extends AutoTuneCriterion { + private transient long swigCPtr; + + protected OneRecallAtRCriterion(long cPtr, boolean cMemoryOwn) { + super(swigfaissJNI.OneRecallAtRCriterion_SWIGUpcast(cPtr), cMemoryOwn); + swigCPtr = cPtr; + } + + protected static long getCPtr(OneRecallAtRCriterion obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_OneRecallAtRCriterion(swigCPtr); + } + swigCPtr = 0; + } + super.delete(); + } + + public void setR(long value) { + swigfaissJNI.OneRecallAtRCriterion_R_set(swigCPtr, this, value); + } + + public long getR() { + return swigfaissJNI.OneRecallAtRCriterion_R_get(swigCPtr, this); +} + + public OneRecallAtRCriterion(long nq, long R) { + this(swigfaissJNI.new_OneRecallAtRCriterion(nq, R), true); + } + + public double evaluate(SWIGTYPE_p_float D, LongVector I) { + return swigfaissJNI.OneRecallAtRCriterion_evaluate(swigCPtr, this, SWIGTYPE_p_float.getCPtr(D), SWIGTYPE_p_long_long.getCPtr(I.data()), I); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/OperatingPoint.java b/ann/src/main/java/com/twitter/ann/faiss/swig/OperatingPoint.java new file mode 100644 index 0000000000..ed0f71f5da --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/OperatingPoint.java @@ -0,0 +1,75 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class OperatingPoint { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected OperatingPoint(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(OperatingPoint obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_OperatingPoint(swigCPtr); + } + swigCPtr = 0; + } + } + + public void setPerf(double value) { + swigfaissJNI.OperatingPoint_perf_set(swigCPtr, this, value); + } + + public double getPerf() { + return swigfaissJNI.OperatingPoint_perf_get(swigCPtr, this); + } + + public void setT(double value) { + swigfaissJNI.OperatingPoint_t_set(swigCPtr, this, value); + } + + public double getT() { + return swigfaissJNI.OperatingPoint_t_get(swigCPtr, this); + } + + public void setKey(String value) { + swigfaissJNI.OperatingPoint_key_set(swigCPtr, this, value); + } + + public String getKey() { + return swigfaissJNI.OperatingPoint_key_get(swigCPtr, this); + } + + public void setCno(long value) { + swigfaissJNI.OperatingPoint_cno_set(swigCPtr, this, value); + } + + public long getCno() { + return swigfaissJNI.OperatingPoint_cno_get(swigCPtr, this); +} + + public OperatingPoint() { + this(swigfaissJNI.new_OperatingPoint(), true); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/OperatingPointVector.java b/ann/src/main/java/com/twitter/ann/faiss/swig/OperatingPointVector.java new file mode 100644 index 0000000000..42fab072b1 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/OperatingPointVector.java @@ -0,0 +1,76 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class OperatingPointVector { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected OperatingPointVector(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(OperatingPointVector obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_OperatingPointVector(swigCPtr); + } + swigCPtr = 0; + } + } + + public OperatingPointVector() { + this(swigfaissJNI.new_OperatingPointVector(), true); + } + + public void push_back(OperatingPoint arg0) { + swigfaissJNI.OperatingPointVector_push_back(swigCPtr, this, OperatingPoint.getCPtr(arg0), arg0); + } + + public void clear() { + swigfaissJNI.OperatingPointVector_clear(swigCPtr, this); + } + + public OperatingPoint data() { + long cPtr = swigfaissJNI.OperatingPointVector_data(swigCPtr, this); + return (cPtr == 0) ? null : new OperatingPoint(cPtr, false); + } + + public long size() { + return swigfaissJNI.OperatingPointVector_size(swigCPtr, this); + } + + public OperatingPoint at(long n) { + return new OperatingPoint(swigfaissJNI.OperatingPointVector_at(swigCPtr, this, n), true); + } + + public void resize(long n) { + swigfaissJNI.OperatingPointVector_resize(swigCPtr, this, n); + } + + public void reserve(long n) { + swigfaissJNI.OperatingPointVector_reserve(swigCPtr, this, n); + } + + public void swap(OperatingPointVector other) { + swigfaissJNI.OperatingPointVector_swap(swigCPtr, this, OperatingPointVector.getCPtr(other), other); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/OperatingPoints.java b/ann/src/main/java/com/twitter/ann/faiss/swig/OperatingPoints.java new file mode 100644 index 0000000000..036e6730de --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/OperatingPoints.java @@ -0,0 +1,101 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class OperatingPoints { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected OperatingPoints(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(OperatingPoints obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_OperatingPoints(swigCPtr); + } + swigCPtr = 0; + } + } + + public void setAll_pts(OperatingPointVector value) { + swigfaissJNI.OperatingPoints_all_pts_set(swigCPtr, this, OperatingPointVector.getCPtr(value), value); + } + + public OperatingPointVector getAll_pts() { + long cPtr = swigfaissJNI.OperatingPoints_all_pts_get(swigCPtr, this); + return (cPtr == 0) ? null : new OperatingPointVector(cPtr, false); + } + + public void setOptimal_pts(OperatingPointVector value) { + swigfaissJNI.OperatingPoints_optimal_pts_set(swigCPtr, this, OperatingPointVector.getCPtr(value), value); + } + + public OperatingPointVector getOptimal_pts() { + long cPtr = swigfaissJNI.OperatingPoints_optimal_pts_get(swigCPtr, this); + return (cPtr == 0) ? null : new OperatingPointVector(cPtr, false); + } + + public OperatingPoints() { + this(swigfaissJNI.new_OperatingPoints(), true); + } + + public int merge_with(OperatingPoints other, String prefix) { + return swigfaissJNI.OperatingPoints_merge_with__SWIG_0(swigCPtr, this, OperatingPoints.getCPtr(other), other, prefix); + } + + public int merge_with(OperatingPoints other) { + return swigfaissJNI.OperatingPoints_merge_with__SWIG_1(swigCPtr, this, OperatingPoints.getCPtr(other), other); + } + + public void clear() { + swigfaissJNI.OperatingPoints_clear(swigCPtr, this); + } + + public boolean add(double perf, double t, String key, long cno) { + return swigfaissJNI.OperatingPoints_add__SWIG_0(swigCPtr, this, perf, t, key, cno); + } + + public boolean add(double perf, double t, String key) { + return swigfaissJNI.OperatingPoints_add__SWIG_1(swigCPtr, this, perf, t, key); + } + + public double t_for_perf(double perf) { + return swigfaissJNI.OperatingPoints_t_for_perf(swigCPtr, this, perf); + } + + public void display(boolean only_optimal) { + swigfaissJNI.OperatingPoints_display__SWIG_0(swigCPtr, this, only_optimal); + } + + public void display() { + swigfaissJNI.OperatingPoints_display__SWIG_1(swigCPtr, this); + } + + public void all_to_gnuplot(String fname) { + swigfaissJNI.OperatingPoints_all_to_gnuplot(swigCPtr, this, fname); + } + + public void optimal_to_gnuplot(String fname) { + swigfaissJNI.OperatingPoints_optimal_to_gnuplot(swigCPtr, this, fname); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/PCAMatrix.java b/ann/src/main/java/com/twitter/ann/faiss/swig/PCAMatrix.java new file mode 100644 index 0000000000..caa7864a0b --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/PCAMatrix.java @@ -0,0 +1,138 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class PCAMatrix extends LinearTransform { + private transient long swigCPtr; + + protected PCAMatrix(long cPtr, boolean cMemoryOwn) { + super(swigfaissJNI.PCAMatrix_SWIGUpcast(cPtr), cMemoryOwn); + swigCPtr = cPtr; + } + + protected static long getCPtr(PCAMatrix obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_PCAMatrix(swigCPtr); + } + swigCPtr = 0; + } + super.delete(); + } + + public void setEigen_power(float value) { + swigfaissJNI.PCAMatrix_eigen_power_set(swigCPtr, this, value); + } + + public float getEigen_power() { + return swigfaissJNI.PCAMatrix_eigen_power_get(swigCPtr, this); + } + + public void setEpsilon(float value) { + swigfaissJNI.PCAMatrix_epsilon_set(swigCPtr, this, value); + } + + public float getEpsilon() { + return swigfaissJNI.PCAMatrix_epsilon_get(swigCPtr, this); + } + + public void setRandom_rotation(boolean value) { + swigfaissJNI.PCAMatrix_random_rotation_set(swigCPtr, this, value); + } + + public boolean getRandom_rotation() { + return swigfaissJNI.PCAMatrix_random_rotation_get(swigCPtr, this); + } + + public void setMax_points_per_d(long value) { + swigfaissJNI.PCAMatrix_max_points_per_d_set(swigCPtr, this, value); + } + + public long getMax_points_per_d() { + return swigfaissJNI.PCAMatrix_max_points_per_d_get(swigCPtr, this); + } + + public void setBalanced_bins(int value) { + swigfaissJNI.PCAMatrix_balanced_bins_set(swigCPtr, this, value); + } + + public int getBalanced_bins() { + return swigfaissJNI.PCAMatrix_balanced_bins_get(swigCPtr, this); + } + + public void setMean(FloatVector value) { + swigfaissJNI.PCAMatrix_mean_set(swigCPtr, this, FloatVector.getCPtr(value), value); + } + + public FloatVector getMean() { + long cPtr = swigfaissJNI.PCAMatrix_mean_get(swigCPtr, this); + return (cPtr == 0) ? null : new FloatVector(cPtr, false); + } + + public void setEigenvalues(FloatVector value) { + swigfaissJNI.PCAMatrix_eigenvalues_set(swigCPtr, this, FloatVector.getCPtr(value), value); + } + + public FloatVector getEigenvalues() { + long cPtr = swigfaissJNI.PCAMatrix_eigenvalues_get(swigCPtr, this); + return (cPtr == 0) ? null : new FloatVector(cPtr, false); + } + + public void setPCAMat(FloatVector value) { + swigfaissJNI.PCAMatrix_PCAMat_set(swigCPtr, this, FloatVector.getCPtr(value), value); + } + + public FloatVector getPCAMat() { + long cPtr = swigfaissJNI.PCAMatrix_PCAMat_get(swigCPtr, this); + return (cPtr == 0) ? null : new FloatVector(cPtr, false); + } + + public PCAMatrix(int d_in, int d_out, float eigen_power, boolean random_rotation) { + this(swigfaissJNI.new_PCAMatrix__SWIG_0(d_in, d_out, eigen_power, random_rotation), true); + } + + public PCAMatrix(int d_in, int d_out, float eigen_power) { + this(swigfaissJNI.new_PCAMatrix__SWIG_1(d_in, d_out, eigen_power), true); + } + + public PCAMatrix(int d_in, int d_out) { + this(swigfaissJNI.new_PCAMatrix__SWIG_2(d_in, d_out), true); + } + + public PCAMatrix(int d_in) { + this(swigfaissJNI.new_PCAMatrix__SWIG_3(d_in), true); + } + + public PCAMatrix() { + this(swigfaissJNI.new_PCAMatrix__SWIG_4(), true); + } + + public void train(long n, SWIGTYPE_p_float x) { + swigfaissJNI.PCAMatrix_train(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x)); + } + + public void copy_from(PCAMatrix other) { + swigfaissJNI.PCAMatrix_copy_from(swigCPtr, this, PCAMatrix.getCPtr(other), other); + } + + public void prepare_Ab() { + swigfaissJNI.PCAMatrix_prepare_Ab(swigCPtr, this); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/PQDecoder16.java b/ann/src/main/java/com/twitter/ann/faiss/swig/PQDecoder16.java new file mode 100644 index 0000000000..f40c708145 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/PQDecoder16.java @@ -0,0 +1,57 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class PQDecoder16 { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected PQDecoder16(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(PQDecoder16 obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_PQDecoder16(swigCPtr); + } + swigCPtr = 0; + } + } + + public void setCode(SWIGTYPE_p_uint16_t value) { + swigfaissJNI.PQDecoder16_code_set(swigCPtr, this, SWIGTYPE_p_uint16_t.getCPtr(value)); + } + + public SWIGTYPE_p_uint16_t getCode() { + long cPtr = swigfaissJNI.PQDecoder16_code_get(swigCPtr, this); + return (cPtr == 0) ? null : new SWIGTYPE_p_uint16_t(cPtr, false); + } + + public PQDecoder16(SWIGTYPE_p_unsigned_char code, int nbits) { + this(swigfaissJNI.new_PQDecoder16(SWIGTYPE_p_unsigned_char.getCPtr(code), nbits), true); + } + + public long decode() { + return swigfaissJNI.PQDecoder16_decode(swigCPtr, this); + } + + public final static int nbits = swigfaissJNI.PQDecoder16_nbits_get(); +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/PQDecoder8.java b/ann/src/main/java/com/twitter/ann/faiss/swig/PQDecoder8.java new file mode 100644 index 0000000000..ee384cdafa --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/PQDecoder8.java @@ -0,0 +1,57 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class PQDecoder8 { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected PQDecoder8(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(PQDecoder8 obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_PQDecoder8(swigCPtr); + } + swigCPtr = 0; + } + } + + public void setCode(SWIGTYPE_p_unsigned_char value) { + swigfaissJNI.PQDecoder8_code_set(swigCPtr, this, SWIGTYPE_p_unsigned_char.getCPtr(value)); + } + + public SWIGTYPE_p_unsigned_char getCode() { + long cPtr = swigfaissJNI.PQDecoder8_code_get(swigCPtr, this); + return (cPtr == 0) ? null : new SWIGTYPE_p_unsigned_char(cPtr, false); + } + + public PQDecoder8(SWIGTYPE_p_unsigned_char code, int nbits) { + this(swigfaissJNI.new_PQDecoder8(SWIGTYPE_p_unsigned_char.getCPtr(code), nbits), true); + } + + public long decode() { + return swigfaissJNI.PQDecoder8_decode(swigCPtr, this); + } + + public final static int nbits = swigfaissJNI.PQDecoder8_nbits_get(); +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/PQDecoderGeneric.java b/ann/src/main/java/com/twitter/ann/faiss/swig/PQDecoderGeneric.java new file mode 100644 index 0000000000..487b48a0f8 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/PQDecoderGeneric.java @@ -0,0 +1,80 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class PQDecoderGeneric { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected PQDecoderGeneric(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(PQDecoderGeneric obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_PQDecoderGeneric(swigCPtr); + } + swigCPtr = 0; + } + } + + public void setCode(SWIGTYPE_p_unsigned_char value) { + swigfaissJNI.PQDecoderGeneric_code_set(swigCPtr, this, SWIGTYPE_p_unsigned_char.getCPtr(value)); + } + + public SWIGTYPE_p_unsigned_char getCode() { + long cPtr = swigfaissJNI.PQDecoderGeneric_code_get(swigCPtr, this); + return (cPtr == 0) ? null : new SWIGTYPE_p_unsigned_char(cPtr, false); + } + + public void setOffset(short value) { + swigfaissJNI.PQDecoderGeneric_offset_set(swigCPtr, this, value); + } + + public short getOffset() { + return swigfaissJNI.PQDecoderGeneric_offset_get(swigCPtr, this); + } + + public int getNbits() { + return swigfaissJNI.PQDecoderGeneric_nbits_get(swigCPtr, this); + } + + public long getMask() { + return swigfaissJNI.PQDecoderGeneric_mask_get(swigCPtr, this); + } + + public void setReg(short value) { + swigfaissJNI.PQDecoderGeneric_reg_set(swigCPtr, this, value); + } + + public short getReg() { + return swigfaissJNI.PQDecoderGeneric_reg_get(swigCPtr, this); + } + + public PQDecoderGeneric(SWIGTYPE_p_unsigned_char code, int nbits) { + this(swigfaissJNI.new_PQDecoderGeneric(SWIGTYPE_p_unsigned_char.getCPtr(code), nbits), true); + } + + public long decode() { + return swigfaissJNI.PQDecoderGeneric_decode(swigCPtr, this); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/PQEncoder16.java b/ann/src/main/java/com/twitter/ann/faiss/swig/PQEncoder16.java new file mode 100644 index 0000000000..56727e7728 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/PQEncoder16.java @@ -0,0 +1,56 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class PQEncoder16 { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected PQEncoder16(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(PQEncoder16 obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_PQEncoder16(swigCPtr); + } + swigCPtr = 0; + } + } + + public void setCode(SWIGTYPE_p_uint16_t value) { + swigfaissJNI.PQEncoder16_code_set(swigCPtr, this, SWIGTYPE_p_uint16_t.getCPtr(value)); + } + + public SWIGTYPE_p_uint16_t getCode() { + long cPtr = swigfaissJNI.PQEncoder16_code_get(swigCPtr, this); + return (cPtr == 0) ? null : new SWIGTYPE_p_uint16_t(cPtr, false); + } + + public PQEncoder16(SWIGTYPE_p_unsigned_char code, int nbits) { + this(swigfaissJNI.new_PQEncoder16(SWIGTYPE_p_unsigned_char.getCPtr(code), nbits), true); + } + + public void encode(long x) { + swigfaissJNI.PQEncoder16_encode(swigCPtr, this, x); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/PQEncoder8.java b/ann/src/main/java/com/twitter/ann/faiss/swig/PQEncoder8.java new file mode 100644 index 0000000000..fdb0fac47b --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/PQEncoder8.java @@ -0,0 +1,56 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class PQEncoder8 { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected PQEncoder8(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(PQEncoder8 obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_PQEncoder8(swigCPtr); + } + swigCPtr = 0; + } + } + + public void setCode(SWIGTYPE_p_unsigned_char value) { + swigfaissJNI.PQEncoder8_code_set(swigCPtr, this, SWIGTYPE_p_unsigned_char.getCPtr(value)); + } + + public SWIGTYPE_p_unsigned_char getCode() { + long cPtr = swigfaissJNI.PQEncoder8_code_get(swigCPtr, this); + return (cPtr == 0) ? null : new SWIGTYPE_p_unsigned_char(cPtr, false); + } + + public PQEncoder8(SWIGTYPE_p_unsigned_char code, int nbits) { + this(swigfaissJNI.new_PQEncoder8(SWIGTYPE_p_unsigned_char.getCPtr(code), nbits), true); + } + + public void encode(long x) { + swigfaissJNI.PQEncoder8_encode(swigCPtr, this, x); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/PQEncoderGeneric.java b/ann/src/main/java/com/twitter/ann/faiss/swig/PQEncoderGeneric.java new file mode 100644 index 0000000000..682f526cf2 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/PQEncoderGeneric.java @@ -0,0 +1,80 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class PQEncoderGeneric { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected PQEncoderGeneric(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(PQEncoderGeneric obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_PQEncoderGeneric(swigCPtr); + } + swigCPtr = 0; + } + } + + public void setCode(SWIGTYPE_p_unsigned_char value) { + swigfaissJNI.PQEncoderGeneric_code_set(swigCPtr, this, SWIGTYPE_p_unsigned_char.getCPtr(value)); + } + + public SWIGTYPE_p_unsigned_char getCode() { + long cPtr = swigfaissJNI.PQEncoderGeneric_code_get(swigCPtr, this); + return (cPtr == 0) ? null : new SWIGTYPE_p_unsigned_char(cPtr, false); + } + + public void setOffset(short value) { + swigfaissJNI.PQEncoderGeneric_offset_set(swigCPtr, this, value); + } + + public short getOffset() { + return swigfaissJNI.PQEncoderGeneric_offset_get(swigCPtr, this); + } + + public int getNbits() { + return swigfaissJNI.PQEncoderGeneric_nbits_get(swigCPtr, this); + } + + public void setReg(short value) { + swigfaissJNI.PQEncoderGeneric_reg_set(swigCPtr, this, value); + } + + public short getReg() { + return swigfaissJNI.PQEncoderGeneric_reg_get(swigCPtr, this); + } + + public PQEncoderGeneric(SWIGTYPE_p_unsigned_char code, int nbits, short offset) { + this(swigfaissJNI.new_PQEncoderGeneric__SWIG_0(SWIGTYPE_p_unsigned_char.getCPtr(code), nbits, offset), true); + } + + public PQEncoderGeneric(SWIGTYPE_p_unsigned_char code, int nbits) { + this(swigfaissJNI.new_PQEncoderGeneric__SWIG_1(SWIGTYPE_p_unsigned_char.getCPtr(code), nbits), true); + } + + public void encode(long x) { + swigfaissJNI.PQEncoderGeneric_encode(swigCPtr, this, x); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/ParameterRange.java b/ann/src/main/java/com/twitter/ann/faiss/swig/ParameterRange.java new file mode 100644 index 0000000000..4f6643daeb --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/ParameterRange.java @@ -0,0 +1,60 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class ParameterRange { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected ParameterRange(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(ParameterRange obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_ParameterRange(swigCPtr); + } + swigCPtr = 0; + } + } + + public void setName(String value) { + swigfaissJNI.ParameterRange_name_set(swigCPtr, this, value); + } + + public String getName() { + return swigfaissJNI.ParameterRange_name_get(swigCPtr, this); + } + + public void setValues(DoubleVector value) { + swigfaissJNI.ParameterRange_values_set(swigCPtr, this, DoubleVector.getCPtr(value), value); + } + + public DoubleVector getValues() { + long cPtr = swigfaissJNI.ParameterRange_values_get(swigCPtr, this); + return (cPtr == 0) ? null : new DoubleVector(cPtr, false); + } + + public ParameterRange() { + this(swigfaissJNI.new_ParameterRange(), true); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/ParameterSpace.java b/ann/src/main/java/com/twitter/ann/faiss/swig/ParameterSpace.java new file mode 100644 index 0000000000..210509c385 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/ParameterSpace.java @@ -0,0 +1,136 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class ParameterSpace { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected ParameterSpace(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(ParameterSpace obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_ParameterSpace(swigCPtr); + } + swigCPtr = 0; + } + } + + public void setParameter_ranges(SWIGTYPE_p_std__vectorT_faiss__ParameterRange_t value) { + swigfaissJNI.ParameterSpace_parameter_ranges_set(swigCPtr, this, SWIGTYPE_p_std__vectorT_faiss__ParameterRange_t.getCPtr(value)); + } + + public SWIGTYPE_p_std__vectorT_faiss__ParameterRange_t getParameter_ranges() { + long cPtr = swigfaissJNI.ParameterSpace_parameter_ranges_get(swigCPtr, this); + return (cPtr == 0) ? null : new SWIGTYPE_p_std__vectorT_faiss__ParameterRange_t(cPtr, false); + } + + public void setVerbose(int value) { + swigfaissJNI.ParameterSpace_verbose_set(swigCPtr, this, value); + } + + public int getVerbose() { + return swigfaissJNI.ParameterSpace_verbose_get(swigCPtr, this); + } + + public void setN_experiments(int value) { + swigfaissJNI.ParameterSpace_n_experiments_set(swigCPtr, this, value); + } + + public int getN_experiments() { + return swigfaissJNI.ParameterSpace_n_experiments_get(swigCPtr, this); + } + + public void setBatchsize(long value) { + swigfaissJNI.ParameterSpace_batchsize_set(swigCPtr, this, value); + } + + public long getBatchsize() { + return swigfaissJNI.ParameterSpace_batchsize_get(swigCPtr, this); + } + + public void setThread_over_batches(boolean value) { + swigfaissJNI.ParameterSpace_thread_over_batches_set(swigCPtr, this, value); + } + + public boolean getThread_over_batches() { + return swigfaissJNI.ParameterSpace_thread_over_batches_get(swigCPtr, this); + } + + public void setMin_test_duration(double value) { + swigfaissJNI.ParameterSpace_min_test_duration_set(swigCPtr, this, value); + } + + public double getMin_test_duration() { + return swigfaissJNI.ParameterSpace_min_test_duration_get(swigCPtr, this); + } + + public ParameterSpace() { + this(swigfaissJNI.new_ParameterSpace(), true); + } + + public long n_combinations() { + return swigfaissJNI.ParameterSpace_n_combinations(swigCPtr, this); + } + + public boolean combination_ge(long c1, long c2) { + return swigfaissJNI.ParameterSpace_combination_ge(swigCPtr, this, c1, c2); + } + + public String combination_name(long cno) { + return swigfaissJNI.ParameterSpace_combination_name(swigCPtr, this, cno); + } + + public void display() { + swigfaissJNI.ParameterSpace_display(swigCPtr, this); + } + + public ParameterRange add_range(String name) { + return new ParameterRange(swigfaissJNI.ParameterSpace_add_range(swigCPtr, this, name), false); + } + + public void initialize(Index index) { + swigfaissJNI.ParameterSpace_initialize(swigCPtr, this, Index.getCPtr(index), index); + } + + public void set_index_parameters(Index index, long cno) { + swigfaissJNI.ParameterSpace_set_index_parameters__SWIG_0(swigCPtr, this, Index.getCPtr(index), index, cno); + } + + public void set_index_parameters(Index index, String param_string) { + swigfaissJNI.ParameterSpace_set_index_parameters__SWIG_1(swigCPtr, this, Index.getCPtr(index), index, param_string); + } + + public void set_index_parameter(Index index, String name, double val) { + swigfaissJNI.ParameterSpace_set_index_parameter(swigCPtr, this, Index.getCPtr(index), index, name, val); + } + + public void update_bounds(long cno, OperatingPoint op, SWIGTYPE_p_double upper_bound_perf, SWIGTYPE_p_double lower_bound_t) { + swigfaissJNI.ParameterSpace_update_bounds(swigCPtr, this, cno, OperatingPoint.getCPtr(op), op, SWIGTYPE_p_double.getCPtr(upper_bound_perf), SWIGTYPE_p_double.getCPtr(lower_bound_t)); + } + + public void explore(Index index, long nq, SWIGTYPE_p_float xq, AutoTuneCriterion crit, OperatingPoints ops) { + swigfaissJNI.ParameterSpace_explore(swigCPtr, this, Index.getCPtr(index), index, nq, SWIGTYPE_p_float.getCPtr(xq), AutoTuneCriterion.getCPtr(crit), crit, OperatingPoints.getCPtr(ops), ops); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/PartitionStats.java b/ann/src/main/java/com/twitter/ann/faiss/swig/PartitionStats.java new file mode 100644 index 0000000000..ea9038771e --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/PartitionStats.java @@ -0,0 +1,63 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class PartitionStats { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected PartitionStats(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(PartitionStats obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_PartitionStats(swigCPtr); + } + swigCPtr = 0; + } + } + + public void setBissect_cycles(long value) { + swigfaissJNI.PartitionStats_bissect_cycles_set(swigCPtr, this, value); + } + + public long getBissect_cycles() { + return swigfaissJNI.PartitionStats_bissect_cycles_get(swigCPtr, this); + } + + public void setCompress_cycles(long value) { + swigfaissJNI.PartitionStats_compress_cycles_set(swigCPtr, this, value); + } + + public long getCompress_cycles() { + return swigfaissJNI.PartitionStats_compress_cycles_get(swigCPtr, this); + } + + public PartitionStats() { + this(swigfaissJNI.new_PartitionStats(), true); + } + + public void reset() { + swigfaissJNI.PartitionStats_reset(swigCPtr, this); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/PermutationObjective.java b/ann/src/main/java/com/twitter/ann/faiss/swig/PermutationObjective.java new file mode 100644 index 0000000000..5a6d0b54c6 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/PermutationObjective.java @@ -0,0 +1,55 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class PermutationObjective { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected PermutationObjective(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(PermutationObjective obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_PermutationObjective(swigCPtr); + } + swigCPtr = 0; + } + } + + public void setN(int value) { + swigfaissJNI.PermutationObjective_n_set(swigCPtr, this, value); + } + + public int getN() { + return swigfaissJNI.PermutationObjective_n_get(swigCPtr, this); + } + + public double compute_cost(SWIGTYPE_p_int perm) { + return swigfaissJNI.PermutationObjective_compute_cost(swigCPtr, this, SWIGTYPE_p_int.getCPtr(perm)); + } + + public double cost_update(SWIGTYPE_p_int perm, int iw, int jw) { + return swigfaissJNI.PermutationObjective_cost_update(swigCPtr, this, SWIGTYPE_p_int.getCPtr(perm), iw, jw); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/PolysemousTraining.java b/ann/src/main/java/com/twitter/ann/faiss/swig/PolysemousTraining.java new file mode 100644 index 0000000000..d65dcfeb8e --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/PolysemousTraining.java @@ -0,0 +1,144 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class PolysemousTraining extends SimulatedAnnealingParameters { + private transient long swigCPtr; + + protected PolysemousTraining(long cPtr, boolean cMemoryOwn) { + super(swigfaissJNI.PolysemousTraining_SWIGUpcast(cPtr), cMemoryOwn); + swigCPtr = cPtr; + } + + protected static long getCPtr(PolysemousTraining obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_PolysemousTraining(swigCPtr); + } + swigCPtr = 0; + } + super.delete(); + } + + public void setOptimization_type(PolysemousTraining.Optimization_type_t value) { + swigfaissJNI.PolysemousTraining_optimization_type_set(swigCPtr, this, value.swigValue()); + } + + public PolysemousTraining.Optimization_type_t getOptimization_type() { + return PolysemousTraining.Optimization_type_t.swigToEnum(swigfaissJNI.PolysemousTraining_optimization_type_get(swigCPtr, this)); + } + + public void setNtrain_permutation(int value) { + swigfaissJNI.PolysemousTraining_ntrain_permutation_set(swigCPtr, this, value); + } + + public int getNtrain_permutation() { + return swigfaissJNI.PolysemousTraining_ntrain_permutation_get(swigCPtr, this); + } + + public void setDis_weight_factor(double value) { + swigfaissJNI.PolysemousTraining_dis_weight_factor_set(swigCPtr, this, value); + } + + public double getDis_weight_factor() { + return swigfaissJNI.PolysemousTraining_dis_weight_factor_get(swigCPtr, this); + } + + public void setMax_memory(long value) { + swigfaissJNI.PolysemousTraining_max_memory_set(swigCPtr, this, value); + } + + public long getMax_memory() { + return swigfaissJNI.PolysemousTraining_max_memory_get(swigCPtr, this); + } + + public void setLog_pattern(String value) { + swigfaissJNI.PolysemousTraining_log_pattern_set(swigCPtr, this, value); + } + + public String getLog_pattern() { + return swigfaissJNI.PolysemousTraining_log_pattern_get(swigCPtr, this); + } + + public PolysemousTraining() { + this(swigfaissJNI.new_PolysemousTraining(), true); + } + + public void optimize_pq_for_hamming(ProductQuantizer pq, long n, SWIGTYPE_p_float x) { + swigfaissJNI.PolysemousTraining_optimize_pq_for_hamming(swigCPtr, this, ProductQuantizer.getCPtr(pq), pq, n, SWIGTYPE_p_float.getCPtr(x)); + } + + public void optimize_ranking(ProductQuantizer pq, long n, SWIGTYPE_p_float x) { + swigfaissJNI.PolysemousTraining_optimize_ranking(swigCPtr, this, ProductQuantizer.getCPtr(pq), pq, n, SWIGTYPE_p_float.getCPtr(x)); + } + + public void optimize_reproduce_distances(ProductQuantizer pq) { + swigfaissJNI.PolysemousTraining_optimize_reproduce_distances(swigCPtr, this, ProductQuantizer.getCPtr(pq), pq); + } + + public long memory_usage_per_thread(ProductQuantizer pq) { + return swigfaissJNI.PolysemousTraining_memory_usage_per_thread(swigCPtr, this, ProductQuantizer.getCPtr(pq), pq); + } + + public final static class Optimization_type_t { + public final static PolysemousTraining.Optimization_type_t OT_None = new PolysemousTraining.Optimization_type_t("OT_None"); + public final static PolysemousTraining.Optimization_type_t OT_ReproduceDistances_affine = new PolysemousTraining.Optimization_type_t("OT_ReproduceDistances_affine"); + public final static PolysemousTraining.Optimization_type_t OT_Ranking_weighted_diff = new PolysemousTraining.Optimization_type_t("OT_Ranking_weighted_diff"); + + public final int swigValue() { + return swigValue; + } + + public String toString() { + return swigName; + } + + public static Optimization_type_t swigToEnum(int swigValue) { + if (swigValue < swigValues.length && swigValue >= 0 && swigValues[swigValue].swigValue == swigValue) + return swigValues[swigValue]; + for (int i = 0; i < swigValues.length; i++) + if (swigValues[i].swigValue == swigValue) + return swigValues[i]; + throw new IllegalArgumentException("No enum " + Optimization_type_t.class + " with value " + swigValue); + } + + private Optimization_type_t(String swigName) { + this.swigName = swigName; + this.swigValue = swigNext++; + } + + private Optimization_type_t(String swigName, int swigValue) { + this.swigName = swigName; + this.swigValue = swigValue; + swigNext = swigValue+1; + } + + private Optimization_type_t(String swigName, Optimization_type_t swigEnum) { + this.swigName = swigName; + this.swigValue = swigEnum.swigValue; + swigNext = this.swigValue+1; + } + + private static Optimization_type_t[] swigValues = { OT_None, OT_ReproduceDistances_affine, OT_Ranking_weighted_diff }; + private static int swigNext = 0; + private final int swigValue; + private final String swigName; + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/ProductQuantizer.java b/ann/src/main/java/com/twitter/ann/faiss/swig/ProductQuantizer.java new file mode 100644 index 0000000000..0249e29e33 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/ProductQuantizer.java @@ -0,0 +1,279 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class ProductQuantizer { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected ProductQuantizer(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(ProductQuantizer obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_ProductQuantizer(swigCPtr); + } + swigCPtr = 0; + } + } + + public void setD(long value) { + swigfaissJNI.ProductQuantizer_d_set(swigCPtr, this, value); + } + + public long getD() { + return swigfaissJNI.ProductQuantizer_d_get(swigCPtr, this); + } + + public void setM(long value) { + swigfaissJNI.ProductQuantizer_M_set(swigCPtr, this, value); + } + + public long getM() { + return swigfaissJNI.ProductQuantizer_M_get(swigCPtr, this); + } + + public void setNbits(long value) { + swigfaissJNI.ProductQuantizer_nbits_set(swigCPtr, this, value); + } + + public long getNbits() { + return swigfaissJNI.ProductQuantizer_nbits_get(swigCPtr, this); + } + + public void setDsub(long value) { + swigfaissJNI.ProductQuantizer_dsub_set(swigCPtr, this, value); + } + + public long getDsub() { + return swigfaissJNI.ProductQuantizer_dsub_get(swigCPtr, this); + } + + public void setCode_size(long value) { + swigfaissJNI.ProductQuantizer_code_size_set(swigCPtr, this, value); + } + + public long getCode_size() { + return swigfaissJNI.ProductQuantizer_code_size_get(swigCPtr, this); + } + + public void setKsub(long value) { + swigfaissJNI.ProductQuantizer_ksub_set(swigCPtr, this, value); + } + + public long getKsub() { + return swigfaissJNI.ProductQuantizer_ksub_get(swigCPtr, this); + } + + public void setVerbose(boolean value) { + swigfaissJNI.ProductQuantizer_verbose_set(swigCPtr, this, value); + } + + public boolean getVerbose() { + return swigfaissJNI.ProductQuantizer_verbose_get(swigCPtr, this); + } + + public void setTrain_type(ProductQuantizer.train_type_t value) { + swigfaissJNI.ProductQuantizer_train_type_set(swigCPtr, this, value.swigValue()); + } + + public ProductQuantizer.train_type_t getTrain_type() { + return ProductQuantizer.train_type_t.swigToEnum(swigfaissJNI.ProductQuantizer_train_type_get(swigCPtr, this)); + } + + public void setCp(ClusteringParameters value) { + swigfaissJNI.ProductQuantizer_cp_set(swigCPtr, this, ClusteringParameters.getCPtr(value), value); + } + + public ClusteringParameters getCp() { + long cPtr = swigfaissJNI.ProductQuantizer_cp_get(swigCPtr, this); + return (cPtr == 0) ? null : new ClusteringParameters(cPtr, false); + } + + public void setAssign_index(Index value) { + swigfaissJNI.ProductQuantizer_assign_index_set(swigCPtr, this, Index.getCPtr(value), value); + } + + public Index getAssign_index() { + long cPtr = swigfaissJNI.ProductQuantizer_assign_index_get(swigCPtr, this); + return (cPtr == 0) ? null : new Index(cPtr, false); + } + + public void setCentroids(FloatVector value) { + swigfaissJNI.ProductQuantizer_centroids_set(swigCPtr, this, FloatVector.getCPtr(value), value); + } + + public FloatVector getCentroids() { + long cPtr = swigfaissJNI.ProductQuantizer_centroids_get(swigCPtr, this); + return (cPtr == 0) ? null : new FloatVector(cPtr, false); + } + + public SWIGTYPE_p_float get_centroids(long m, long i) { + long cPtr = swigfaissJNI.ProductQuantizer_get_centroids(swigCPtr, this, m, i); + return (cPtr == 0) ? null : new SWIGTYPE_p_float(cPtr, false); + } + + public void train(int n, SWIGTYPE_p_float x) { + swigfaissJNI.ProductQuantizer_train(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x)); + } + + public ProductQuantizer(long d, long M, long nbits) { + this(swigfaissJNI.new_ProductQuantizer__SWIG_0(d, M, nbits), true); + } + + public ProductQuantizer() { + this(swigfaissJNI.new_ProductQuantizer__SWIG_1(), true); + } + + public void set_derived_values() { + swigfaissJNI.ProductQuantizer_set_derived_values(swigCPtr, this); + } + + public void set_params(SWIGTYPE_p_float centroids, int m) { + swigfaissJNI.ProductQuantizer_set_params(swigCPtr, this, SWIGTYPE_p_float.getCPtr(centroids), m); + } + + public void compute_code(SWIGTYPE_p_float x, SWIGTYPE_p_unsigned_char code) { + swigfaissJNI.ProductQuantizer_compute_code(swigCPtr, this, SWIGTYPE_p_float.getCPtr(x), SWIGTYPE_p_unsigned_char.getCPtr(code)); + } + + public void compute_codes(SWIGTYPE_p_float x, SWIGTYPE_p_unsigned_char codes, long n) { + swigfaissJNI.ProductQuantizer_compute_codes(swigCPtr, this, SWIGTYPE_p_float.getCPtr(x), SWIGTYPE_p_unsigned_char.getCPtr(codes), n); + } + + public void compute_codes_with_assign_index(SWIGTYPE_p_float x, SWIGTYPE_p_unsigned_char codes, long n) { + swigfaissJNI.ProductQuantizer_compute_codes_with_assign_index(swigCPtr, this, SWIGTYPE_p_float.getCPtr(x), SWIGTYPE_p_unsigned_char.getCPtr(codes), n); + } + + public void decode(SWIGTYPE_p_unsigned_char code, SWIGTYPE_p_float x) { + swigfaissJNI.ProductQuantizer_decode__SWIG_0(swigCPtr, this, SWIGTYPE_p_unsigned_char.getCPtr(code), SWIGTYPE_p_float.getCPtr(x)); + } + + public void decode(SWIGTYPE_p_unsigned_char code, SWIGTYPE_p_float x, long n) { + swigfaissJNI.ProductQuantizer_decode__SWIG_1(swigCPtr, this, SWIGTYPE_p_unsigned_char.getCPtr(code), SWIGTYPE_p_float.getCPtr(x), n); + } + + public void compute_code_from_distance_table(SWIGTYPE_p_float tab, SWIGTYPE_p_unsigned_char code) { + swigfaissJNI.ProductQuantizer_compute_code_from_distance_table(swigCPtr, this, SWIGTYPE_p_float.getCPtr(tab), SWIGTYPE_p_unsigned_char.getCPtr(code)); + } + + public void compute_distance_table(SWIGTYPE_p_float x, SWIGTYPE_p_float dis_table) { + swigfaissJNI.ProductQuantizer_compute_distance_table(swigCPtr, this, SWIGTYPE_p_float.getCPtr(x), SWIGTYPE_p_float.getCPtr(dis_table)); + } + + public void compute_inner_prod_table(SWIGTYPE_p_float x, SWIGTYPE_p_float dis_table) { + swigfaissJNI.ProductQuantizer_compute_inner_prod_table(swigCPtr, this, SWIGTYPE_p_float.getCPtr(x), SWIGTYPE_p_float.getCPtr(dis_table)); + } + + public void compute_distance_tables(long nx, SWIGTYPE_p_float x, SWIGTYPE_p_float dis_tables) { + swigfaissJNI.ProductQuantizer_compute_distance_tables(swigCPtr, this, nx, SWIGTYPE_p_float.getCPtr(x), SWIGTYPE_p_float.getCPtr(dis_tables)); + } + + public void compute_inner_prod_tables(long nx, SWIGTYPE_p_float x, SWIGTYPE_p_float dis_tables) { + swigfaissJNI.ProductQuantizer_compute_inner_prod_tables(swigCPtr, this, nx, SWIGTYPE_p_float.getCPtr(x), SWIGTYPE_p_float.getCPtr(dis_tables)); + } + + public void search(SWIGTYPE_p_float x, long nx, SWIGTYPE_p_unsigned_char codes, long ncodes, SWIGTYPE_p_faiss__HeapArrayT_faiss__CMaxT_float_int64_t_t_t res, boolean init_finalize_heap) { + swigfaissJNI.ProductQuantizer_search__SWIG_0(swigCPtr, this, SWIGTYPE_p_float.getCPtr(x), nx, SWIGTYPE_p_unsigned_char.getCPtr(codes), ncodes, SWIGTYPE_p_faiss__HeapArrayT_faiss__CMaxT_float_int64_t_t_t.getCPtr(res), init_finalize_heap); + } + + public void search(SWIGTYPE_p_float x, long nx, SWIGTYPE_p_unsigned_char codes, long ncodes, SWIGTYPE_p_faiss__HeapArrayT_faiss__CMaxT_float_int64_t_t_t res) { + swigfaissJNI.ProductQuantizer_search__SWIG_1(swigCPtr, this, SWIGTYPE_p_float.getCPtr(x), nx, SWIGTYPE_p_unsigned_char.getCPtr(codes), ncodes, SWIGTYPE_p_faiss__HeapArrayT_faiss__CMaxT_float_int64_t_t_t.getCPtr(res)); + } + + public void search_ip(SWIGTYPE_p_float x, long nx, SWIGTYPE_p_unsigned_char codes, long ncodes, SWIGTYPE_p_faiss__HeapArrayT_faiss__CMinT_float_int64_t_t_t res, boolean init_finalize_heap) { + swigfaissJNI.ProductQuantizer_search_ip__SWIG_0(swigCPtr, this, SWIGTYPE_p_float.getCPtr(x), nx, SWIGTYPE_p_unsigned_char.getCPtr(codes), ncodes, SWIGTYPE_p_faiss__HeapArrayT_faiss__CMinT_float_int64_t_t_t.getCPtr(res), init_finalize_heap); + } + + public void search_ip(SWIGTYPE_p_float x, long nx, SWIGTYPE_p_unsigned_char codes, long ncodes, SWIGTYPE_p_faiss__HeapArrayT_faiss__CMinT_float_int64_t_t_t res) { + swigfaissJNI.ProductQuantizer_search_ip__SWIG_1(swigCPtr, this, SWIGTYPE_p_float.getCPtr(x), nx, SWIGTYPE_p_unsigned_char.getCPtr(codes), ncodes, SWIGTYPE_p_faiss__HeapArrayT_faiss__CMinT_float_int64_t_t_t.getCPtr(res)); + } + + public void setSdc_table(FloatVector value) { + swigfaissJNI.ProductQuantizer_sdc_table_set(swigCPtr, this, FloatVector.getCPtr(value), value); + } + + public FloatVector getSdc_table() { + long cPtr = swigfaissJNI.ProductQuantizer_sdc_table_get(swigCPtr, this); + return (cPtr == 0) ? null : new FloatVector(cPtr, false); + } + + public void compute_sdc_table() { + swigfaissJNI.ProductQuantizer_compute_sdc_table(swigCPtr, this); + } + + public void search_sdc(SWIGTYPE_p_unsigned_char qcodes, long nq, SWIGTYPE_p_unsigned_char bcodes, long ncodes, SWIGTYPE_p_faiss__HeapArrayT_faiss__CMaxT_float_int64_t_t_t res, boolean init_finalize_heap) { + swigfaissJNI.ProductQuantizer_search_sdc__SWIG_0(swigCPtr, this, SWIGTYPE_p_unsigned_char.getCPtr(qcodes), nq, SWIGTYPE_p_unsigned_char.getCPtr(bcodes), ncodes, SWIGTYPE_p_faiss__HeapArrayT_faiss__CMaxT_float_int64_t_t_t.getCPtr(res), init_finalize_heap); + } + + public void search_sdc(SWIGTYPE_p_unsigned_char qcodes, long nq, SWIGTYPE_p_unsigned_char bcodes, long ncodes, SWIGTYPE_p_faiss__HeapArrayT_faiss__CMaxT_float_int64_t_t_t res) { + swigfaissJNI.ProductQuantizer_search_sdc__SWIG_1(swigCPtr, this, SWIGTYPE_p_unsigned_char.getCPtr(qcodes), nq, SWIGTYPE_p_unsigned_char.getCPtr(bcodes), ncodes, SWIGTYPE_p_faiss__HeapArrayT_faiss__CMaxT_float_int64_t_t_t.getCPtr(res)); + } + + public final static class train_type_t { + public final static ProductQuantizer.train_type_t Train_default = new ProductQuantizer.train_type_t("Train_default"); + public final static ProductQuantizer.train_type_t Train_hot_start = new ProductQuantizer.train_type_t("Train_hot_start"); + public final static ProductQuantizer.train_type_t Train_shared = new ProductQuantizer.train_type_t("Train_shared"); + public final static ProductQuantizer.train_type_t Train_hypercube = new ProductQuantizer.train_type_t("Train_hypercube"); + public final static ProductQuantizer.train_type_t Train_hypercube_pca = new ProductQuantizer.train_type_t("Train_hypercube_pca"); + + public final int swigValue() { + return swigValue; + } + + public String toString() { + return swigName; + } + + public static train_type_t swigToEnum(int swigValue) { + if (swigValue < swigValues.length && swigValue >= 0 && swigValues[swigValue].swigValue == swigValue) + return swigValues[swigValue]; + for (int i = 0; i < swigValues.length; i++) + if (swigValues[i].swigValue == swigValue) + return swigValues[i]; + throw new IllegalArgumentException("No enum " + train_type_t.class + " with value " + swigValue); + } + + private train_type_t(String swigName) { + this.swigName = swigName; + this.swigValue = swigNext++; + } + + private train_type_t(String swigName, int swigValue) { + this.swigName = swigName; + this.swigValue = swigValue; + swigNext = swigValue+1; + } + + private train_type_t(String swigName, train_type_t swigEnum) { + this.swigName = swigName; + this.swigValue = swigEnum.swigValue; + swigNext = this.swigValue+1; + } + + private static train_type_t[] swigValues = { Train_default, Train_hot_start, Train_shared, Train_hypercube, Train_hypercube_pca }; + private static int swigNext = 0; + private final int swigValue; + private final String swigName; + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/ProgressiveDimClustering.java b/ann/src/main/java/com/twitter/ann/faiss/swig/ProgressiveDimClustering.java new file mode 100644 index 0000000000..2fbd8e406d --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/ProgressiveDimClustering.java @@ -0,0 +1,85 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class ProgressiveDimClustering extends ProgressiveDimClusteringParameters { + private transient long swigCPtr; + + protected ProgressiveDimClustering(long cPtr, boolean cMemoryOwn) { + super(swigfaissJNI.ProgressiveDimClustering_SWIGUpcast(cPtr), cMemoryOwn); + swigCPtr = cPtr; + } + + protected static long getCPtr(ProgressiveDimClustering obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_ProgressiveDimClustering(swigCPtr); + } + swigCPtr = 0; + } + super.delete(); + } + + public void setD(long value) { + swigfaissJNI.ProgressiveDimClustering_d_set(swigCPtr, this, value); + } + + public long getD() { + return swigfaissJNI.ProgressiveDimClustering_d_get(swigCPtr, this); + } + + public void setK(long value) { + swigfaissJNI.ProgressiveDimClustering_k_set(swigCPtr, this, value); + } + + public long getK() { + return swigfaissJNI.ProgressiveDimClustering_k_get(swigCPtr, this); + } + + public void setCentroids(FloatVector value) { + swigfaissJNI.ProgressiveDimClustering_centroids_set(swigCPtr, this, FloatVector.getCPtr(value), value); + } + + public FloatVector getCentroids() { + long cPtr = swigfaissJNI.ProgressiveDimClustering_centroids_get(swigCPtr, this); + return (cPtr == 0) ? null : new FloatVector(cPtr, false); + } + + public void setIteration_stats(SWIGTYPE_p_std__vectorT_faiss__ClusteringIterationStats_t value) { + swigfaissJNI.ProgressiveDimClustering_iteration_stats_set(swigCPtr, this, SWIGTYPE_p_std__vectorT_faiss__ClusteringIterationStats_t.getCPtr(value)); + } + + public SWIGTYPE_p_std__vectorT_faiss__ClusteringIterationStats_t getIteration_stats() { + long cPtr = swigfaissJNI.ProgressiveDimClustering_iteration_stats_get(swigCPtr, this); + return (cPtr == 0) ? null : new SWIGTYPE_p_std__vectorT_faiss__ClusteringIterationStats_t(cPtr, false); + } + + public ProgressiveDimClustering(int d, int k) { + this(swigfaissJNI.new_ProgressiveDimClustering__SWIG_0(d, k), true); + } + + public ProgressiveDimClustering(int d, int k, ProgressiveDimClusteringParameters cp) { + this(swigfaissJNI.new_ProgressiveDimClustering__SWIG_1(d, k, ProgressiveDimClusteringParameters.getCPtr(cp), cp), true); + } + + public void train(long n, SWIGTYPE_p_float x, ProgressiveDimIndexFactory factory) { + swigfaissJNI.ProgressiveDimClustering_train(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x), ProgressiveDimIndexFactory.getCPtr(factory), factory); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/ProgressiveDimClusteringParameters.java b/ann/src/main/java/com/twitter/ann/faiss/swig/ProgressiveDimClusteringParameters.java new file mode 100644 index 0000000000..927c92c185 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/ProgressiveDimClusteringParameters.java @@ -0,0 +1,59 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class ProgressiveDimClusteringParameters extends ClusteringParameters { + private transient long swigCPtr; + + protected ProgressiveDimClusteringParameters(long cPtr, boolean cMemoryOwn) { + super(swigfaissJNI.ProgressiveDimClusteringParameters_SWIGUpcast(cPtr), cMemoryOwn); + swigCPtr = cPtr; + } + + protected static long getCPtr(ProgressiveDimClusteringParameters obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_ProgressiveDimClusteringParameters(swigCPtr); + } + swigCPtr = 0; + } + super.delete(); + } + + public void setProgressive_dim_steps(int value) { + swigfaissJNI.ProgressiveDimClusteringParameters_progressive_dim_steps_set(swigCPtr, this, value); + } + + public int getProgressive_dim_steps() { + return swigfaissJNI.ProgressiveDimClusteringParameters_progressive_dim_steps_get(swigCPtr, this); + } + + public void setApply_pca(boolean value) { + swigfaissJNI.ProgressiveDimClusteringParameters_apply_pca_set(swigCPtr, this, value); + } + + public boolean getApply_pca() { + return swigfaissJNI.ProgressiveDimClusteringParameters_apply_pca_get(swigCPtr, this); + } + + public ProgressiveDimClusteringParameters() { + this(swigfaissJNI.new_ProgressiveDimClusteringParameters(), true); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/ProgressiveDimIndexFactory.java b/ann/src/main/java/com/twitter/ann/faiss/swig/ProgressiveDimIndexFactory.java new file mode 100644 index 0000000000..d01483ca9a --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/ProgressiveDimIndexFactory.java @@ -0,0 +1,43 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class ProgressiveDimIndexFactory { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected ProgressiveDimIndexFactory(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(ProgressiveDimIndexFactory obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_ProgressiveDimIndexFactory(swigCPtr); + } + swigCPtr = 0; + } + } + + public ProgressiveDimIndexFactory() { + this(swigfaissJNI.new_ProgressiveDimIndexFactory(), true); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/RandomRotationMatrix.java b/ann/src/main/java/com/twitter/ann/faiss/swig/RandomRotationMatrix.java new file mode 100644 index 0000000000..104487d46b --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/RandomRotationMatrix.java @@ -0,0 +1,55 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class RandomRotationMatrix extends LinearTransform { + private transient long swigCPtr; + + protected RandomRotationMatrix(long cPtr, boolean cMemoryOwn) { + super(swigfaissJNI.RandomRotationMatrix_SWIGUpcast(cPtr), cMemoryOwn); + swigCPtr = cPtr; + } + + protected static long getCPtr(RandomRotationMatrix obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_RandomRotationMatrix(swigCPtr); + } + swigCPtr = 0; + } + super.delete(); + } + + public RandomRotationMatrix(int d_in, int d_out) { + this(swigfaissJNI.new_RandomRotationMatrix__SWIG_0(d_in, d_out), true); + } + + public void init(int seed) { + swigfaissJNI.RandomRotationMatrix_init(swigCPtr, this, seed); + } + + public void train(long n, SWIGTYPE_p_float x) { + swigfaissJNI.RandomRotationMatrix_train(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x)); + } + + public RandomRotationMatrix() { + this(swigfaissJNI.new_RandomRotationMatrix__SWIG_1(), true); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/RangeQueryResult.java b/ann/src/main/java/com/twitter/ann/faiss/swig/RangeQueryResult.java new file mode 100644 index 0000000000..83fb5e284a --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/RangeQueryResult.java @@ -0,0 +1,72 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class RangeQueryResult { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected RangeQueryResult(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(RangeQueryResult obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_RangeQueryResult(swigCPtr); + } + swigCPtr = 0; + } + } + + public void setQno(long value) { + swigfaissJNI.RangeQueryResult_qno_set(swigCPtr, this, value); + } + + public long getQno() { + return swigfaissJNI.RangeQueryResult_qno_get(swigCPtr, this); +} + + public void setNres(long value) { + swigfaissJNI.RangeQueryResult_nres_set(swigCPtr, this, value); + } + + public long getNres() { + return swigfaissJNI.RangeQueryResult_nres_get(swigCPtr, this); + } + + public void setPres(RangeSearchPartialResult value) { + swigfaissJNI.RangeQueryResult_pres_set(swigCPtr, this, RangeSearchPartialResult.getCPtr(value), value); + } + + public RangeSearchPartialResult getPres() { + long cPtr = swigfaissJNI.RangeQueryResult_pres_get(swigCPtr, this); + return (cPtr == 0) ? null : new RangeSearchPartialResult(cPtr, false); + } + + public void add(float dis, long id) { + swigfaissJNI.RangeQueryResult_add(swigCPtr, this, dis, id); + } + + public RangeQueryResult() { + this(swigfaissJNI.new_RangeQueryResult(), true); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/RangeSearchPartialResult.java b/ann/src/main/java/com/twitter/ann/faiss/swig/RangeSearchPartialResult.java new file mode 100644 index 0000000000..b7348762a8 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/RangeSearchPartialResult.java @@ -0,0 +1,81 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class RangeSearchPartialResult extends BufferList { + private transient long swigCPtr; + + protected RangeSearchPartialResult(long cPtr, boolean cMemoryOwn) { + super(swigfaissJNI.RangeSearchPartialResult_SWIGUpcast(cPtr), cMemoryOwn); + swigCPtr = cPtr; + } + + protected static long getCPtr(RangeSearchPartialResult obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_RangeSearchPartialResult(swigCPtr); + } + swigCPtr = 0; + } + super.delete(); + } + + public void setRes(RangeSearchResult value) { + swigfaissJNI.RangeSearchPartialResult_res_set(swigCPtr, this, RangeSearchResult.getCPtr(value), value); + } + + public RangeSearchResult getRes() { + long cPtr = swigfaissJNI.RangeSearchPartialResult_res_get(swigCPtr, this); + return (cPtr == 0) ? null : new RangeSearchResult(cPtr, false); + } + + public void setQueries(SWIGTYPE_p_std__vectorT_faiss__RangeQueryResult_t value) { + swigfaissJNI.RangeSearchPartialResult_queries_set(swigCPtr, this, SWIGTYPE_p_std__vectorT_faiss__RangeQueryResult_t.getCPtr(value)); + } + + public SWIGTYPE_p_std__vectorT_faiss__RangeQueryResult_t getQueries() { + long cPtr = swigfaissJNI.RangeSearchPartialResult_queries_get(swigCPtr, this); + return (cPtr == 0) ? null : new SWIGTYPE_p_std__vectorT_faiss__RangeQueryResult_t(cPtr, false); + } + + public RangeQueryResult new_result(long qno) { + return new RangeQueryResult(swigfaissJNI.RangeSearchPartialResult_new_result(swigCPtr, this, qno), false); + } + + public void set_lims() { + swigfaissJNI.RangeSearchPartialResult_set_lims(swigCPtr, this); + } + + public void copy_result(boolean incremental) { + swigfaissJNI.RangeSearchPartialResult_copy_result__SWIG_0(swigCPtr, this, incremental); + } + + public void copy_result() { + swigfaissJNI.RangeSearchPartialResult_copy_result__SWIG_1(swigCPtr, this); + } + + public static void merge(SWIGTYPE_p_std__vectorT_faiss__RangeSearchPartialResult_p_t partial_results, boolean do_delete) { + swigfaissJNI.RangeSearchPartialResult_merge__SWIG_0(SWIGTYPE_p_std__vectorT_faiss__RangeSearchPartialResult_p_t.getCPtr(partial_results), do_delete); + } + + public static void merge(SWIGTYPE_p_std__vectorT_faiss__RangeSearchPartialResult_p_t partial_results) { + swigfaissJNI.RangeSearchPartialResult_merge__SWIG_1(SWIGTYPE_p_std__vectorT_faiss__RangeSearchPartialResult_p_t.getCPtr(partial_results)); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/RangeSearchResult.java b/ann/src/main/java/com/twitter/ann/faiss/swig/RangeSearchResult.java new file mode 100644 index 0000000000..0779b6d708 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/RangeSearchResult.java @@ -0,0 +1,85 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class RangeSearchResult { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected RangeSearchResult(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(RangeSearchResult obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_RangeSearchResult(swigCPtr); + } + swigCPtr = 0; + } + } + + public void setNq(long value) { + swigfaissJNI.RangeSearchResult_nq_set(swigCPtr, this, value); + } + + public long getNq() { + return swigfaissJNI.RangeSearchResult_nq_get(swigCPtr, this); + } + + public void setLims(SWIGTYPE_p_unsigned_long value) { + swigfaissJNI.RangeSearchResult_lims_set(swigCPtr, this, SWIGTYPE_p_unsigned_long.getCPtr(value)); + } + + public SWIGTYPE_p_unsigned_long getLims() { + long cPtr = swigfaissJNI.RangeSearchResult_lims_get(swigCPtr, this); + return (cPtr == 0) ? null : new SWIGTYPE_p_unsigned_long(cPtr, false); + } + + public void setLabels(LongVector value) { + swigfaissJNI.RangeSearchResult_labels_set(swigCPtr, this, SWIGTYPE_p_long_long.getCPtr(value.data()), value); + } + + public LongVector getLabels() { + return new LongVector(swigfaissJNI.RangeSearchResult_labels_get(swigCPtr, this), false); +} + + public void setDistances(SWIGTYPE_p_float value) { + swigfaissJNI.RangeSearchResult_distances_set(swigCPtr, this, SWIGTYPE_p_float.getCPtr(value)); + } + + public SWIGTYPE_p_float getDistances() { + long cPtr = swigfaissJNI.RangeSearchResult_distances_get(swigCPtr, this); + return (cPtr == 0) ? null : new SWIGTYPE_p_float(cPtr, false); + } + + public void setBuffer_size(long value) { + swigfaissJNI.RangeSearchResult_buffer_size_set(swigCPtr, this, value); + } + + public long getBuffer_size() { + return swigfaissJNI.RangeSearchResult_buffer_size_get(swigCPtr, this); + } + + public void do_allocation() { + swigfaissJNI.RangeSearchResult_do_allocation(swigCPtr, this); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/ReadOnlyInvertedLists.java b/ann/src/main/java/com/twitter/ann/faiss/swig/ReadOnlyInvertedLists.java new file mode 100644 index 0000000000..8e85a70bcf --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/ReadOnlyInvertedLists.java @@ -0,0 +1,51 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class ReadOnlyInvertedLists extends InvertedLists { + private transient long swigCPtr; + + protected ReadOnlyInvertedLists(long cPtr, boolean cMemoryOwn) { + super(swigfaissJNI.ReadOnlyInvertedLists_SWIGUpcast(cPtr), cMemoryOwn); + swigCPtr = cPtr; + } + + protected static long getCPtr(ReadOnlyInvertedLists obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_ReadOnlyInvertedLists(swigCPtr); + } + swigCPtr = 0; + } + super.delete(); + } + + public long add_entries(long list_no, long n_entry, LongVector ids, SWIGTYPE_p_unsigned_char code) { + return swigfaissJNI.ReadOnlyInvertedLists_add_entries(swigCPtr, this, list_no, n_entry, SWIGTYPE_p_long_long.getCPtr(ids.data()), ids, SWIGTYPE_p_unsigned_char.getCPtr(code)); + } + + public void update_entries(long list_no, long offset, long n_entry, LongVector ids, SWIGTYPE_p_unsigned_char code) { + swigfaissJNI.ReadOnlyInvertedLists_update_entries(swigCPtr, this, list_no, offset, n_entry, SWIGTYPE_p_long_long.getCPtr(ids.data()), ids, SWIGTYPE_p_unsigned_char.getCPtr(code)); + } + + public void resize(long list_no, long new_size) { + swigfaissJNI.ReadOnlyInvertedLists_resize(swigCPtr, this, list_no, new_size); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/ReconstructFromNeighbors.java b/ann/src/main/java/com/twitter/ann/faiss/swig/ReconstructFromNeighbors.java new file mode 100644 index 0000000000..31eaab625e --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/ReconstructFromNeighbors.java @@ -0,0 +1,161 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class ReconstructFromNeighbors { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected ReconstructFromNeighbors(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(ReconstructFromNeighbors obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_ReconstructFromNeighbors(swigCPtr); + } + swigCPtr = 0; + } + } + + public IndexHNSW getIndex() { + return new IndexHNSW(swigfaissJNI.ReconstructFromNeighbors_index_get(swigCPtr, this), false); + } + + public void setM(long value) { + swigfaissJNI.ReconstructFromNeighbors_M_set(swigCPtr, this, value); + } + + public long getM() { + return swigfaissJNI.ReconstructFromNeighbors_M_get(swigCPtr, this); + } + + public void setK(long value) { + swigfaissJNI.ReconstructFromNeighbors_k_set(swigCPtr, this, value); + } + + public long getK() { + return swigfaissJNI.ReconstructFromNeighbors_k_get(swigCPtr, this); + } + + public void setNsq(long value) { + swigfaissJNI.ReconstructFromNeighbors_nsq_set(swigCPtr, this, value); + } + + public long getNsq() { + return swigfaissJNI.ReconstructFromNeighbors_nsq_get(swigCPtr, this); + } + + public void setCode_size(long value) { + swigfaissJNI.ReconstructFromNeighbors_code_size_set(swigCPtr, this, value); + } + + public long getCode_size() { + return swigfaissJNI.ReconstructFromNeighbors_code_size_get(swigCPtr, this); + } + + public void setK_reorder(int value) { + swigfaissJNI.ReconstructFromNeighbors_k_reorder_set(swigCPtr, this, value); + } + + public int getK_reorder() { + return swigfaissJNI.ReconstructFromNeighbors_k_reorder_get(swigCPtr, this); + } + + public void setCodebook(FloatVector value) { + swigfaissJNI.ReconstructFromNeighbors_codebook_set(swigCPtr, this, FloatVector.getCPtr(value), value); + } + + public FloatVector getCodebook() { + long cPtr = swigfaissJNI.ReconstructFromNeighbors_codebook_get(swigCPtr, this); + return (cPtr == 0) ? null : new FloatVector(cPtr, false); + } + + public void setCodes(ByteVector value) { + swigfaissJNI.ReconstructFromNeighbors_codes_set(swigCPtr, this, ByteVector.getCPtr(value), value); + } + + public ByteVector getCodes() { + long cPtr = swigfaissJNI.ReconstructFromNeighbors_codes_get(swigCPtr, this); + return (cPtr == 0) ? null : new ByteVector(cPtr, false); + } + + public void setNtotal(long value) { + swigfaissJNI.ReconstructFromNeighbors_ntotal_set(swigCPtr, this, value); + } + + public long getNtotal() { + return swigfaissJNI.ReconstructFromNeighbors_ntotal_get(swigCPtr, this); + } + + public void setD(long value) { + swigfaissJNI.ReconstructFromNeighbors_d_set(swigCPtr, this, value); + } + + public long getD() { + return swigfaissJNI.ReconstructFromNeighbors_d_get(swigCPtr, this); + } + + public void setDsub(long value) { + swigfaissJNI.ReconstructFromNeighbors_dsub_set(swigCPtr, this, value); + } + + public long getDsub() { + return swigfaissJNI.ReconstructFromNeighbors_dsub_get(swigCPtr, this); + } + + public ReconstructFromNeighbors(IndexHNSW index, long k, long nsq) { + this(swigfaissJNI.new_ReconstructFromNeighbors__SWIG_0(IndexHNSW.getCPtr(index), index, k, nsq), true); + } + + public ReconstructFromNeighbors(IndexHNSW index, long k) { + this(swigfaissJNI.new_ReconstructFromNeighbors__SWIG_1(IndexHNSW.getCPtr(index), index, k), true); + } + + public ReconstructFromNeighbors(IndexHNSW index) { + this(swigfaissJNI.new_ReconstructFromNeighbors__SWIG_2(IndexHNSW.getCPtr(index), index), true); + } + + public void add_codes(long n, SWIGTYPE_p_float x) { + swigfaissJNI.ReconstructFromNeighbors_add_codes(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x)); + } + + public long compute_distances(long n, LongVector shortlist, SWIGTYPE_p_float query, SWIGTYPE_p_float distances) { + return swigfaissJNI.ReconstructFromNeighbors_compute_distances(swigCPtr, this, n, SWIGTYPE_p_long_long.getCPtr(shortlist.data()), shortlist, SWIGTYPE_p_float.getCPtr(query), SWIGTYPE_p_float.getCPtr(distances)); + } + + public void estimate_code(SWIGTYPE_p_float x, int i, SWIGTYPE_p_unsigned_char code) { + swigfaissJNI.ReconstructFromNeighbors_estimate_code(swigCPtr, this, SWIGTYPE_p_float.getCPtr(x), i, SWIGTYPE_p_unsigned_char.getCPtr(code)); + } + + public void reconstruct(int i, SWIGTYPE_p_float x, SWIGTYPE_p_float tmp) { + swigfaissJNI.ReconstructFromNeighbors_reconstruct(swigCPtr, this, i, SWIGTYPE_p_float.getCPtr(x), SWIGTYPE_p_float.getCPtr(tmp)); + } + + public void reconstruct_n(int n0, int ni, SWIGTYPE_p_float x) { + swigfaissJNI.ReconstructFromNeighbors_reconstruct_n(swigCPtr, this, n0, ni, SWIGTYPE_p_float.getCPtr(x)); + } + + public void get_neighbor_table(int i, SWIGTYPE_p_float out) { + swigfaissJNI.ReconstructFromNeighbors_get_neighbor_table(swigCPtr, this, i, SWIGTYPE_p_float.getCPtr(out)); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/RemapDimensionsTransform.java b/ann/src/main/java/com/twitter/ann/faiss/swig/RemapDimensionsTransform.java new file mode 100644 index 0000000000..3b1af9df92 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/RemapDimensionsTransform.java @@ -0,0 +1,72 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class RemapDimensionsTransform extends VectorTransform { + private transient long swigCPtr; + + protected RemapDimensionsTransform(long cPtr, boolean cMemoryOwn) { + super(swigfaissJNI.RemapDimensionsTransform_SWIGUpcast(cPtr), cMemoryOwn); + swigCPtr = cPtr; + } + + protected static long getCPtr(RemapDimensionsTransform obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_RemapDimensionsTransform(swigCPtr); + } + swigCPtr = 0; + } + super.delete(); + } + + public void setMap(IntVector value) { + swigfaissJNI.RemapDimensionsTransform_map_set(swigCPtr, this, IntVector.getCPtr(value), value); + } + + public IntVector getMap() { + long cPtr = swigfaissJNI.RemapDimensionsTransform_map_get(swigCPtr, this); + return (cPtr == 0) ? null : new IntVector(cPtr, false); + } + + public RemapDimensionsTransform(int d_in, int d_out, SWIGTYPE_p_int map) { + this(swigfaissJNI.new_RemapDimensionsTransform__SWIG_0(d_in, d_out, SWIGTYPE_p_int.getCPtr(map)), true); + } + + public RemapDimensionsTransform(int d_in, int d_out, boolean uniform) { + this(swigfaissJNI.new_RemapDimensionsTransform__SWIG_1(d_in, d_out, uniform), true); + } + + public RemapDimensionsTransform(int d_in, int d_out) { + this(swigfaissJNI.new_RemapDimensionsTransform__SWIG_2(d_in, d_out), true); + } + + public void apply_noalloc(long n, SWIGTYPE_p_float x, SWIGTYPE_p_float xt) { + swigfaissJNI.RemapDimensionsTransform_apply_noalloc(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x), SWIGTYPE_p_float.getCPtr(xt)); + } + + public void reverse_transform(long n, SWIGTYPE_p_float xt, SWIGTYPE_p_float x) { + swigfaissJNI.RemapDimensionsTransform_reverse_transform(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(xt), SWIGTYPE_p_float.getCPtr(x)); + } + + public RemapDimensionsTransform() { + this(swigfaissJNI.new_RemapDimensionsTransform__SWIG_3(), true); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/ReproduceDistancesObjective.java b/ann/src/main/java/com/twitter/ann/faiss/swig/ReproduceDistancesObjective.java new file mode 100644 index 0000000000..19b762d7e4 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/ReproduceDistancesObjective.java @@ -0,0 +1,106 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class ReproduceDistancesObjective extends PermutationObjective { + private transient long swigCPtr; + + protected ReproduceDistancesObjective(long cPtr, boolean cMemoryOwn) { + super(swigfaissJNI.ReproduceDistancesObjective_SWIGUpcast(cPtr), cMemoryOwn); + swigCPtr = cPtr; + } + + protected static long getCPtr(ReproduceDistancesObjective obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_ReproduceDistancesObjective(swigCPtr); + } + swigCPtr = 0; + } + super.delete(); + } + + public void setDis_weight_factor(double value) { + swigfaissJNI.ReproduceDistancesObjective_dis_weight_factor_set(swigCPtr, this, value); + } + + public double getDis_weight_factor() { + return swigfaissJNI.ReproduceDistancesObjective_dis_weight_factor_get(swigCPtr, this); + } + + public static double sqr(double x) { + return swigfaissJNI.ReproduceDistancesObjective_sqr(x); + } + + public double dis_weight(double x) { + return swigfaissJNI.ReproduceDistancesObjective_dis_weight(swigCPtr, this, x); + } + + public void setSource_dis(DoubleVector value) { + swigfaissJNI.ReproduceDistancesObjective_source_dis_set(swigCPtr, this, DoubleVector.getCPtr(value), value); + } + + public DoubleVector getSource_dis() { + long cPtr = swigfaissJNI.ReproduceDistancesObjective_source_dis_get(swigCPtr, this); + return (cPtr == 0) ? null : new DoubleVector(cPtr, false); + } + + public void setTarget_dis(SWIGTYPE_p_double value) { + swigfaissJNI.ReproduceDistancesObjective_target_dis_set(swigCPtr, this, SWIGTYPE_p_double.getCPtr(value)); + } + + public SWIGTYPE_p_double getTarget_dis() { + long cPtr = swigfaissJNI.ReproduceDistancesObjective_target_dis_get(swigCPtr, this); + return (cPtr == 0) ? null : new SWIGTYPE_p_double(cPtr, false); + } + + public void setWeights(DoubleVector value) { + swigfaissJNI.ReproduceDistancesObjective_weights_set(swigCPtr, this, DoubleVector.getCPtr(value), value); + } + + public DoubleVector getWeights() { + long cPtr = swigfaissJNI.ReproduceDistancesObjective_weights_get(swigCPtr, this); + return (cPtr == 0) ? null : new DoubleVector(cPtr, false); + } + + public double get_source_dis(int i, int j) { + return swigfaissJNI.ReproduceDistancesObjective_get_source_dis(swigCPtr, this, i, j); + } + + public double compute_cost(SWIGTYPE_p_int perm) { + return swigfaissJNI.ReproduceDistancesObjective_compute_cost(swigCPtr, this, SWIGTYPE_p_int.getCPtr(perm)); + } + + public double cost_update(SWIGTYPE_p_int perm, int iw, int jw) { + return swigfaissJNI.ReproduceDistancesObjective_cost_update(swigCPtr, this, SWIGTYPE_p_int.getCPtr(perm), iw, jw); + } + + public ReproduceDistancesObjective(int n, SWIGTYPE_p_double source_dis_in, SWIGTYPE_p_double target_dis_in, double dis_weight_factor) { + this(swigfaissJNI.new_ReproduceDistancesObjective(n, SWIGTYPE_p_double.getCPtr(source_dis_in), SWIGTYPE_p_double.getCPtr(target_dis_in), dis_weight_factor), true); + } + + public static void compute_mean_stdev(SWIGTYPE_p_double tab, long n2, SWIGTYPE_p_double mean_out, SWIGTYPE_p_double stddev_out) { + swigfaissJNI.ReproduceDistancesObjective_compute_mean_stdev(SWIGTYPE_p_double.getCPtr(tab), n2, SWIGTYPE_p_double.getCPtr(mean_out), SWIGTYPE_p_double.getCPtr(stddev_out)); + } + + public void set_affine_target_dis(SWIGTYPE_p_double source_dis_in) { + swigfaissJNI.ReproduceDistancesObjective_set_affine_target_dis(swigCPtr, this, SWIGTYPE_p_double.getCPtr(source_dis_in)); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_AlignedTableT_float_32_t.java b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_AlignedTableT_float_32_t.java new file mode 100644 index 0000000000..c2ca5d9955 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_AlignedTableT_float_32_t.java @@ -0,0 +1,26 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class SWIGTYPE_p_AlignedTableT_float_32_t { + private transient long swigCPtr; + + protected SWIGTYPE_p_AlignedTableT_float_32_t(long cPtr, @SuppressWarnings("unused") boolean futureUse) { + swigCPtr = cPtr; + } + + protected SWIGTYPE_p_AlignedTableT_float_32_t() { + swigCPtr = 0; + } + + protected static long getCPtr(SWIGTYPE_p_AlignedTableT_float_32_t obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } +} + diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_AlignedTableT_float_t.java b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_AlignedTableT_float_t.java new file mode 100644 index 0000000000..f77626d23c --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_AlignedTableT_float_t.java @@ -0,0 +1,26 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class SWIGTYPE_p_AlignedTableT_float_t { + private transient long swigCPtr; + + protected SWIGTYPE_p_AlignedTableT_float_t(long cPtr, @SuppressWarnings("unused") boolean futureUse) { + swigCPtr = cPtr; + } + + protected SWIGTYPE_p_AlignedTableT_float_t() { + swigCPtr = 0; + } + + protected static long getCPtr(SWIGTYPE_p_AlignedTableT_float_t obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } +} + diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_DirectMap.java b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_DirectMap.java new file mode 100644 index 0000000000..d4a2f0015c --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_DirectMap.java @@ -0,0 +1,26 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class SWIGTYPE_p_DirectMap { + private transient long swigCPtr; + + protected SWIGTYPE_p_DirectMap(long cPtr, @SuppressWarnings("unused") boolean futureUse) { + swigCPtr = cPtr; + } + + protected SWIGTYPE_p_DirectMap() { + swigCPtr = 0; + } + + protected static long getCPtr(SWIGTYPE_p_DirectMap obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } +} + diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_DirectMap__Type.java b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_DirectMap__Type.java new file mode 100644 index 0000000000..94763e8df8 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_DirectMap__Type.java @@ -0,0 +1,26 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class SWIGTYPE_p_DirectMap__Type { + private transient long swigCPtr; + + protected SWIGTYPE_p_DirectMap__Type(long cPtr, @SuppressWarnings("unused") boolean futureUse) { + swigCPtr = cPtr; + } + + protected SWIGTYPE_p_DirectMap__Type() { + swigCPtr = 0; + } + + protected static long getCPtr(SWIGTYPE_p_DirectMap__Type obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } +} + diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_FILE.java b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_FILE.java new file mode 100644 index 0000000000..7e27331074 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_FILE.java @@ -0,0 +1,26 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class SWIGTYPE_p_FILE { + private transient long swigCPtr; + + protected SWIGTYPE_p_FILE(long cPtr, @SuppressWarnings("unused") boolean futureUse) { + swigCPtr = cPtr; + } + + protected SWIGTYPE_p_FILE() { + swigCPtr = 0; + } + + protected static long getCPtr(SWIGTYPE_p_FILE obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } +} + diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_IOReader.java b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_IOReader.java new file mode 100644 index 0000000000..e4003ea9cc --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_IOReader.java @@ -0,0 +1,26 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class SWIGTYPE_p_IOReader { + private transient long swigCPtr; + + protected SWIGTYPE_p_IOReader(long cPtr, @SuppressWarnings("unused") boolean futureUse) { + swigCPtr = cPtr; + } + + protected SWIGTYPE_p_IOReader() { + swigCPtr = 0; + } + + protected static long getCPtr(SWIGTYPE_p_IOReader obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } +} + diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_IOWriter.java b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_IOWriter.java new file mode 100644 index 0000000000..c618a7728a --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_IOWriter.java @@ -0,0 +1,26 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class SWIGTYPE_p_IOWriter { + private transient long swigCPtr; + + protected SWIGTYPE_p_IOWriter(long cPtr, @SuppressWarnings("unused") boolean futureUse) { + swigCPtr = cPtr; + } + + protected SWIGTYPE_p_IOWriter() { + swigCPtr = 0; + } + + protected static long getCPtr(SWIGTYPE_p_IOWriter obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } +} + diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_ScalarQuantizer.java b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_ScalarQuantizer.java new file mode 100644 index 0000000000..aa797717ec --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_ScalarQuantizer.java @@ -0,0 +1,26 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class SWIGTYPE_p_ScalarQuantizer { + private transient long swigCPtr; + + protected SWIGTYPE_p_ScalarQuantizer(long cPtr, @SuppressWarnings("unused") boolean futureUse) { + swigCPtr = cPtr; + } + + protected SWIGTYPE_p_ScalarQuantizer() { + swigCPtr = 0; + } + + protected static long getCPtr(SWIGTYPE_p_ScalarQuantizer obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } +} + diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_ScalarQuantizer__QuantizerType.java b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_ScalarQuantizer__QuantizerType.java new file mode 100644 index 0000000000..79c1e62c48 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_ScalarQuantizer__QuantizerType.java @@ -0,0 +1,26 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class SWIGTYPE_p_ScalarQuantizer__QuantizerType { + private transient long swigCPtr; + + protected SWIGTYPE_p_ScalarQuantizer__QuantizerType(long cPtr, @SuppressWarnings("unused") boolean futureUse) { + swigCPtr = cPtr; + } + + protected SWIGTYPE_p_ScalarQuantizer__QuantizerType() { + swigCPtr = 0; + } + + protected static long getCPtr(SWIGTYPE_p_ScalarQuantizer__QuantizerType obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } +} + diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_double.java b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_double.java new file mode 100644 index 0000000000..3fc49d14ef --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_double.java @@ -0,0 +1,26 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class SWIGTYPE_p_double { + private transient long swigCPtr; + + protected SWIGTYPE_p_double(long cPtr, @SuppressWarnings("unused") boolean futureUse) { + swigCPtr = cPtr; + } + + protected SWIGTYPE_p_double() { + swigCPtr = 0; + } + + protected static long getCPtr(SWIGTYPE_p_double obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } +} + diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_faiss__AlignedTableTightAllocT_float_32_t.java b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_faiss__AlignedTableTightAllocT_float_32_t.java new file mode 100644 index 0000000000..bef0e5fd1f --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_faiss__AlignedTableTightAllocT_float_32_t.java @@ -0,0 +1,26 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class SWIGTYPE_p_faiss__AlignedTableTightAllocT_float_32_t { + private transient long swigCPtr; + + protected SWIGTYPE_p_faiss__AlignedTableTightAllocT_float_32_t(long cPtr, @SuppressWarnings("unused") boolean futureUse) { + swigCPtr = cPtr; + } + + protected SWIGTYPE_p_faiss__AlignedTableTightAllocT_float_32_t() { + swigCPtr = 0; + } + + protected static long getCPtr(SWIGTYPE_p_faiss__AlignedTableTightAllocT_float_32_t obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } +} + diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_faiss__AlignedTableTightAllocT_uint16_t_32_t.java b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_faiss__AlignedTableTightAllocT_uint16_t_32_t.java new file mode 100644 index 0000000000..0c63b2e1cf --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_faiss__AlignedTableTightAllocT_uint16_t_32_t.java @@ -0,0 +1,26 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class SWIGTYPE_p_faiss__AlignedTableTightAllocT_uint16_t_32_t { + private transient long swigCPtr; + + protected SWIGTYPE_p_faiss__AlignedTableTightAllocT_uint16_t_32_t(long cPtr, @SuppressWarnings("unused") boolean futureUse) { + swigCPtr = cPtr; + } + + protected SWIGTYPE_p_faiss__AlignedTableTightAllocT_uint16_t_32_t() { + swigCPtr = 0; + } + + protected static long getCPtr(SWIGTYPE_p_faiss__AlignedTableTightAllocT_uint16_t_32_t obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } +} + diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_faiss__AlignedTableTightAllocT_unsigned_char_32_t.java b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_faiss__AlignedTableTightAllocT_unsigned_char_32_t.java new file mode 100644 index 0000000000..a347de470b --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_faiss__AlignedTableTightAllocT_unsigned_char_32_t.java @@ -0,0 +1,26 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class SWIGTYPE_p_faiss__AlignedTableTightAllocT_unsigned_char_32_t { + private transient long swigCPtr; + + protected SWIGTYPE_p_faiss__AlignedTableTightAllocT_unsigned_char_32_t(long cPtr, @SuppressWarnings("unused") boolean futureUse) { + swigCPtr = cPtr; + } + + protected SWIGTYPE_p_faiss__AlignedTableTightAllocT_unsigned_char_32_t() { + swigCPtr = 0; + } + + protected static long getCPtr(SWIGTYPE_p_faiss__AlignedTableTightAllocT_unsigned_char_32_t obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } +} + diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_faiss__BinaryInvertedListScanner.java b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_faiss__BinaryInvertedListScanner.java new file mode 100644 index 0000000000..9c0c3af144 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_faiss__BinaryInvertedListScanner.java @@ -0,0 +1,26 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class SWIGTYPE_p_faiss__BinaryInvertedListScanner { + private transient long swigCPtr; + + protected SWIGTYPE_p_faiss__BinaryInvertedListScanner(long cPtr, @SuppressWarnings("unused") boolean futureUse) { + swigCPtr = cPtr; + } + + protected SWIGTYPE_p_faiss__BinaryInvertedListScanner() { + swigCPtr = 0; + } + + protected static long getCPtr(SWIGTYPE_p_faiss__BinaryInvertedListScanner obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } +} + diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_faiss__HeapArrayT_faiss__CMaxT_float_int64_t_t_t.java b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_faiss__HeapArrayT_faiss__CMaxT_float_int64_t_t_t.java new file mode 100644 index 0000000000..8f084c1260 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_faiss__HeapArrayT_faiss__CMaxT_float_int64_t_t_t.java @@ -0,0 +1,26 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class SWIGTYPE_p_faiss__HeapArrayT_faiss__CMaxT_float_int64_t_t_t { + private transient long swigCPtr; + + protected SWIGTYPE_p_faiss__HeapArrayT_faiss__CMaxT_float_int64_t_t_t(long cPtr, @SuppressWarnings("unused") boolean futureUse) { + swigCPtr = cPtr; + } + + protected SWIGTYPE_p_faiss__HeapArrayT_faiss__CMaxT_float_int64_t_t_t() { + swigCPtr = 0; + } + + protected static long getCPtr(SWIGTYPE_p_faiss__HeapArrayT_faiss__CMaxT_float_int64_t_t_t obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } +} + diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_faiss__HeapArrayT_faiss__CMaxT_int_int64_t_t_t.java b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_faiss__HeapArrayT_faiss__CMaxT_int_int64_t_t_t.java new file mode 100644 index 0000000000..2200ff2f0e --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_faiss__HeapArrayT_faiss__CMaxT_int_int64_t_t_t.java @@ -0,0 +1,26 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class SWIGTYPE_p_faiss__HeapArrayT_faiss__CMaxT_int_int64_t_t_t { + private transient long swigCPtr; + + protected SWIGTYPE_p_faiss__HeapArrayT_faiss__CMaxT_int_int64_t_t_t(long cPtr, @SuppressWarnings("unused") boolean futureUse) { + swigCPtr = cPtr; + } + + protected SWIGTYPE_p_faiss__HeapArrayT_faiss__CMaxT_int_int64_t_t_t() { + swigCPtr = 0; + } + + protected static long getCPtr(SWIGTYPE_p_faiss__HeapArrayT_faiss__CMaxT_int_int64_t_t_t obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } +} + diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_faiss__HeapArrayT_faiss__CMinT_float_int64_t_t_t.java b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_faiss__HeapArrayT_faiss__CMinT_float_int64_t_t_t.java new file mode 100644 index 0000000000..e8582625f7 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_faiss__HeapArrayT_faiss__CMinT_float_int64_t_t_t.java @@ -0,0 +1,26 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class SWIGTYPE_p_faiss__HeapArrayT_faiss__CMinT_float_int64_t_t_t { + private transient long swigCPtr; + + protected SWIGTYPE_p_faiss__HeapArrayT_faiss__CMinT_float_int64_t_t_t(long cPtr, @SuppressWarnings("unused") boolean futureUse) { + swigCPtr = cPtr; + } + + protected SWIGTYPE_p_faiss__HeapArrayT_faiss__CMinT_float_int64_t_t_t() { + swigCPtr = 0; + } + + protected static long getCPtr(SWIGTYPE_p_faiss__HeapArrayT_faiss__CMinT_float_int64_t_t_t obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } +} + diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_faiss__IOReader.java b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_faiss__IOReader.java new file mode 100644 index 0000000000..26aa9a153c --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_faiss__IOReader.java @@ -0,0 +1,26 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class SWIGTYPE_p_faiss__IOReader { + private transient long swigCPtr; + + protected SWIGTYPE_p_faiss__IOReader(long cPtr, @SuppressWarnings("unused") boolean futureUse) { + swigCPtr = cPtr; + } + + protected SWIGTYPE_p_faiss__IOReader() { + swigCPtr = 0; + } + + protected static long getCPtr(SWIGTYPE_p_faiss__IOReader obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } +} + diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_faiss__IOWriter.java b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_faiss__IOWriter.java new file mode 100644 index 0000000000..7733421c19 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_faiss__IOWriter.java @@ -0,0 +1,26 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class SWIGTYPE_p_faiss__IOWriter { + private transient long swigCPtr; + + protected SWIGTYPE_p_faiss__IOWriter(long cPtr, @SuppressWarnings("unused") boolean futureUse) { + swigCPtr = cPtr; + } + + protected SWIGTYPE_p_faiss__IOWriter() { + swigCPtr = 0; + } + + protected static long getCPtr(SWIGTYPE_p_faiss__IOWriter obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } +} + diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_faiss__InvertedListScanner.java b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_faiss__InvertedListScanner.java new file mode 100644 index 0000000000..c94ea86047 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_faiss__InvertedListScanner.java @@ -0,0 +1,26 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class SWIGTYPE_p_faiss__InvertedListScanner { + private transient long swigCPtr; + + protected SWIGTYPE_p_faiss__InvertedListScanner(long cPtr, @SuppressWarnings("unused") boolean futureUse) { + swigCPtr = cPtr; + } + + protected SWIGTYPE_p_faiss__InvertedListScanner() { + swigCPtr = 0; + } + + protected static long getCPtr(SWIGTYPE_p_faiss__InvertedListScanner obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } +} + diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_faiss__LockLevels.java b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_faiss__LockLevels.java new file mode 100644 index 0000000000..6f709e65b7 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_faiss__LockLevels.java @@ -0,0 +1,26 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class SWIGTYPE_p_faiss__LockLevels { + private transient long swigCPtr; + + protected SWIGTYPE_p_faiss__LockLevels(long cPtr, @SuppressWarnings("unused") boolean futureUse) { + swigCPtr = cPtr; + } + + protected SWIGTYPE_p_faiss__LockLevels() { + swigCPtr = 0; + } + + protected static long getCPtr(SWIGTYPE_p_faiss__LockLevels obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } +} + diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_faiss__OnDiskInvertedLists__OngoingPrefetch.java b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_faiss__OnDiskInvertedLists__OngoingPrefetch.java new file mode 100644 index 0000000000..49c0a423f8 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_faiss__OnDiskInvertedLists__OngoingPrefetch.java @@ -0,0 +1,26 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class SWIGTYPE_p_faiss__OnDiskInvertedLists__OngoingPrefetch { + private transient long swigCPtr; + + protected SWIGTYPE_p_faiss__OnDiskInvertedLists__OngoingPrefetch(long cPtr, @SuppressWarnings("unused") boolean futureUse) { + swigCPtr = cPtr; + } + + protected SWIGTYPE_p_faiss__OnDiskInvertedLists__OngoingPrefetch() { + swigCPtr = 0; + } + + protected static long getCPtr(SWIGTYPE_p_faiss__OnDiskInvertedLists__OngoingPrefetch obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } +} + diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_faiss__RandomGenerator.java b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_faiss__RandomGenerator.java new file mode 100644 index 0000000000..6c3a988781 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_faiss__RandomGenerator.java @@ -0,0 +1,26 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class SWIGTYPE_p_faiss__RandomGenerator { + private transient long swigCPtr; + + protected SWIGTYPE_p_faiss__RandomGenerator(long cPtr, @SuppressWarnings("unused") boolean futureUse) { + swigCPtr = cPtr; + } + + protected SWIGTYPE_p_faiss__RandomGenerator() { + swigCPtr = 0; + } + + protected static long getCPtr(SWIGTYPE_p_faiss__RandomGenerator obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } +} + diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_float.java b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_float.java new file mode 100644 index 0000000000..e4856b9831 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_float.java @@ -0,0 +1,26 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class SWIGTYPE_p_float { + private transient long swigCPtr; + + protected SWIGTYPE_p_float(long cPtr, @SuppressWarnings("unused") boolean futureUse) { + swigCPtr = cPtr; + } + + protected SWIGTYPE_p_float() { + swigCPtr = 0; + } + + protected static long getCPtr(SWIGTYPE_p_float obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } +} + diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_int.java b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_int.java new file mode 100644 index 0000000000..b3df7335d5 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_int.java @@ -0,0 +1,26 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class SWIGTYPE_p_int { + private transient long swigCPtr; + + protected SWIGTYPE_p_int(long cPtr, @SuppressWarnings("unused") boolean futureUse) { + swigCPtr = cPtr; + } + + protected SWIGTYPE_p_int() { + swigCPtr = 0; + } + + protected static long getCPtr(SWIGTYPE_p_int obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } +} + diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_long.java b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_long.java new file mode 100644 index 0000000000..4f8fa73700 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_long.java @@ -0,0 +1,26 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class SWIGTYPE_p_long { + private transient long swigCPtr; + + protected SWIGTYPE_p_long(long cPtr, @SuppressWarnings("unused") boolean futureUse) { + swigCPtr = cPtr; + } + + protected SWIGTYPE_p_long() { + swigCPtr = 0; + } + + protected static long getCPtr(SWIGTYPE_p_long obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } +} + diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_long_long.java b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_long_long.java new file mode 100644 index 0000000000..e3eda38869 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_long_long.java @@ -0,0 +1,26 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class SWIGTYPE_p_long_long { + private transient long swigCPtr; + + protected SWIGTYPE_p_long_long(long cPtr, @SuppressWarnings("unused") boolean futureUse) { + swigCPtr = cPtr; + } + + protected SWIGTYPE_p_long_long() { + swigCPtr = 0; + } + + protected static long getCPtr(SWIGTYPE_p_long_long obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } +} + diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_omp_lock_t.java b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_omp_lock_t.java new file mode 100644 index 0000000000..dd051ae052 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_omp_lock_t.java @@ -0,0 +1,26 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class SWIGTYPE_p_omp_lock_t { + private transient long swigCPtr; + + protected SWIGTYPE_p_omp_lock_t(long cPtr, @SuppressWarnings("unused") boolean futureUse) { + swigCPtr = cPtr; + } + + protected SWIGTYPE_p_omp_lock_t() { + swigCPtr = 0; + } + + protected static long getCPtr(SWIGTYPE_p_omp_lock_t obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } +} + diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_p_faiss__Index.java b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_p_faiss__Index.java new file mode 100644 index 0000000000..6521ff4579 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_p_faiss__Index.java @@ -0,0 +1,26 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class SWIGTYPE_p_p_faiss__Index { + private transient long swigCPtr; + + protected SWIGTYPE_p_p_faiss__Index(long cPtr, @SuppressWarnings("unused") boolean futureUse) { + swigCPtr = cPtr; + } + + protected SWIGTYPE_p_p_faiss__Index() { + swigCPtr = 0; + } + + protected static long getCPtr(SWIGTYPE_p_p_faiss__Index obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } +} + diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_p_faiss__InvertedLists.java b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_p_faiss__InvertedLists.java new file mode 100644 index 0000000000..f59b0d9ca4 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_p_faiss__InvertedLists.java @@ -0,0 +1,26 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class SWIGTYPE_p_p_faiss__InvertedLists { + private transient long swigCPtr; + + protected SWIGTYPE_p_p_faiss__InvertedLists(long cPtr, @SuppressWarnings("unused") boolean futureUse) { + swigCPtr = cPtr; + } + + protected SWIGTYPE_p_p_faiss__InvertedLists() { + swigCPtr = 0; + } + + protected static long getCPtr(SWIGTYPE_p_p_faiss__InvertedLists obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } +} + diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_p_faiss__VectorTransform.java b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_p_faiss__VectorTransform.java new file mode 100644 index 0000000000..7868f49b59 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_p_faiss__VectorTransform.java @@ -0,0 +1,26 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class SWIGTYPE_p_p_faiss__VectorTransform { + private transient long swigCPtr; + + protected SWIGTYPE_p_p_faiss__VectorTransform(long cPtr, @SuppressWarnings("unused") boolean futureUse) { + swigCPtr = cPtr; + } + + protected SWIGTYPE_p_p_faiss__VectorTransform() { + swigCPtr = 0; + } + + protected static long getCPtr(SWIGTYPE_p_p_faiss__VectorTransform obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } +} + diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_std__listT_faiss__OnDiskInvertedLists__Slot_t.java b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_std__listT_faiss__OnDiskInvertedLists__Slot_t.java new file mode 100644 index 0000000000..a57effe964 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_std__listT_faiss__OnDiskInvertedLists__Slot_t.java @@ -0,0 +1,26 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class SWIGTYPE_p_std__listT_faiss__OnDiskInvertedLists__Slot_t { + private transient long swigCPtr; + + protected SWIGTYPE_p_std__listT_faiss__OnDiskInvertedLists__Slot_t(long cPtr, @SuppressWarnings("unused") boolean futureUse) { + swigCPtr = cPtr; + } + + protected SWIGTYPE_p_std__listT_faiss__OnDiskInvertedLists__Slot_t() { + swigCPtr = 0; + } + + protected static long getCPtr(SWIGTYPE_p_std__listT_faiss__OnDiskInvertedLists__Slot_t obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } +} + diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_std__pairT_float_int_t.java b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_std__pairT_float_int_t.java new file mode 100644 index 0000000000..e168f768c0 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_std__pairT_float_int_t.java @@ -0,0 +1,26 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class SWIGTYPE_p_std__pairT_float_int_t { + private transient long swigCPtr; + + protected SWIGTYPE_p_std__pairT_float_int_t(long cPtr, @SuppressWarnings("unused") boolean futureUse) { + swigCPtr = cPtr; + } + + protected SWIGTYPE_p_std__pairT_float_int_t() { + swigCPtr = 0; + } + + protected static long getCPtr(SWIGTYPE_p_std__pairT_float_int_t obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } +} + diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_std__priority_queueT_faiss__HNSW__NodeDistFarther_t.java b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_std__priority_queueT_faiss__HNSW__NodeDistFarther_t.java new file mode 100644 index 0000000000..e36ee5f130 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_std__priority_queueT_faiss__HNSW__NodeDistFarther_t.java @@ -0,0 +1,26 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class SWIGTYPE_p_std__priority_queueT_faiss__HNSW__NodeDistFarther_t { + private transient long swigCPtr; + + protected SWIGTYPE_p_std__priority_queueT_faiss__HNSW__NodeDistFarther_t(long cPtr, @SuppressWarnings("unused") boolean futureUse) { + swigCPtr = cPtr; + } + + protected SWIGTYPE_p_std__priority_queueT_faiss__HNSW__NodeDistFarther_t() { + swigCPtr = 0; + } + + protected static long getCPtr(SWIGTYPE_p_std__priority_queueT_faiss__HNSW__NodeDistFarther_t obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } +} + diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_std__priority_queueT_std__pairT_float_int_t_t.java b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_std__priority_queueT_std__pairT_float_int_t_t.java new file mode 100644 index 0000000000..582f8a1c5a --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_std__priority_queueT_std__pairT_float_int_t_t.java @@ -0,0 +1,26 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class SWIGTYPE_p_std__priority_queueT_std__pairT_float_int_t_t { + private transient long swigCPtr; + + protected SWIGTYPE_p_std__priority_queueT_std__pairT_float_int_t_t(long cPtr, @SuppressWarnings("unused") boolean futureUse) { + swigCPtr = cPtr; + } + + protected SWIGTYPE_p_std__priority_queueT_std__pairT_float_int_t_t() { + swigCPtr = 0; + } + + protected static long getCPtr(SWIGTYPE_p_std__priority_queueT_std__pairT_float_int_t_t obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } +} + diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_std__unordered_mapT_long_long_t.java b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_std__unordered_mapT_long_long_t.java new file mode 100644 index 0000000000..2d3e1688dc --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_std__unordered_mapT_long_long_t.java @@ -0,0 +1,26 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class SWIGTYPE_p_std__unordered_mapT_long_long_t { + private transient long swigCPtr; + + protected SWIGTYPE_p_std__unordered_mapT_long_long_t(long cPtr, @SuppressWarnings("unused") boolean futureUse) { + swigCPtr = cPtr; + } + + protected SWIGTYPE_p_std__unordered_mapT_long_long_t() { + swigCPtr = 0; + } + + protected static long getCPtr(SWIGTYPE_p_std__unordered_mapT_long_long_t obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } +} + diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_std__unordered_multimapT_int64_t_int64_t_t.java b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_std__unordered_multimapT_int64_t_int64_t_t.java new file mode 100644 index 0000000000..443e100db1 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_std__unordered_multimapT_int64_t_int64_t_t.java @@ -0,0 +1,26 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class SWIGTYPE_p_std__unordered_multimapT_int64_t_int64_t_t { + private transient long swigCPtr; + + protected SWIGTYPE_p_std__unordered_multimapT_int64_t_int64_t_t(long cPtr, @SuppressWarnings("unused") boolean futureUse) { + swigCPtr = cPtr; + } + + protected SWIGTYPE_p_std__unordered_multimapT_int64_t_int64_t_t() { + swigCPtr = 0; + } + + protected static long getCPtr(SWIGTYPE_p_std__unordered_multimapT_int64_t_int64_t_t obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } +} + diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_std__vectorT_faiss__BufferList__Buffer_t.java b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_std__vectorT_faiss__BufferList__Buffer_t.java new file mode 100644 index 0000000000..0e65925c3c --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_std__vectorT_faiss__BufferList__Buffer_t.java @@ -0,0 +1,26 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class SWIGTYPE_p_std__vectorT_faiss__BufferList__Buffer_t { + private transient long swigCPtr; + + protected SWIGTYPE_p_std__vectorT_faiss__BufferList__Buffer_t(long cPtr, @SuppressWarnings("unused") boolean futureUse) { + swigCPtr = cPtr; + } + + protected SWIGTYPE_p_std__vectorT_faiss__BufferList__Buffer_t() { + swigCPtr = 0; + } + + protected static long getCPtr(SWIGTYPE_p_std__vectorT_faiss__BufferList__Buffer_t obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } +} + diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_std__vectorT_faiss__ClusteringIterationStats_t.java b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_std__vectorT_faiss__ClusteringIterationStats_t.java new file mode 100644 index 0000000000..c0eaa547f4 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_std__vectorT_faiss__ClusteringIterationStats_t.java @@ -0,0 +1,26 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class SWIGTYPE_p_std__vectorT_faiss__ClusteringIterationStats_t { + private transient long swigCPtr; + + protected SWIGTYPE_p_std__vectorT_faiss__ClusteringIterationStats_t(long cPtr, @SuppressWarnings("unused") boolean futureUse) { + swigCPtr = cPtr; + } + + protected SWIGTYPE_p_std__vectorT_faiss__ClusteringIterationStats_t() { + swigCPtr = 0; + } + + protected static long getCPtr(SWIGTYPE_p_std__vectorT_faiss__ClusteringIterationStats_t obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } +} + diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_std__vectorT_faiss__HNSW__NodeDistFarther_t.java b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_std__vectorT_faiss__HNSW__NodeDistFarther_t.java new file mode 100644 index 0000000000..52c8371c70 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_std__vectorT_faiss__HNSW__NodeDistFarther_t.java @@ -0,0 +1,26 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class SWIGTYPE_p_std__vectorT_faiss__HNSW__NodeDistFarther_t { + private transient long swigCPtr; + + protected SWIGTYPE_p_std__vectorT_faiss__HNSW__NodeDistFarther_t(long cPtr, @SuppressWarnings("unused") boolean futureUse) { + swigCPtr = cPtr; + } + + protected SWIGTYPE_p_std__vectorT_faiss__HNSW__NodeDistFarther_t() { + swigCPtr = 0; + } + + protected static long getCPtr(SWIGTYPE_p_std__vectorT_faiss__HNSW__NodeDistFarther_t obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } +} + diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_std__vectorT_faiss__Index_p_t.java b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_std__vectorT_faiss__Index_p_t.java new file mode 100644 index 0000000000..f79e29ad6f --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_std__vectorT_faiss__Index_p_t.java @@ -0,0 +1,26 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class SWIGTYPE_p_std__vectorT_faiss__Index_p_t { + private transient long swigCPtr; + + protected SWIGTYPE_p_std__vectorT_faiss__Index_p_t(long cPtr, @SuppressWarnings("unused") boolean futureUse) { + swigCPtr = cPtr; + } + + protected SWIGTYPE_p_std__vectorT_faiss__Index_p_t() { + swigCPtr = 0; + } + + protected static long getCPtr(SWIGTYPE_p_std__vectorT_faiss__Index_p_t obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } +} + diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_std__vectorT_faiss__InvertedLists_const_p_t.java b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_std__vectorT_faiss__InvertedLists_const_p_t.java new file mode 100644 index 0000000000..2e14f35c5c --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_std__vectorT_faiss__InvertedLists_const_p_t.java @@ -0,0 +1,26 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class SWIGTYPE_p_std__vectorT_faiss__InvertedLists_const_p_t { + private transient long swigCPtr; + + protected SWIGTYPE_p_std__vectorT_faiss__InvertedLists_const_p_t(long cPtr, @SuppressWarnings("unused") boolean futureUse) { + swigCPtr = cPtr; + } + + protected SWIGTYPE_p_std__vectorT_faiss__InvertedLists_const_p_t() { + swigCPtr = 0; + } + + protected static long getCPtr(SWIGTYPE_p_std__vectorT_faiss__InvertedLists_const_p_t obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } +} + diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_std__vectorT_faiss__OnDiskOneList_t.java b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_std__vectorT_faiss__OnDiskOneList_t.java new file mode 100644 index 0000000000..9d79a184c3 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_std__vectorT_faiss__OnDiskOneList_t.java @@ -0,0 +1,26 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class SWIGTYPE_p_std__vectorT_faiss__OnDiskOneList_t { + private transient long swigCPtr; + + protected SWIGTYPE_p_std__vectorT_faiss__OnDiskOneList_t(long cPtr, @SuppressWarnings("unused") boolean futureUse) { + swigCPtr = cPtr; + } + + protected SWIGTYPE_p_std__vectorT_faiss__OnDiskOneList_t() { + swigCPtr = 0; + } + + protected static long getCPtr(SWIGTYPE_p_std__vectorT_faiss__OnDiskOneList_t obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } +} + diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_std__vectorT_faiss__ParameterRange_t.java b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_std__vectorT_faiss__ParameterRange_t.java new file mode 100644 index 0000000000..99e3adabfe --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_std__vectorT_faiss__ParameterRange_t.java @@ -0,0 +1,26 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class SWIGTYPE_p_std__vectorT_faiss__ParameterRange_t { + private transient long swigCPtr; + + protected SWIGTYPE_p_std__vectorT_faiss__ParameterRange_t(long cPtr, @SuppressWarnings("unused") boolean futureUse) { + swigCPtr = cPtr; + } + + protected SWIGTYPE_p_std__vectorT_faiss__ParameterRange_t() { + swigCPtr = 0; + } + + protected static long getCPtr(SWIGTYPE_p_std__vectorT_faiss__ParameterRange_t obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } +} + diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_std__vectorT_faiss__RangeQueryResult_t.java b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_std__vectorT_faiss__RangeQueryResult_t.java new file mode 100644 index 0000000000..64a1711ab0 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_std__vectorT_faiss__RangeQueryResult_t.java @@ -0,0 +1,26 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class SWIGTYPE_p_std__vectorT_faiss__RangeQueryResult_t { + private transient long swigCPtr; + + protected SWIGTYPE_p_std__vectorT_faiss__RangeQueryResult_t(long cPtr, @SuppressWarnings("unused") boolean futureUse) { + swigCPtr = cPtr; + } + + protected SWIGTYPE_p_std__vectorT_faiss__RangeQueryResult_t() { + swigCPtr = 0; + } + + protected static long getCPtr(SWIGTYPE_p_std__vectorT_faiss__RangeQueryResult_t obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } +} + diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_std__vectorT_faiss__RangeSearchPartialResult_p_t.java b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_std__vectorT_faiss__RangeSearchPartialResult_p_t.java new file mode 100644 index 0000000000..67cd2d26a6 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_std__vectorT_faiss__RangeSearchPartialResult_p_t.java @@ -0,0 +1,26 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class SWIGTYPE_p_std__vectorT_faiss__RangeSearchPartialResult_p_t { + private transient long swigCPtr; + + protected SWIGTYPE_p_std__vectorT_faiss__RangeSearchPartialResult_p_t(long cPtr, @SuppressWarnings("unused") boolean futureUse) { + swigCPtr = cPtr; + } + + protected SWIGTYPE_p_std__vectorT_faiss__RangeSearchPartialResult_p_t() { + swigCPtr = 0; + } + + protected static long getCPtr(SWIGTYPE_p_std__vectorT_faiss__RangeSearchPartialResult_p_t obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } +} + diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_std__vectorT_int64_t_t.java b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_std__vectorT_int64_t_t.java new file mode 100644 index 0000000000..06db35e078 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_std__vectorT_int64_t_t.java @@ -0,0 +1,26 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class SWIGTYPE_p_std__vectorT_int64_t_t { + private transient long swigCPtr; + + protected SWIGTYPE_p_std__vectorT_int64_t_t(long cPtr, @SuppressWarnings("unused") boolean futureUse) { + swigCPtr = cPtr; + } + + protected SWIGTYPE_p_std__vectorT_int64_t_t() { + swigCPtr = 0; + } + + protected static long getCPtr(SWIGTYPE_p_std__vectorT_int64_t_t obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } +} + diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_std__vectorT_long_t.java b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_std__vectorT_long_t.java new file mode 100644 index 0000000000..4ad496dc72 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_std__vectorT_long_t.java @@ -0,0 +1,26 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class SWIGTYPE_p_std__vectorT_long_t { + private transient long swigCPtr; + + protected SWIGTYPE_p_std__vectorT_long_t(long cPtr, @SuppressWarnings("unused") boolean futureUse) { + swigCPtr = cPtr; + } + + protected SWIGTYPE_p_std__vectorT_long_t() { + swigCPtr = 0; + } + + protected static long getCPtr(SWIGTYPE_p_std__vectorT_long_t obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } +} + diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_std__vectorT_omp_lock_t_t.java b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_std__vectorT_omp_lock_t_t.java new file mode 100644 index 0000000000..81c9e88a12 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_std__vectorT_omp_lock_t_t.java @@ -0,0 +1,26 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class SWIGTYPE_p_std__vectorT_omp_lock_t_t { + private transient long swigCPtr; + + protected SWIGTYPE_p_std__vectorT_omp_lock_t_t(long cPtr, @SuppressWarnings("unused") boolean futureUse) { + swigCPtr = cPtr; + } + + protected SWIGTYPE_p_std__vectorT_omp_lock_t_t() { + swigCPtr = 0; + } + + protected static long getCPtr(SWIGTYPE_p_std__vectorT_omp_lock_t_t obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } +} + diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_std__vectorT_std__vectorT_int64_t_t_t.java b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_std__vectorT_std__vectorT_int64_t_t_t.java new file mode 100644 index 0000000000..7e5504484f --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_std__vectorT_std__vectorT_int64_t_t_t.java @@ -0,0 +1,26 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class SWIGTYPE_p_std__vectorT_std__vectorT_int64_t_t_t { + private transient long swigCPtr; + + protected SWIGTYPE_p_std__vectorT_std__vectorT_int64_t_t_t(long cPtr, @SuppressWarnings("unused") boolean futureUse) { + swigCPtr = cPtr; + } + + protected SWIGTYPE_p_std__vectorT_std__vectorT_int64_t_t_t() { + swigCPtr = 0; + } + + protected static long getCPtr(SWIGTYPE_p_std__vectorT_std__vectorT_int64_t_t_t obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } +} + diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_std__vectorT_std__vectorT_unsigned_long_t_t.java b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_std__vectorT_std__vectorT_unsigned_long_t_t.java new file mode 100644 index 0000000000..dbd515dcc8 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_std__vectorT_std__vectorT_unsigned_long_t_t.java @@ -0,0 +1,26 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class SWIGTYPE_p_std__vectorT_std__vectorT_unsigned_long_t_t { + private transient long swigCPtr; + + protected SWIGTYPE_p_std__vectorT_std__vectorT_unsigned_long_t_t(long cPtr, @SuppressWarnings("unused") boolean futureUse) { + swigCPtr = cPtr; + } + + protected SWIGTYPE_p_std__vectorT_std__vectorT_unsigned_long_t_t() { + swigCPtr = 0; + } + + protected static long getCPtr(SWIGTYPE_p_std__vectorT_std__vectorT_unsigned_long_t_t obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } +} + diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_uint16_t.java b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_uint16_t.java new file mode 100644 index 0000000000..5ac22962af --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_uint16_t.java @@ -0,0 +1,26 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class SWIGTYPE_p_uint16_t { + private transient long swigCPtr; + + protected SWIGTYPE_p_uint16_t(long cPtr, @SuppressWarnings("unused") boolean futureUse) { + swigCPtr = cPtr; + } + + protected SWIGTYPE_p_uint16_t() { + swigCPtr = 0; + } + + protected static long getCPtr(SWIGTYPE_p_uint16_t obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } +} + diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_uint32_t.java b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_uint32_t.java new file mode 100644 index 0000000000..3f2db28738 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_uint32_t.java @@ -0,0 +1,26 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class SWIGTYPE_p_uint32_t { + private transient long swigCPtr; + + protected SWIGTYPE_p_uint32_t(long cPtr, @SuppressWarnings("unused") boolean futureUse) { + swigCPtr = cPtr; + } + + protected SWIGTYPE_p_uint32_t() { + swigCPtr = 0; + } + + protected static long getCPtr(SWIGTYPE_p_uint32_t obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } +} + diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_unsigned_char.java b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_unsigned_char.java new file mode 100644 index 0000000000..c34696c2cf --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_unsigned_char.java @@ -0,0 +1,26 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class SWIGTYPE_p_unsigned_char { + private transient long swigCPtr; + + protected SWIGTYPE_p_unsigned_char(long cPtr, @SuppressWarnings("unused") boolean futureUse) { + swigCPtr = cPtr; + } + + protected SWIGTYPE_p_unsigned_char() { + swigCPtr = 0; + } + + protected static long getCPtr(SWIGTYPE_p_unsigned_char obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } +} + diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_unsigned_long.java b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_unsigned_long.java new file mode 100644 index 0000000000..f303bca61c --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_unsigned_long.java @@ -0,0 +1,26 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class SWIGTYPE_p_unsigned_long { + private transient long swigCPtr; + + protected SWIGTYPE_p_unsigned_long(long cPtr, @SuppressWarnings("unused") boolean futureUse) { + swigCPtr = cPtr; + } + + protected SWIGTYPE_p_unsigned_long() { + swigCPtr = 0; + } + + protected static long getCPtr(SWIGTYPE_p_unsigned_long obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } +} + diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_void.java b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_void.java new file mode 100644 index 0000000000..2ec4d574cd --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/SWIGTYPE_p_void.java @@ -0,0 +1,26 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class SWIGTYPE_p_void { + private transient long swigCPtr; + + protected SWIGTYPE_p_void(long cPtr, @SuppressWarnings("unused") boolean futureUse) { + swigCPtr = cPtr; + } + + protected SWIGTYPE_p_void() { + swigCPtr = 0; + } + + protected static long getCPtr(SWIGTYPE_p_void obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } +} + diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/SimulatedAnnealingOptimizer.java b/ann/src/main/java/com/twitter/ann/faiss/swig/SimulatedAnnealingOptimizer.java new file mode 100644 index 0000000000..e3b2af66c4 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/SimulatedAnnealingOptimizer.java @@ -0,0 +1,94 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class SimulatedAnnealingOptimizer extends SimulatedAnnealingParameters { + private transient long swigCPtr; + + protected SimulatedAnnealingOptimizer(long cPtr, boolean cMemoryOwn) { + super(swigfaissJNI.SimulatedAnnealingOptimizer_SWIGUpcast(cPtr), cMemoryOwn); + swigCPtr = cPtr; + } + + protected static long getCPtr(SimulatedAnnealingOptimizer obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_SimulatedAnnealingOptimizer(swigCPtr); + } + swigCPtr = 0; + } + super.delete(); + } + + public void setObj(PermutationObjective value) { + swigfaissJNI.SimulatedAnnealingOptimizer_obj_set(swigCPtr, this, PermutationObjective.getCPtr(value), value); + } + + public PermutationObjective getObj() { + long cPtr = swigfaissJNI.SimulatedAnnealingOptimizer_obj_get(swigCPtr, this); + return (cPtr == 0) ? null : new PermutationObjective(cPtr, false); + } + + public void setN(int value) { + swigfaissJNI.SimulatedAnnealingOptimizer_n_set(swigCPtr, this, value); + } + + public int getN() { + return swigfaissJNI.SimulatedAnnealingOptimizer_n_get(swigCPtr, this); + } + + public void setLogfile(SWIGTYPE_p_FILE value) { + swigfaissJNI.SimulatedAnnealingOptimizer_logfile_set(swigCPtr, this, SWIGTYPE_p_FILE.getCPtr(value)); + } + + public SWIGTYPE_p_FILE getLogfile() { + long cPtr = swigfaissJNI.SimulatedAnnealingOptimizer_logfile_get(swigCPtr, this); + return (cPtr == 0) ? null : new SWIGTYPE_p_FILE(cPtr, false); + } + + public SimulatedAnnealingOptimizer(PermutationObjective obj, SimulatedAnnealingParameters p) { + this(swigfaissJNI.new_SimulatedAnnealingOptimizer(PermutationObjective.getCPtr(obj), obj, SimulatedAnnealingParameters.getCPtr(p), p), true); + } + + public void setRnd(SWIGTYPE_p_faiss__RandomGenerator value) { + swigfaissJNI.SimulatedAnnealingOptimizer_rnd_set(swigCPtr, this, SWIGTYPE_p_faiss__RandomGenerator.getCPtr(value)); + } + + public SWIGTYPE_p_faiss__RandomGenerator getRnd() { + long cPtr = swigfaissJNI.SimulatedAnnealingOptimizer_rnd_get(swigCPtr, this); + return (cPtr == 0) ? null : new SWIGTYPE_p_faiss__RandomGenerator(cPtr, false); + } + + public void setInit_cost(double value) { + swigfaissJNI.SimulatedAnnealingOptimizer_init_cost_set(swigCPtr, this, value); + } + + public double getInit_cost() { + return swigfaissJNI.SimulatedAnnealingOptimizer_init_cost_get(swigCPtr, this); + } + + public double optimize(SWIGTYPE_p_int perm) { + return swigfaissJNI.SimulatedAnnealingOptimizer_optimize(swigCPtr, this, SWIGTYPE_p_int.getCPtr(perm)); + } + + public double run_optimization(SWIGTYPE_p_int best_perm) { + return swigfaissJNI.SimulatedAnnealingOptimizer_run_optimization(swigCPtr, this, SWIGTYPE_p_int.getCPtr(best_perm)); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/SimulatedAnnealingParameters.java b/ann/src/main/java/com/twitter/ann/faiss/swig/SimulatedAnnealingParameters.java new file mode 100644 index 0000000000..612d9be4eb --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/SimulatedAnnealingParameters.java @@ -0,0 +1,107 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class SimulatedAnnealingParameters { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected SimulatedAnnealingParameters(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(SimulatedAnnealingParameters obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_SimulatedAnnealingParameters(swigCPtr); + } + swigCPtr = 0; + } + } + + public void setInit_temperature(double value) { + swigfaissJNI.SimulatedAnnealingParameters_init_temperature_set(swigCPtr, this, value); + } + + public double getInit_temperature() { + return swigfaissJNI.SimulatedAnnealingParameters_init_temperature_get(swigCPtr, this); + } + + public void setTemperature_decay(double value) { + swigfaissJNI.SimulatedAnnealingParameters_temperature_decay_set(swigCPtr, this, value); + } + + public double getTemperature_decay() { + return swigfaissJNI.SimulatedAnnealingParameters_temperature_decay_get(swigCPtr, this); + } + + public void setN_iter(int value) { + swigfaissJNI.SimulatedAnnealingParameters_n_iter_set(swigCPtr, this, value); + } + + public int getN_iter() { + return swigfaissJNI.SimulatedAnnealingParameters_n_iter_get(swigCPtr, this); + } + + public void setN_redo(int value) { + swigfaissJNI.SimulatedAnnealingParameters_n_redo_set(swigCPtr, this, value); + } + + public int getN_redo() { + return swigfaissJNI.SimulatedAnnealingParameters_n_redo_get(swigCPtr, this); + } + + public void setSeed(int value) { + swigfaissJNI.SimulatedAnnealingParameters_seed_set(swigCPtr, this, value); + } + + public int getSeed() { + return swigfaissJNI.SimulatedAnnealingParameters_seed_get(swigCPtr, this); + } + + public void setVerbose(int value) { + swigfaissJNI.SimulatedAnnealingParameters_verbose_set(swigCPtr, this, value); + } + + public int getVerbose() { + return swigfaissJNI.SimulatedAnnealingParameters_verbose_get(swigCPtr, this); + } + + public void setOnly_bit_flips(boolean value) { + swigfaissJNI.SimulatedAnnealingParameters_only_bit_flips_set(swigCPtr, this, value); + } + + public boolean getOnly_bit_flips() { + return swigfaissJNI.SimulatedAnnealingParameters_only_bit_flips_get(swigCPtr, this); + } + + public void setInit_random(boolean value) { + swigfaissJNI.SimulatedAnnealingParameters_init_random_set(swigCPtr, this, value); + } + + public boolean getInit_random() { + return swigfaissJNI.SimulatedAnnealingParameters_init_random_get(swigCPtr, this); + } + + public SimulatedAnnealingParameters() { + this(swigfaissJNI.new_SimulatedAnnealingParameters(), true); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/SliceInvertedLists.java b/ann/src/main/java/com/twitter/ann/faiss/swig/SliceInvertedLists.java new file mode 100644 index 0000000000..6a551db149 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/SliceInvertedLists.java @@ -0,0 +1,102 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class SliceInvertedLists extends ReadOnlyInvertedLists { + private transient long swigCPtr; + + protected SliceInvertedLists(long cPtr, boolean cMemoryOwn) { + super(swigfaissJNI.SliceInvertedLists_SWIGUpcast(cPtr), cMemoryOwn); + swigCPtr = cPtr; + } + + protected static long getCPtr(SliceInvertedLists obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_SliceInvertedLists(swigCPtr); + } + swigCPtr = 0; + } + super.delete(); + } + + public void setIl(InvertedLists value) { + swigfaissJNI.SliceInvertedLists_il_set(swigCPtr, this, InvertedLists.getCPtr(value), value); + } + + public InvertedLists getIl() { + long cPtr = swigfaissJNI.SliceInvertedLists_il_get(swigCPtr, this); + return (cPtr == 0) ? null : new InvertedLists(cPtr, false); + } + + public void setI0(long value) { + swigfaissJNI.SliceInvertedLists_i0_set(swigCPtr, this, value); + } + + public long getI0() { + return swigfaissJNI.SliceInvertedLists_i0_get(swigCPtr, this); +} + + public void setI1(long value) { + swigfaissJNI.SliceInvertedLists_i1_set(swigCPtr, this, value); + } + + public long getI1() { + return swigfaissJNI.SliceInvertedLists_i1_get(swigCPtr, this); +} + + public SliceInvertedLists(InvertedLists il, long i0, long i1) { + this(swigfaissJNI.new_SliceInvertedLists(InvertedLists.getCPtr(il), il, i0, i1), true); + } + + public long list_size(long list_no) { + return swigfaissJNI.SliceInvertedLists_list_size(swigCPtr, this, list_no); + } + + public SWIGTYPE_p_unsigned_char get_codes(long list_no) { + long cPtr = swigfaissJNI.SliceInvertedLists_get_codes(swigCPtr, this, list_no); + return (cPtr == 0) ? null : new SWIGTYPE_p_unsigned_char(cPtr, false); + } + + public LongVector get_ids(long list_no) { + return new LongVector(swigfaissJNI.SliceInvertedLists_get_ids(swigCPtr, this, list_no), false); +} + + public void release_codes(long list_no, SWIGTYPE_p_unsigned_char codes) { + swigfaissJNI.SliceInvertedLists_release_codes(swigCPtr, this, list_no, SWIGTYPE_p_unsigned_char.getCPtr(codes)); + } + + public void release_ids(long list_no, LongVector ids) { + swigfaissJNI.SliceInvertedLists_release_ids(swigCPtr, this, list_no, SWIGTYPE_p_long_long.getCPtr(ids.data()), ids); + } + + public long get_single_id(long list_no, long offset) { + return swigfaissJNI.SliceInvertedLists_get_single_id(swigCPtr, this, list_no, offset); +} + + public SWIGTYPE_p_unsigned_char get_single_code(long list_no, long offset) { + long cPtr = swigfaissJNI.SliceInvertedLists_get_single_code(swigCPtr, this, list_no, offset); + return (cPtr == 0) ? null : new SWIGTYPE_p_unsigned_char(cPtr, false); + } + + public void prefetch_lists(LongVector list_nos, int nlist) { + swigfaissJNI.SliceInvertedLists_prefetch_lists(swigCPtr, this, SWIGTYPE_p_long_long.getCPtr(list_nos.data()), list_nos, nlist); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/SlidingIndexWindow.java b/ann/src/main/java/com/twitter/ann/faiss/swig/SlidingIndexWindow.java new file mode 100644 index 0000000000..c866e2fa2d --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/SlidingIndexWindow.java @@ -0,0 +1,90 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class SlidingIndexWindow { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected SlidingIndexWindow(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(SlidingIndexWindow obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_SlidingIndexWindow(swigCPtr); + } + swigCPtr = 0; + } + } + + public void setIndex(Index value) { + swigfaissJNI.SlidingIndexWindow_index_set(swigCPtr, this, Index.getCPtr(value), value); + } + + public Index getIndex() { + long cPtr = swigfaissJNI.SlidingIndexWindow_index_get(swigCPtr, this); + return (cPtr == 0) ? null : new Index(cPtr, false); + } + + public void setIls(ArrayInvertedLists value) { + swigfaissJNI.SlidingIndexWindow_ils_set(swigCPtr, this, ArrayInvertedLists.getCPtr(value), value); + } + + public ArrayInvertedLists getIls() { + long cPtr = swigfaissJNI.SlidingIndexWindow_ils_get(swigCPtr, this); + return (cPtr == 0) ? null : new ArrayInvertedLists(cPtr, false); + } + + public void setN_slice(int value) { + swigfaissJNI.SlidingIndexWindow_n_slice_set(swigCPtr, this, value); + } + + public int getN_slice() { + return swigfaissJNI.SlidingIndexWindow_n_slice_get(swigCPtr, this); + } + + public void setNlist(long value) { + swigfaissJNI.SlidingIndexWindow_nlist_set(swigCPtr, this, value); + } + + public long getNlist() { + return swigfaissJNI.SlidingIndexWindow_nlist_get(swigCPtr, this); + } + + public void setSizes(SWIGTYPE_p_std__vectorT_std__vectorT_unsigned_long_t_t value) { + swigfaissJNI.SlidingIndexWindow_sizes_set(swigCPtr, this, SWIGTYPE_p_std__vectorT_std__vectorT_unsigned_long_t_t.getCPtr(value)); + } + + public SWIGTYPE_p_std__vectorT_std__vectorT_unsigned_long_t_t getSizes() { + long cPtr = swigfaissJNI.SlidingIndexWindow_sizes_get(swigCPtr, this); + return (cPtr == 0) ? null : new SWIGTYPE_p_std__vectorT_std__vectorT_unsigned_long_t_t(cPtr, false); + } + + public SlidingIndexWindow(Index index) { + this(swigfaissJNI.new_SlidingIndexWindow(Index.getCPtr(index), index), true); + } + + public void step(Index sub_index, boolean remove_oldest) { + swigfaissJNI.SlidingIndexWindow_step(swigCPtr, this, Index.getCPtr(sub_index), sub_index, remove_oldest); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/StopWordsInvertedLists.java b/ann/src/main/java/com/twitter/ann/faiss/swig/StopWordsInvertedLists.java new file mode 100644 index 0000000000..d951d207d3 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/StopWordsInvertedLists.java @@ -0,0 +1,94 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class StopWordsInvertedLists extends ReadOnlyInvertedLists { + private transient long swigCPtr; + + protected StopWordsInvertedLists(long cPtr, boolean cMemoryOwn) { + super(swigfaissJNI.StopWordsInvertedLists_SWIGUpcast(cPtr), cMemoryOwn); + swigCPtr = cPtr; + } + + protected static long getCPtr(StopWordsInvertedLists obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_StopWordsInvertedLists(swigCPtr); + } + swigCPtr = 0; + } + super.delete(); + } + + public void setIl0(InvertedLists value) { + swigfaissJNI.StopWordsInvertedLists_il0_set(swigCPtr, this, InvertedLists.getCPtr(value), value); + } + + public InvertedLists getIl0() { + long cPtr = swigfaissJNI.StopWordsInvertedLists_il0_get(swigCPtr, this); + return (cPtr == 0) ? null : new InvertedLists(cPtr, false); + } + + public void setMaxsize(long value) { + swigfaissJNI.StopWordsInvertedLists_maxsize_set(swigCPtr, this, value); + } + + public long getMaxsize() { + return swigfaissJNI.StopWordsInvertedLists_maxsize_get(swigCPtr, this); + } + + public StopWordsInvertedLists(InvertedLists il, long maxsize) { + this(swigfaissJNI.new_StopWordsInvertedLists(InvertedLists.getCPtr(il), il, maxsize), true); + } + + public long list_size(long list_no) { + return swigfaissJNI.StopWordsInvertedLists_list_size(swigCPtr, this, list_no); + } + + public SWIGTYPE_p_unsigned_char get_codes(long list_no) { + long cPtr = swigfaissJNI.StopWordsInvertedLists_get_codes(swigCPtr, this, list_no); + return (cPtr == 0) ? null : new SWIGTYPE_p_unsigned_char(cPtr, false); + } + + public LongVector get_ids(long list_no) { + return new LongVector(swigfaissJNI.StopWordsInvertedLists_get_ids(swigCPtr, this, list_no), false); +} + + public void release_codes(long list_no, SWIGTYPE_p_unsigned_char codes) { + swigfaissJNI.StopWordsInvertedLists_release_codes(swigCPtr, this, list_no, SWIGTYPE_p_unsigned_char.getCPtr(codes)); + } + + public void release_ids(long list_no, LongVector ids) { + swigfaissJNI.StopWordsInvertedLists_release_ids(swigCPtr, this, list_no, SWIGTYPE_p_long_long.getCPtr(ids.data()), ids); + } + + public long get_single_id(long list_no, long offset) { + return swigfaissJNI.StopWordsInvertedLists_get_single_id(swigCPtr, this, list_no, offset); +} + + public SWIGTYPE_p_unsigned_char get_single_code(long list_no, long offset) { + long cPtr = swigfaissJNI.StopWordsInvertedLists_get_single_code(swigCPtr, this, list_no, offset); + return (cPtr == 0) ? null : new SWIGTYPE_p_unsigned_char(cPtr, false); + } + + public void prefetch_lists(LongVector list_nos, int nlist) { + swigfaissJNI.StopWordsInvertedLists_prefetch_lists(swigCPtr, this, SWIGTYPE_p_long_long.getCPtr(list_nos.data()), list_nos, nlist); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/Uint64Vector.java b/ann/src/main/java/com/twitter/ann/faiss/swig/Uint64Vector.java new file mode 100644 index 0000000000..cd9b40be7c --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/Uint64Vector.java @@ -0,0 +1,76 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class Uint64Vector { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected Uint64Vector(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(Uint64Vector obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_Uint64Vector(swigCPtr); + } + swigCPtr = 0; + } + } + + public Uint64Vector() { + this(swigfaissJNI.new_Uint64Vector(), true); + } + + public void push_back(long arg0) { + swigfaissJNI.Uint64Vector_push_back(swigCPtr, this, arg0); + } + + public void clear() { + swigfaissJNI.Uint64Vector_clear(swigCPtr, this); + } + + public SWIGTYPE_p_unsigned_long data() { + long cPtr = swigfaissJNI.Uint64Vector_data(swigCPtr, this); + return (cPtr == 0) ? null : new SWIGTYPE_p_unsigned_long(cPtr, false); + } + + public long size() { + return swigfaissJNI.Uint64Vector_size(swigCPtr, this); + } + + public long at(long n) { + return swigfaissJNI.Uint64Vector_at(swigCPtr, this, n); + } + + public void resize(long n) { + swigfaissJNI.Uint64Vector_resize(swigCPtr, this, n); + } + + public void reserve(long n) { + swigfaissJNI.Uint64Vector_reserve(swigCPtr, this, n); + } + + public void swap(Uint64Vector other) { + swigfaissJNI.Uint64Vector_swap(swigCPtr, this, Uint64Vector.getCPtr(other), other); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/VStackInvertedLists.java b/ann/src/main/java/com/twitter/ann/faiss/swig/VStackInvertedLists.java new file mode 100644 index 0000000000..127a4a9532 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/VStackInvertedLists.java @@ -0,0 +1,95 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class VStackInvertedLists extends ReadOnlyInvertedLists { + private transient long swigCPtr; + + protected VStackInvertedLists(long cPtr, boolean cMemoryOwn) { + super(swigfaissJNI.VStackInvertedLists_SWIGUpcast(cPtr), cMemoryOwn); + swigCPtr = cPtr; + } + + protected static long getCPtr(VStackInvertedLists obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_VStackInvertedLists(swigCPtr); + } + swigCPtr = 0; + } + super.delete(); + } + + public void setIls(SWIGTYPE_p_std__vectorT_faiss__InvertedLists_const_p_t value) { + swigfaissJNI.VStackInvertedLists_ils_set(swigCPtr, this, SWIGTYPE_p_std__vectorT_faiss__InvertedLists_const_p_t.getCPtr(value)); + } + + public SWIGTYPE_p_std__vectorT_faiss__InvertedLists_const_p_t getIls() { + long cPtr = swigfaissJNI.VStackInvertedLists_ils_get(swigCPtr, this); + return (cPtr == 0) ? null : new SWIGTYPE_p_std__vectorT_faiss__InvertedLists_const_p_t(cPtr, false); + } + + public void setCumsz(SWIGTYPE_p_std__vectorT_int64_t_t value) { + swigfaissJNI.VStackInvertedLists_cumsz_set(swigCPtr, this, SWIGTYPE_p_std__vectorT_int64_t_t.getCPtr(value)); + } + + public SWIGTYPE_p_std__vectorT_int64_t_t getCumsz() { + long cPtr = swigfaissJNI.VStackInvertedLists_cumsz_get(swigCPtr, this); + return (cPtr == 0) ? null : new SWIGTYPE_p_std__vectorT_int64_t_t(cPtr, false); + } + + public VStackInvertedLists(int nil, SWIGTYPE_p_p_faiss__InvertedLists ils) { + this(swigfaissJNI.new_VStackInvertedLists(nil, SWIGTYPE_p_p_faiss__InvertedLists.getCPtr(ils)), true); + } + + public long list_size(long list_no) { + return swigfaissJNI.VStackInvertedLists_list_size(swigCPtr, this, list_no); + } + + public SWIGTYPE_p_unsigned_char get_codes(long list_no) { + long cPtr = swigfaissJNI.VStackInvertedLists_get_codes(swigCPtr, this, list_no); + return (cPtr == 0) ? null : new SWIGTYPE_p_unsigned_char(cPtr, false); + } + + public LongVector get_ids(long list_no) { + return new LongVector(swigfaissJNI.VStackInvertedLists_get_ids(swigCPtr, this, list_no), false); +} + + public void release_codes(long list_no, SWIGTYPE_p_unsigned_char codes) { + swigfaissJNI.VStackInvertedLists_release_codes(swigCPtr, this, list_no, SWIGTYPE_p_unsigned_char.getCPtr(codes)); + } + + public void release_ids(long list_no, LongVector ids) { + swigfaissJNI.VStackInvertedLists_release_ids(swigCPtr, this, list_no, SWIGTYPE_p_long_long.getCPtr(ids.data()), ids); + } + + public long get_single_id(long list_no, long offset) { + return swigfaissJNI.VStackInvertedLists_get_single_id(swigCPtr, this, list_no, offset); +} + + public SWIGTYPE_p_unsigned_char get_single_code(long list_no, long offset) { + long cPtr = swigfaissJNI.VStackInvertedLists_get_single_code(swigCPtr, this, list_no, offset); + return (cPtr == 0) ? null : new SWIGTYPE_p_unsigned_char(cPtr, false); + } + + public void prefetch_lists(LongVector list_nos, int nlist) { + swigfaissJNI.VStackInvertedLists_prefetch_lists(swigCPtr, this, SWIGTYPE_p_long_long.getCPtr(list_nos.data()), list_nos, nlist); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/VectorTransform.java b/ann/src/main/java/com/twitter/ann/faiss/swig/VectorTransform.java new file mode 100644 index 0000000000..e17ad0f4d1 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/VectorTransform.java @@ -0,0 +1,80 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class VectorTransform { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected VectorTransform(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(VectorTransform obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_VectorTransform(swigCPtr); + } + swigCPtr = 0; + } + } + + public void setD_in(int value) { + swigfaissJNI.VectorTransform_d_in_set(swigCPtr, this, value); + } + + public int getD_in() { + return swigfaissJNI.VectorTransform_d_in_get(swigCPtr, this); + } + + public void setD_out(int value) { + swigfaissJNI.VectorTransform_d_out_set(swigCPtr, this, value); + } + + public int getD_out() { + return swigfaissJNI.VectorTransform_d_out_get(swigCPtr, this); + } + + public void setIs_trained(boolean value) { + swigfaissJNI.VectorTransform_is_trained_set(swigCPtr, this, value); + } + + public boolean getIs_trained() { + return swigfaissJNI.VectorTransform_is_trained_get(swigCPtr, this); + } + + public void train(long n, SWIGTYPE_p_float x) { + swigfaissJNI.VectorTransform_train(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x)); + } + + public SWIGTYPE_p_float apply(long n, SWIGTYPE_p_float x) { + long cPtr = swigfaissJNI.VectorTransform_apply(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x)); + return (cPtr == 0) ? null : new SWIGTYPE_p_float(cPtr, false); + } + + public void apply_noalloc(long n, SWIGTYPE_p_float x, SWIGTYPE_p_float xt) { + swigfaissJNI.VectorTransform_apply_noalloc(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(x), SWIGTYPE_p_float.getCPtr(xt)); + } + + public void reverse_transform(long n, SWIGTYPE_p_float xt, SWIGTYPE_p_float x) { + swigfaissJNI.VectorTransform_reverse_transform(swigCPtr, this, n, SWIGTYPE_p_float.getCPtr(xt), SWIGTYPE_p_float.getCPtr(x)); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/VectorTransformVector.java b/ann/src/main/java/com/twitter/ann/faiss/swig/VectorTransformVector.java new file mode 100644 index 0000000000..d7a1a6f6f1 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/VectorTransformVector.java @@ -0,0 +1,77 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class VectorTransformVector { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected VectorTransformVector(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(VectorTransformVector obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_VectorTransformVector(swigCPtr); + } + swigCPtr = 0; + } + } + + public VectorTransformVector() { + this(swigfaissJNI.new_VectorTransformVector(), true); + } + + public void push_back(VectorTransform arg0) { + swigfaissJNI.VectorTransformVector_push_back(swigCPtr, this, VectorTransform.getCPtr(arg0), arg0); + } + + public void clear() { + swigfaissJNI.VectorTransformVector_clear(swigCPtr, this); + } + + public SWIGTYPE_p_p_faiss__VectorTransform data() { + long cPtr = swigfaissJNI.VectorTransformVector_data(swigCPtr, this); + return (cPtr == 0) ? null : new SWIGTYPE_p_p_faiss__VectorTransform(cPtr, false); + } + + public long size() { + return swigfaissJNI.VectorTransformVector_size(swigCPtr, this); + } + + public VectorTransform at(long n) { + long cPtr = swigfaissJNI.VectorTransformVector_at(swigCPtr, this, n); + return (cPtr == 0) ? null : new VectorTransform(cPtr, false); + } + + public void resize(long n) { + swigfaissJNI.VectorTransformVector_resize(swigCPtr, this, n); + } + + public void reserve(long n) { + swigfaissJNI.VectorTransformVector_reserve(swigCPtr, this, n); + } + + public void swap(VectorTransformVector other) { + swigfaissJNI.VectorTransformVector_swap(swigCPtr, this, VectorTransformVector.getCPtr(other), other); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/VisitedTable.java b/ann/src/main/java/com/twitter/ann/faiss/swig/VisitedTable.java new file mode 100644 index 0000000000..69b1108a4b --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/VisitedTable.java @@ -0,0 +1,72 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class VisitedTable { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected VisitedTable(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(VisitedTable obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_VisitedTable(swigCPtr); + } + swigCPtr = 0; + } + } + + public void setVisited(ByteVector value) { + swigfaissJNI.VisitedTable_visited_set(swigCPtr, this, ByteVector.getCPtr(value), value); + } + + public ByteVector getVisited() { + long cPtr = swigfaissJNI.VisitedTable_visited_get(swigCPtr, this); + return (cPtr == 0) ? null : new ByteVector(cPtr, false); + } + + public void setVisno(int value) { + swigfaissJNI.VisitedTable_visno_set(swigCPtr, this, value); + } + + public int getVisno() { + return swigfaissJNI.VisitedTable_visno_get(swigCPtr, this); + } + + public VisitedTable(int size) { + this(swigfaissJNI.new_VisitedTable(size), true); + } + + public void set(int no) { + swigfaissJNI.VisitedTable_set(swigCPtr, this, no); + } + + public boolean get(int no) { + return swigfaissJNI.VisitedTable_get(swigCPtr, this, no); + } + + public void advance() { + swigfaissJNI.VisitedTable_advance(swigCPtr, this); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/doubleArray.java b/ann/src/main/java/com/twitter/ann/faiss/swig/doubleArray.java new file mode 100644 index 0000000000..f37b56e4a7 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/doubleArray.java @@ -0,0 +1,61 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class doubleArray { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected doubleArray(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(doubleArray obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_doubleArray(swigCPtr); + } + swigCPtr = 0; + } + } + + public doubleArray(int nelements) { + this(swigfaissJNI.new_doubleArray(nelements), true); + } + + public double getitem(int index) { + return swigfaissJNI.doubleArray_getitem(swigCPtr, this, index); + } + + public void setitem(int index, double value) { + swigfaissJNI.doubleArray_setitem(swigCPtr, this, index, value); + } + + public SWIGTYPE_p_double cast() { + long cPtr = swigfaissJNI.doubleArray_cast(swigCPtr, this); + return (cPtr == 0) ? null : new SWIGTYPE_p_double(cPtr, false); + } + + public static doubleArray frompointer(SWIGTYPE_p_double t) { + long cPtr = swigfaissJNI.doubleArray_frompointer(SWIGTYPE_p_double.getCPtr(t)); + return (cPtr == 0) ? null : new doubleArray(cPtr, false); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/floatArray.java b/ann/src/main/java/com/twitter/ann/faiss/swig/floatArray.java new file mode 100644 index 0000000000..641dc844b3 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/floatArray.java @@ -0,0 +1,61 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class floatArray { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected floatArray(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(floatArray obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_floatArray(swigCPtr); + } + swigCPtr = 0; + } + } + + public floatArray(int nelements) { + this(swigfaissJNI.new_floatArray(nelements), true); + } + + public float getitem(int index) { + return swigfaissJNI.floatArray_getitem(swigCPtr, this, index); + } + + public void setitem(int index, float value) { + swigfaissJNI.floatArray_setitem(swigCPtr, this, index, value); + } + + public SWIGTYPE_p_float cast() { + long cPtr = swigfaissJNI.floatArray_cast(swigCPtr, this); + return (cPtr == 0) ? null : new SWIGTYPE_p_float(cPtr, false); + } + + public static floatArray frompointer(SWIGTYPE_p_float t) { + long cPtr = swigfaissJNI.floatArray_frompointer(SWIGTYPE_p_float.getCPtr(t)); + return (cPtr == 0) ? null : new floatArray(cPtr, false); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/float_maxheap_array_t.java b/ann/src/main/java/com/twitter/ann/faiss/swig/float_maxheap_array_t.java new file mode 100644 index 0000000000..875ef1c4a5 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/float_maxheap_array_t.java @@ -0,0 +1,133 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class float_maxheap_array_t { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected float_maxheap_array_t(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(float_maxheap_array_t obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_float_maxheap_array_t(swigCPtr); + } + swigCPtr = 0; + } + } + + public void setNh(long value) { + swigfaissJNI.float_maxheap_array_t_nh_set(swigCPtr, this, value); + } + + public long getNh() { + return swigfaissJNI.float_maxheap_array_t_nh_get(swigCPtr, this); + } + + public void setK(long value) { + swigfaissJNI.float_maxheap_array_t_k_set(swigCPtr, this, value); + } + + public long getK() { + return swigfaissJNI.float_maxheap_array_t_k_get(swigCPtr, this); + } + + public void setIds(LongVector value) { + swigfaissJNI.float_maxheap_array_t_ids_set(swigCPtr, this, SWIGTYPE_p_long_long.getCPtr(value.data()), value); + } + + public LongVector getIds() { + return new LongVector(swigfaissJNI.float_maxheap_array_t_ids_get(swigCPtr, this), false); +} + + public void setVal(SWIGTYPE_p_float value) { + swigfaissJNI.float_maxheap_array_t_val_set(swigCPtr, this, SWIGTYPE_p_float.getCPtr(value)); + } + + public SWIGTYPE_p_float getVal() { + long cPtr = swigfaissJNI.float_maxheap_array_t_val_get(swigCPtr, this); + return (cPtr == 0) ? null : new SWIGTYPE_p_float(cPtr, false); + } + + public SWIGTYPE_p_float get_val(long key) { + long cPtr = swigfaissJNI.float_maxheap_array_t_get_val(swigCPtr, this, key); + return (cPtr == 0) ? null : new SWIGTYPE_p_float(cPtr, false); + } + + public LongVector get_ids(long key) { + return new LongVector(swigfaissJNI.float_maxheap_array_t_get_ids(swigCPtr, this, key), false); +} + + public void heapify() { + swigfaissJNI.float_maxheap_array_t_heapify(swigCPtr, this); + } + + public void addn(long nj, SWIGTYPE_p_float vin, long j0, long i0, long ni) { + swigfaissJNI.float_maxheap_array_t_addn__SWIG_0(swigCPtr, this, nj, SWIGTYPE_p_float.getCPtr(vin), j0, i0, ni); + } + + public void addn(long nj, SWIGTYPE_p_float vin, long j0, long i0) { + swigfaissJNI.float_maxheap_array_t_addn__SWIG_1(swigCPtr, this, nj, SWIGTYPE_p_float.getCPtr(vin), j0, i0); + } + + public void addn(long nj, SWIGTYPE_p_float vin, long j0) { + swigfaissJNI.float_maxheap_array_t_addn__SWIG_2(swigCPtr, this, nj, SWIGTYPE_p_float.getCPtr(vin), j0); + } + + public void addn(long nj, SWIGTYPE_p_float vin) { + swigfaissJNI.float_maxheap_array_t_addn__SWIG_3(swigCPtr, this, nj, SWIGTYPE_p_float.getCPtr(vin)); + } + + public void addn_with_ids(long nj, SWIGTYPE_p_float vin, LongVector id_in, long id_stride, long i0, long ni) { + swigfaissJNI.float_maxheap_array_t_addn_with_ids__SWIG_0(swigCPtr, this, nj, SWIGTYPE_p_float.getCPtr(vin), SWIGTYPE_p_long_long.getCPtr(id_in.data()), id_in, id_stride, i0, ni); + } + + public void addn_with_ids(long nj, SWIGTYPE_p_float vin, LongVector id_in, long id_stride, long i0) { + swigfaissJNI.float_maxheap_array_t_addn_with_ids__SWIG_1(swigCPtr, this, nj, SWIGTYPE_p_float.getCPtr(vin), SWIGTYPE_p_long_long.getCPtr(id_in.data()), id_in, id_stride, i0); + } + + public void addn_with_ids(long nj, SWIGTYPE_p_float vin, LongVector id_in, long id_stride) { + swigfaissJNI.float_maxheap_array_t_addn_with_ids__SWIG_2(swigCPtr, this, nj, SWIGTYPE_p_float.getCPtr(vin), SWIGTYPE_p_long_long.getCPtr(id_in.data()), id_in, id_stride); + } + + public void addn_with_ids(long nj, SWIGTYPE_p_float vin, LongVector id_in) { + swigfaissJNI.float_maxheap_array_t_addn_with_ids__SWIG_3(swigCPtr, this, nj, SWIGTYPE_p_float.getCPtr(vin), SWIGTYPE_p_long_long.getCPtr(id_in.data()), id_in); + } + + public void addn_with_ids(long nj, SWIGTYPE_p_float vin) { + swigfaissJNI.float_maxheap_array_t_addn_with_ids__SWIG_4(swigCPtr, this, nj, SWIGTYPE_p_float.getCPtr(vin)); + } + + public void reorder() { + swigfaissJNI.float_maxheap_array_t_reorder(swigCPtr, this); + } + + public void per_line_extrema(SWIGTYPE_p_float vals_out, LongVector idx_out) { + swigfaissJNI.float_maxheap_array_t_per_line_extrema(swigCPtr, this, SWIGTYPE_p_float.getCPtr(vals_out), SWIGTYPE_p_long_long.getCPtr(idx_out.data()), idx_out); + } + + public float_maxheap_array_t() { + this(swigfaissJNI.new_float_maxheap_array_t(), true); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/float_minheap_array_t.java b/ann/src/main/java/com/twitter/ann/faiss/swig/float_minheap_array_t.java new file mode 100644 index 0000000000..5944f2cc9c --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/float_minheap_array_t.java @@ -0,0 +1,133 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class float_minheap_array_t { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected float_minheap_array_t(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(float_minheap_array_t obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_float_minheap_array_t(swigCPtr); + } + swigCPtr = 0; + } + } + + public void setNh(long value) { + swigfaissJNI.float_minheap_array_t_nh_set(swigCPtr, this, value); + } + + public long getNh() { + return swigfaissJNI.float_minheap_array_t_nh_get(swigCPtr, this); + } + + public void setK(long value) { + swigfaissJNI.float_minheap_array_t_k_set(swigCPtr, this, value); + } + + public long getK() { + return swigfaissJNI.float_minheap_array_t_k_get(swigCPtr, this); + } + + public void setIds(LongVector value) { + swigfaissJNI.float_minheap_array_t_ids_set(swigCPtr, this, SWIGTYPE_p_long_long.getCPtr(value.data()), value); + } + + public LongVector getIds() { + return new LongVector(swigfaissJNI.float_minheap_array_t_ids_get(swigCPtr, this), false); +} + + public void setVal(SWIGTYPE_p_float value) { + swigfaissJNI.float_minheap_array_t_val_set(swigCPtr, this, SWIGTYPE_p_float.getCPtr(value)); + } + + public SWIGTYPE_p_float getVal() { + long cPtr = swigfaissJNI.float_minheap_array_t_val_get(swigCPtr, this); + return (cPtr == 0) ? null : new SWIGTYPE_p_float(cPtr, false); + } + + public SWIGTYPE_p_float get_val(long key) { + long cPtr = swigfaissJNI.float_minheap_array_t_get_val(swigCPtr, this, key); + return (cPtr == 0) ? null : new SWIGTYPE_p_float(cPtr, false); + } + + public LongVector get_ids(long key) { + return new LongVector(swigfaissJNI.float_minheap_array_t_get_ids(swigCPtr, this, key), false); +} + + public void heapify() { + swigfaissJNI.float_minheap_array_t_heapify(swigCPtr, this); + } + + public void addn(long nj, SWIGTYPE_p_float vin, long j0, long i0, long ni) { + swigfaissJNI.float_minheap_array_t_addn__SWIG_0(swigCPtr, this, nj, SWIGTYPE_p_float.getCPtr(vin), j0, i0, ni); + } + + public void addn(long nj, SWIGTYPE_p_float vin, long j0, long i0) { + swigfaissJNI.float_minheap_array_t_addn__SWIG_1(swigCPtr, this, nj, SWIGTYPE_p_float.getCPtr(vin), j0, i0); + } + + public void addn(long nj, SWIGTYPE_p_float vin, long j0) { + swigfaissJNI.float_minheap_array_t_addn__SWIG_2(swigCPtr, this, nj, SWIGTYPE_p_float.getCPtr(vin), j0); + } + + public void addn(long nj, SWIGTYPE_p_float vin) { + swigfaissJNI.float_minheap_array_t_addn__SWIG_3(swigCPtr, this, nj, SWIGTYPE_p_float.getCPtr(vin)); + } + + public void addn_with_ids(long nj, SWIGTYPE_p_float vin, LongVector id_in, long id_stride, long i0, long ni) { + swigfaissJNI.float_minheap_array_t_addn_with_ids__SWIG_0(swigCPtr, this, nj, SWIGTYPE_p_float.getCPtr(vin), SWIGTYPE_p_long_long.getCPtr(id_in.data()), id_in, id_stride, i0, ni); + } + + public void addn_with_ids(long nj, SWIGTYPE_p_float vin, LongVector id_in, long id_stride, long i0) { + swigfaissJNI.float_minheap_array_t_addn_with_ids__SWIG_1(swigCPtr, this, nj, SWIGTYPE_p_float.getCPtr(vin), SWIGTYPE_p_long_long.getCPtr(id_in.data()), id_in, id_stride, i0); + } + + public void addn_with_ids(long nj, SWIGTYPE_p_float vin, LongVector id_in, long id_stride) { + swigfaissJNI.float_minheap_array_t_addn_with_ids__SWIG_2(swigCPtr, this, nj, SWIGTYPE_p_float.getCPtr(vin), SWIGTYPE_p_long_long.getCPtr(id_in.data()), id_in, id_stride); + } + + public void addn_with_ids(long nj, SWIGTYPE_p_float vin, LongVector id_in) { + swigfaissJNI.float_minheap_array_t_addn_with_ids__SWIG_3(swigCPtr, this, nj, SWIGTYPE_p_float.getCPtr(vin), SWIGTYPE_p_long_long.getCPtr(id_in.data()), id_in); + } + + public void addn_with_ids(long nj, SWIGTYPE_p_float vin) { + swigfaissJNI.float_minheap_array_t_addn_with_ids__SWIG_4(swigCPtr, this, nj, SWIGTYPE_p_float.getCPtr(vin)); + } + + public void reorder() { + swigfaissJNI.float_minheap_array_t_reorder(swigCPtr, this); + } + + public void per_line_extrema(SWIGTYPE_p_float vals_out, LongVector idx_out) { + swigfaissJNI.float_minheap_array_t_per_line_extrema(swigCPtr, this, SWIGTYPE_p_float.getCPtr(vals_out), SWIGTYPE_p_long_long.getCPtr(idx_out.data()), idx_out); + } + + public float_minheap_array_t() { + this(swigfaissJNI.new_float_minheap_array_t(), true); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/intArray.java b/ann/src/main/java/com/twitter/ann/faiss/swig/intArray.java new file mode 100644 index 0000000000..9774521a10 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/intArray.java @@ -0,0 +1,61 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class intArray { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected intArray(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(intArray obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_intArray(swigCPtr); + } + swigCPtr = 0; + } + } + + public intArray(int nelements) { + this(swigfaissJNI.new_intArray(nelements), true); + } + + public int getitem(int index) { + return swigfaissJNI.intArray_getitem(swigCPtr, this, index); + } + + public void setitem(int index, int value) { + swigfaissJNI.intArray_setitem(swigCPtr, this, index, value); + } + + public SWIGTYPE_p_int cast() { + long cPtr = swigfaissJNI.intArray_cast(swigCPtr, this); + return (cPtr == 0) ? null : new SWIGTYPE_p_int(cPtr, false); + } + + public static intArray frompointer(SWIGTYPE_p_int t) { + long cPtr = swigfaissJNI.intArray_frompointer(SWIGTYPE_p_int.getCPtr(t)); + return (cPtr == 0) ? null : new intArray(cPtr, false); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/int_maxheap_array_t.java b/ann/src/main/java/com/twitter/ann/faiss/swig/int_maxheap_array_t.java new file mode 100644 index 0000000000..07bf7a570a --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/int_maxheap_array_t.java @@ -0,0 +1,133 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class int_maxheap_array_t { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected int_maxheap_array_t(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(int_maxheap_array_t obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_int_maxheap_array_t(swigCPtr); + } + swigCPtr = 0; + } + } + + public void setNh(long value) { + swigfaissJNI.int_maxheap_array_t_nh_set(swigCPtr, this, value); + } + + public long getNh() { + return swigfaissJNI.int_maxheap_array_t_nh_get(swigCPtr, this); + } + + public void setK(long value) { + swigfaissJNI.int_maxheap_array_t_k_set(swigCPtr, this, value); + } + + public long getK() { + return swigfaissJNI.int_maxheap_array_t_k_get(swigCPtr, this); + } + + public void setIds(LongVector value) { + swigfaissJNI.int_maxheap_array_t_ids_set(swigCPtr, this, SWIGTYPE_p_long_long.getCPtr(value.data()), value); + } + + public LongVector getIds() { + return new LongVector(swigfaissJNI.int_maxheap_array_t_ids_get(swigCPtr, this), false); +} + + public void setVal(SWIGTYPE_p_int value) { + swigfaissJNI.int_maxheap_array_t_val_set(swigCPtr, this, SWIGTYPE_p_int.getCPtr(value)); + } + + public SWIGTYPE_p_int getVal() { + long cPtr = swigfaissJNI.int_maxheap_array_t_val_get(swigCPtr, this); + return (cPtr == 0) ? null : new SWIGTYPE_p_int(cPtr, false); + } + + public SWIGTYPE_p_int get_val(long key) { + long cPtr = swigfaissJNI.int_maxheap_array_t_get_val(swigCPtr, this, key); + return (cPtr == 0) ? null : new SWIGTYPE_p_int(cPtr, false); + } + + public LongVector get_ids(long key) { + return new LongVector(swigfaissJNI.int_maxheap_array_t_get_ids(swigCPtr, this, key), false); +} + + public void heapify() { + swigfaissJNI.int_maxheap_array_t_heapify(swigCPtr, this); + } + + public void addn(long nj, SWIGTYPE_p_int vin, long j0, long i0, long ni) { + swigfaissJNI.int_maxheap_array_t_addn__SWIG_0(swigCPtr, this, nj, SWIGTYPE_p_int.getCPtr(vin), j0, i0, ni); + } + + public void addn(long nj, SWIGTYPE_p_int vin, long j0, long i0) { + swigfaissJNI.int_maxheap_array_t_addn__SWIG_1(swigCPtr, this, nj, SWIGTYPE_p_int.getCPtr(vin), j0, i0); + } + + public void addn(long nj, SWIGTYPE_p_int vin, long j0) { + swigfaissJNI.int_maxheap_array_t_addn__SWIG_2(swigCPtr, this, nj, SWIGTYPE_p_int.getCPtr(vin), j0); + } + + public void addn(long nj, SWIGTYPE_p_int vin) { + swigfaissJNI.int_maxheap_array_t_addn__SWIG_3(swigCPtr, this, nj, SWIGTYPE_p_int.getCPtr(vin)); + } + + public void addn_with_ids(long nj, SWIGTYPE_p_int vin, LongVector id_in, long id_stride, long i0, long ni) { + swigfaissJNI.int_maxheap_array_t_addn_with_ids__SWIG_0(swigCPtr, this, nj, SWIGTYPE_p_int.getCPtr(vin), SWIGTYPE_p_long_long.getCPtr(id_in.data()), id_in, id_stride, i0, ni); + } + + public void addn_with_ids(long nj, SWIGTYPE_p_int vin, LongVector id_in, long id_stride, long i0) { + swigfaissJNI.int_maxheap_array_t_addn_with_ids__SWIG_1(swigCPtr, this, nj, SWIGTYPE_p_int.getCPtr(vin), SWIGTYPE_p_long_long.getCPtr(id_in.data()), id_in, id_stride, i0); + } + + public void addn_with_ids(long nj, SWIGTYPE_p_int vin, LongVector id_in, long id_stride) { + swigfaissJNI.int_maxheap_array_t_addn_with_ids__SWIG_2(swigCPtr, this, nj, SWIGTYPE_p_int.getCPtr(vin), SWIGTYPE_p_long_long.getCPtr(id_in.data()), id_in, id_stride); + } + + public void addn_with_ids(long nj, SWIGTYPE_p_int vin, LongVector id_in) { + swigfaissJNI.int_maxheap_array_t_addn_with_ids__SWIG_3(swigCPtr, this, nj, SWIGTYPE_p_int.getCPtr(vin), SWIGTYPE_p_long_long.getCPtr(id_in.data()), id_in); + } + + public void addn_with_ids(long nj, SWIGTYPE_p_int vin) { + swigfaissJNI.int_maxheap_array_t_addn_with_ids__SWIG_4(swigCPtr, this, nj, SWIGTYPE_p_int.getCPtr(vin)); + } + + public void reorder() { + swigfaissJNI.int_maxheap_array_t_reorder(swigCPtr, this); + } + + public void per_line_extrema(SWIGTYPE_p_int vals_out, LongVector idx_out) { + swigfaissJNI.int_maxheap_array_t_per_line_extrema(swigCPtr, this, SWIGTYPE_p_int.getCPtr(vals_out), SWIGTYPE_p_long_long.getCPtr(idx_out.data()), idx_out); + } + + public int_maxheap_array_t() { + this(swigfaissJNI.new_int_maxheap_array_t(), true); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/int_minheap_array_t.java b/ann/src/main/java/com/twitter/ann/faiss/swig/int_minheap_array_t.java new file mode 100644 index 0000000000..0a60648761 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/int_minheap_array_t.java @@ -0,0 +1,133 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class int_minheap_array_t { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected int_minheap_array_t(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(int_minheap_array_t obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_int_minheap_array_t(swigCPtr); + } + swigCPtr = 0; + } + } + + public void setNh(long value) { + swigfaissJNI.int_minheap_array_t_nh_set(swigCPtr, this, value); + } + + public long getNh() { + return swigfaissJNI.int_minheap_array_t_nh_get(swigCPtr, this); + } + + public void setK(long value) { + swigfaissJNI.int_minheap_array_t_k_set(swigCPtr, this, value); + } + + public long getK() { + return swigfaissJNI.int_minheap_array_t_k_get(swigCPtr, this); + } + + public void setIds(LongVector value) { + swigfaissJNI.int_minheap_array_t_ids_set(swigCPtr, this, SWIGTYPE_p_long_long.getCPtr(value.data()), value); + } + + public LongVector getIds() { + return new LongVector(swigfaissJNI.int_minheap_array_t_ids_get(swigCPtr, this), false); +} + + public void setVal(SWIGTYPE_p_int value) { + swigfaissJNI.int_minheap_array_t_val_set(swigCPtr, this, SWIGTYPE_p_int.getCPtr(value)); + } + + public SWIGTYPE_p_int getVal() { + long cPtr = swigfaissJNI.int_minheap_array_t_val_get(swigCPtr, this); + return (cPtr == 0) ? null : new SWIGTYPE_p_int(cPtr, false); + } + + public SWIGTYPE_p_int get_val(long key) { + long cPtr = swigfaissJNI.int_minheap_array_t_get_val(swigCPtr, this, key); + return (cPtr == 0) ? null : new SWIGTYPE_p_int(cPtr, false); + } + + public LongVector get_ids(long key) { + return new LongVector(swigfaissJNI.int_minheap_array_t_get_ids(swigCPtr, this, key), false); +} + + public void heapify() { + swigfaissJNI.int_minheap_array_t_heapify(swigCPtr, this); + } + + public void addn(long nj, SWIGTYPE_p_int vin, long j0, long i0, long ni) { + swigfaissJNI.int_minheap_array_t_addn__SWIG_0(swigCPtr, this, nj, SWIGTYPE_p_int.getCPtr(vin), j0, i0, ni); + } + + public void addn(long nj, SWIGTYPE_p_int vin, long j0, long i0) { + swigfaissJNI.int_minheap_array_t_addn__SWIG_1(swigCPtr, this, nj, SWIGTYPE_p_int.getCPtr(vin), j0, i0); + } + + public void addn(long nj, SWIGTYPE_p_int vin, long j0) { + swigfaissJNI.int_minheap_array_t_addn__SWIG_2(swigCPtr, this, nj, SWIGTYPE_p_int.getCPtr(vin), j0); + } + + public void addn(long nj, SWIGTYPE_p_int vin) { + swigfaissJNI.int_minheap_array_t_addn__SWIG_3(swigCPtr, this, nj, SWIGTYPE_p_int.getCPtr(vin)); + } + + public void addn_with_ids(long nj, SWIGTYPE_p_int vin, LongVector id_in, long id_stride, long i0, long ni) { + swigfaissJNI.int_minheap_array_t_addn_with_ids__SWIG_0(swigCPtr, this, nj, SWIGTYPE_p_int.getCPtr(vin), SWIGTYPE_p_long_long.getCPtr(id_in.data()), id_in, id_stride, i0, ni); + } + + public void addn_with_ids(long nj, SWIGTYPE_p_int vin, LongVector id_in, long id_stride, long i0) { + swigfaissJNI.int_minheap_array_t_addn_with_ids__SWIG_1(swigCPtr, this, nj, SWIGTYPE_p_int.getCPtr(vin), SWIGTYPE_p_long_long.getCPtr(id_in.data()), id_in, id_stride, i0); + } + + public void addn_with_ids(long nj, SWIGTYPE_p_int vin, LongVector id_in, long id_stride) { + swigfaissJNI.int_minheap_array_t_addn_with_ids__SWIG_2(swigCPtr, this, nj, SWIGTYPE_p_int.getCPtr(vin), SWIGTYPE_p_long_long.getCPtr(id_in.data()), id_in, id_stride); + } + + public void addn_with_ids(long nj, SWIGTYPE_p_int vin, LongVector id_in) { + swigfaissJNI.int_minheap_array_t_addn_with_ids__SWIG_3(swigCPtr, this, nj, SWIGTYPE_p_int.getCPtr(vin), SWIGTYPE_p_long_long.getCPtr(id_in.data()), id_in); + } + + public void addn_with_ids(long nj, SWIGTYPE_p_int vin) { + swigfaissJNI.int_minheap_array_t_addn_with_ids__SWIG_4(swigCPtr, this, nj, SWIGTYPE_p_int.getCPtr(vin)); + } + + public void reorder() { + swigfaissJNI.int_minheap_array_t_reorder(swigCPtr, this); + } + + public void per_line_extrema(SWIGTYPE_p_int vals_out, LongVector idx_out) { + swigfaissJNI.int_minheap_array_t_per_line_extrema(swigCPtr, this, SWIGTYPE_p_int.getCPtr(vals_out), SWIGTYPE_p_long_long.getCPtr(idx_out.data()), idx_out); + } + + public int_minheap_array_t() { + this(swigfaissJNI.new_int_minheap_array_t(), true); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/longArray.java b/ann/src/main/java/com/twitter/ann/faiss/swig/longArray.java new file mode 100644 index 0000000000..6d51d8faeb --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/longArray.java @@ -0,0 +1,61 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class longArray { + private transient long swigCPtr; + protected transient boolean swigCMemOwn; + + protected longArray(long cPtr, boolean cMemoryOwn) { + swigCMemOwn = cMemoryOwn; + swigCPtr = cPtr; + } + + protected static long getCPtr(longArray obj) { + return (obj == null) ? 0 : obj.swigCPtr; + } + + @SuppressWarnings("deprecation") + protected void finalize() { + delete(); + } + + public synchronized void delete() { + if (swigCPtr != 0) { + if (swigCMemOwn) { + swigCMemOwn = false; + swigfaissJNI.delete_longArray(swigCPtr); + } + swigCPtr = 0; + } + } + + public longArray(int nelements) { + this(swigfaissJNI.new_longArray(nelements), true); + } + + public long getitem(int index) { + return swigfaissJNI.longArray_getitem(swigCPtr, this, index); + } + + public void setitem(int index, long value) { + swigfaissJNI.longArray_setitem(swigCPtr, this, index, value); + } + + public SWIGTYPE_p_long_long cast() { + long cPtr = swigfaissJNI.longArray_cast(swigCPtr, this); + return (cPtr == 0) ? null : new SWIGTYPE_p_long_long(cPtr, false); + } + + public static longArray frompointer(SWIGTYPE_p_long_long t) { + long cPtr = swigfaissJNI.longArray_frompointer(SWIGTYPE_p_long_long.getCPtr(t)); + return (cPtr == 0) ? null : new longArray(cPtr, false); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/resources/.gitignore b/ann/src/main/java/com/twitter/ann/faiss/swig/resources/.gitignore new file mode 100644 index 0000000000..caf63c285d --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/resources/.gitignore @@ -0,0 +1,7 @@ +*.so +*.so.0 +*.so.1 +*.so.3 +*.so.5 +*.so.6 +*.dylib diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/resources/.gitkeep b/ann/src/main/java/com/twitter/ann/faiss/swig/resources/.gitkeep new file mode 100644 index 0000000000..e69de29bb2 diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/resources/BUILD b/ann/src/main/java/com/twitter/ann/faiss/swig/resources/BUILD new file mode 100644 index 0000000000..58ba4703b4 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/resources/BUILD @@ -0,0 +1,17 @@ +resources( + name = "resources", + sources = [ + "*.dylib", + "*.so", + "*.so.0", + "*.so.1", + "*.so.3", + "*.so.5", + "*.so.6", + ], + tags = [ + "bazel-compatible", + "bazel-only", + "visibility://visibility:private", + ], +) diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/swigfaiss.java b/ann/src/main/java/com/twitter/ann/faiss/swig/swigfaiss.java new file mode 100644 index 0000000000..d55793a697 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/swigfaiss.java @@ -0,0 +1,575 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public class swigfaiss implements swigfaissConstants { + public static void bitvec_print(SWIGTYPE_p_unsigned_char b, long d) { + swigfaissJNI.bitvec_print(SWIGTYPE_p_unsigned_char.getCPtr(b), d); + } + + public static void fvecs2bitvecs(SWIGTYPE_p_float x, SWIGTYPE_p_unsigned_char b, long d, long n) { + swigfaissJNI.fvecs2bitvecs(SWIGTYPE_p_float.getCPtr(x), SWIGTYPE_p_unsigned_char.getCPtr(b), d, n); + } + + public static void bitvecs2fvecs(SWIGTYPE_p_unsigned_char b, SWIGTYPE_p_float x, long d, long n) { + swigfaissJNI.bitvecs2fvecs(SWIGTYPE_p_unsigned_char.getCPtr(b), SWIGTYPE_p_float.getCPtr(x), d, n); + } + + public static void fvec2bitvec(SWIGTYPE_p_float x, SWIGTYPE_p_unsigned_char b, long d) { + swigfaissJNI.fvec2bitvec(SWIGTYPE_p_float.getCPtr(x), SWIGTYPE_p_unsigned_char.getCPtr(b), d); + } + + public static void bitvec_shuffle(long n, long da, long db, SWIGTYPE_p_int order, SWIGTYPE_p_unsigned_char a, SWIGTYPE_p_unsigned_char b) { + swigfaissJNI.bitvec_shuffle(n, da, db, SWIGTYPE_p_int.getCPtr(order), SWIGTYPE_p_unsigned_char.getCPtr(a), SWIGTYPE_p_unsigned_char.getCPtr(b)); + } + + public static void setHamming_batch_size(long value) { + swigfaissJNI.hamming_batch_size_set(value); + } + + public static long getHamming_batch_size() { + return swigfaissJNI.hamming_batch_size_get(); + } + + public static int popcount64(long x) { + return swigfaissJNI.popcount64(x); + } + + public static void hammings(SWIGTYPE_p_unsigned_char a, SWIGTYPE_p_unsigned_char b, long na, long nb, long nbytespercode, SWIGTYPE_p_int dis) { + swigfaissJNI.hammings(SWIGTYPE_p_unsigned_char.getCPtr(a), SWIGTYPE_p_unsigned_char.getCPtr(b), na, nb, nbytespercode, SWIGTYPE_p_int.getCPtr(dis)); + } + + public static void hammings_knn_hc(SWIGTYPE_p_faiss__HeapArrayT_faiss__CMaxT_int_int64_t_t_t ha, SWIGTYPE_p_unsigned_char a, SWIGTYPE_p_unsigned_char b, long nb, long ncodes, int ordered) { + swigfaissJNI.hammings_knn_hc(SWIGTYPE_p_faiss__HeapArrayT_faiss__CMaxT_int_int64_t_t_t.getCPtr(ha), SWIGTYPE_p_unsigned_char.getCPtr(a), SWIGTYPE_p_unsigned_char.getCPtr(b), nb, ncodes, ordered); + } + + public static void hammings_knn(SWIGTYPE_p_faiss__HeapArrayT_faiss__CMaxT_int_int64_t_t_t ha, SWIGTYPE_p_unsigned_char a, SWIGTYPE_p_unsigned_char b, long nb, long ncodes, int ordered) { + swigfaissJNI.hammings_knn(SWIGTYPE_p_faiss__HeapArrayT_faiss__CMaxT_int_int64_t_t_t.getCPtr(ha), SWIGTYPE_p_unsigned_char.getCPtr(a), SWIGTYPE_p_unsigned_char.getCPtr(b), nb, ncodes, ordered); + } + + public static void hammings_knn_mc(SWIGTYPE_p_unsigned_char a, SWIGTYPE_p_unsigned_char b, long na, long nb, long k, long ncodes, SWIGTYPE_p_int distances, LongVector labels) { + swigfaissJNI.hammings_knn_mc(SWIGTYPE_p_unsigned_char.getCPtr(a), SWIGTYPE_p_unsigned_char.getCPtr(b), na, nb, k, ncodes, SWIGTYPE_p_int.getCPtr(distances), SWIGTYPE_p_long_long.getCPtr(labels.data()), labels); + } + + public static void hamming_range_search(SWIGTYPE_p_unsigned_char a, SWIGTYPE_p_unsigned_char b, long na, long nb, int radius, long ncodes, RangeSearchResult result) { + swigfaissJNI.hamming_range_search(SWIGTYPE_p_unsigned_char.getCPtr(a), SWIGTYPE_p_unsigned_char.getCPtr(b), na, nb, radius, ncodes, RangeSearchResult.getCPtr(result), result); + } + + public static void hamming_count_thres(SWIGTYPE_p_unsigned_char bs1, SWIGTYPE_p_unsigned_char bs2, long n1, long n2, int ht, long ncodes, SWIGTYPE_p_unsigned_long nptr) { + swigfaissJNI.hamming_count_thres(SWIGTYPE_p_unsigned_char.getCPtr(bs1), SWIGTYPE_p_unsigned_char.getCPtr(bs2), n1, n2, ht, ncodes, SWIGTYPE_p_unsigned_long.getCPtr(nptr)); + } + + public static long match_hamming_thres(SWIGTYPE_p_unsigned_char bs1, SWIGTYPE_p_unsigned_char bs2, long n1, long n2, int ht, long ncodes, LongVector idx, SWIGTYPE_p_int dis) { + return swigfaissJNI.match_hamming_thres(SWIGTYPE_p_unsigned_char.getCPtr(bs1), SWIGTYPE_p_unsigned_char.getCPtr(bs2), n1, n2, ht, ncodes, SWIGTYPE_p_long_long.getCPtr(idx.data()), idx, SWIGTYPE_p_int.getCPtr(dis)); + } + + public static void crosshamming_count_thres(SWIGTYPE_p_unsigned_char dbs, long n, int ht, long ncodes, SWIGTYPE_p_unsigned_long nptr) { + swigfaissJNI.crosshamming_count_thres(SWIGTYPE_p_unsigned_char.getCPtr(dbs), n, ht, ncodes, SWIGTYPE_p_unsigned_long.getCPtr(nptr)); + } + + public static int get_num_gpus() { + return swigfaissJNI.get_num_gpus(); + } + + public static String get_compile_options() { + return swigfaissJNI.get_compile_options(); + } + + public static double getmillisecs() { + return swigfaissJNI.getmillisecs(); + } + + public static long get_mem_usage_kb() { + return swigfaissJNI.get_mem_usage_kb(); + } + + public static long get_cycles() { + return swigfaissJNI.get_cycles(); + } + + public static void fvec_madd(long n, SWIGTYPE_p_float a, float bf, SWIGTYPE_p_float b, SWIGTYPE_p_float c) { + swigfaissJNI.fvec_madd(n, SWIGTYPE_p_float.getCPtr(a), bf, SWIGTYPE_p_float.getCPtr(b), SWIGTYPE_p_float.getCPtr(c)); + } + + public static int fvec_madd_and_argmin(long n, SWIGTYPE_p_float a, float bf, SWIGTYPE_p_float b, SWIGTYPE_p_float c) { + return swigfaissJNI.fvec_madd_and_argmin(n, SWIGTYPE_p_float.getCPtr(a), bf, SWIGTYPE_p_float.getCPtr(b), SWIGTYPE_p_float.getCPtr(c)); + } + + public static void reflection(SWIGTYPE_p_float u, SWIGTYPE_p_float x, long n, long d, long nu) { + swigfaissJNI.reflection(SWIGTYPE_p_float.getCPtr(u), SWIGTYPE_p_float.getCPtr(x), n, d, nu); + } + + public static void matrix_qr(int m, int n, SWIGTYPE_p_float a) { + swigfaissJNI.matrix_qr(m, n, SWIGTYPE_p_float.getCPtr(a)); + } + + public static void ranklist_handle_ties(int k, LongVector idx, SWIGTYPE_p_float dis) { + swigfaissJNI.ranklist_handle_ties(k, SWIGTYPE_p_long_long.getCPtr(idx.data()), idx, SWIGTYPE_p_float.getCPtr(dis)); + } + + public static long ranklist_intersection_size(long k1, LongVector v1, long k2, LongVector v2) { + return swigfaissJNI.ranklist_intersection_size(k1, SWIGTYPE_p_long_long.getCPtr(v1.data()), v1, k2, SWIGTYPE_p_long_long.getCPtr(v2.data()), v2); + } + + public static long merge_result_table_with(long n, long k, LongVector I0, SWIGTYPE_p_float D0, LongVector I1, SWIGTYPE_p_float D1, boolean keep_min, long translation) { + return swigfaissJNI.merge_result_table_with__SWIG_0(n, k, SWIGTYPE_p_long_long.getCPtr(I0.data()), I0, SWIGTYPE_p_float.getCPtr(D0), SWIGTYPE_p_long_long.getCPtr(I1.data()), I1, SWIGTYPE_p_float.getCPtr(D1), keep_min, translation); + } + + public static long merge_result_table_with(long n, long k, LongVector I0, SWIGTYPE_p_float D0, LongVector I1, SWIGTYPE_p_float D1, boolean keep_min) { + return swigfaissJNI.merge_result_table_with__SWIG_1(n, k, SWIGTYPE_p_long_long.getCPtr(I0.data()), I0, SWIGTYPE_p_float.getCPtr(D0), SWIGTYPE_p_long_long.getCPtr(I1.data()), I1, SWIGTYPE_p_float.getCPtr(D1), keep_min); + } + + public static long merge_result_table_with(long n, long k, LongVector I0, SWIGTYPE_p_float D0, LongVector I1, SWIGTYPE_p_float D1) { + return swigfaissJNI.merge_result_table_with__SWIG_2(n, k, SWIGTYPE_p_long_long.getCPtr(I0.data()), I0, SWIGTYPE_p_float.getCPtr(D0), SWIGTYPE_p_long_long.getCPtr(I1.data()), I1, SWIGTYPE_p_float.getCPtr(D1)); + } + + public static double imbalance_factor(int n, int k, LongVector assign) { + return swigfaissJNI.imbalance_factor__SWIG_0(n, k, SWIGTYPE_p_long_long.getCPtr(assign.data()), assign); + } + + public static double imbalance_factor(int k, SWIGTYPE_p_int hist) { + return swigfaissJNI.imbalance_factor__SWIG_1(k, SWIGTYPE_p_int.getCPtr(hist)); + } + + public static void fvec_argsort(long n, SWIGTYPE_p_float vals, SWIGTYPE_p_unsigned_long perm) { + swigfaissJNI.fvec_argsort(n, SWIGTYPE_p_float.getCPtr(vals), SWIGTYPE_p_unsigned_long.getCPtr(perm)); + } + + public static void fvec_argsort_parallel(long n, SWIGTYPE_p_float vals, SWIGTYPE_p_unsigned_long perm) { + swigfaissJNI.fvec_argsort_parallel(n, SWIGTYPE_p_float.getCPtr(vals), SWIGTYPE_p_unsigned_long.getCPtr(perm)); + } + + public static int ivec_hist(long n, SWIGTYPE_p_int v, int vmax, SWIGTYPE_p_int hist) { + return swigfaissJNI.ivec_hist(n, SWIGTYPE_p_int.getCPtr(v), vmax, SWIGTYPE_p_int.getCPtr(hist)); + } + + public static void bincode_hist(long n, long nbits, SWIGTYPE_p_unsigned_char codes, SWIGTYPE_p_int hist) { + swigfaissJNI.bincode_hist(n, nbits, SWIGTYPE_p_unsigned_char.getCPtr(codes), SWIGTYPE_p_int.getCPtr(hist)); + } + + public static long ivec_checksum(long n, SWIGTYPE_p_int a) { + return swigfaissJNI.ivec_checksum(n, SWIGTYPE_p_int.getCPtr(a)); + } + + public static SWIGTYPE_p_float fvecs_maybe_subsample(long d, SWIGTYPE_p_unsigned_long n, long nmax, SWIGTYPE_p_float x, boolean verbose, long seed) { + long cPtr = swigfaissJNI.fvecs_maybe_subsample__SWIG_0(d, SWIGTYPE_p_unsigned_long.getCPtr(n), nmax, SWIGTYPE_p_float.getCPtr(x), verbose, seed); + return (cPtr == 0) ? null : new SWIGTYPE_p_float(cPtr, false); + } + + public static SWIGTYPE_p_float fvecs_maybe_subsample(long d, SWIGTYPE_p_unsigned_long n, long nmax, SWIGTYPE_p_float x, boolean verbose) { + long cPtr = swigfaissJNI.fvecs_maybe_subsample__SWIG_1(d, SWIGTYPE_p_unsigned_long.getCPtr(n), nmax, SWIGTYPE_p_float.getCPtr(x), verbose); + return (cPtr == 0) ? null : new SWIGTYPE_p_float(cPtr, false); + } + + public static SWIGTYPE_p_float fvecs_maybe_subsample(long d, SWIGTYPE_p_unsigned_long n, long nmax, SWIGTYPE_p_float x) { + long cPtr = swigfaissJNI.fvecs_maybe_subsample__SWIG_2(d, SWIGTYPE_p_unsigned_long.getCPtr(n), nmax, SWIGTYPE_p_float.getCPtr(x)); + return (cPtr == 0) ? null : new SWIGTYPE_p_float(cPtr, false); + } + + public static void binary_to_real(long d, SWIGTYPE_p_unsigned_char x_in, SWIGTYPE_p_float x_out) { + swigfaissJNI.binary_to_real(d, SWIGTYPE_p_unsigned_char.getCPtr(x_in), SWIGTYPE_p_float.getCPtr(x_out)); + } + + public static void real_to_binary(long d, SWIGTYPE_p_float x_in, SWIGTYPE_p_unsigned_char x_out) { + swigfaissJNI.real_to_binary(d, SWIGTYPE_p_float.getCPtr(x_in), SWIGTYPE_p_unsigned_char.getCPtr(x_out)); + } + + public static long hash_bytes(SWIGTYPE_p_unsigned_char bytes, long n) { + return swigfaissJNI.hash_bytes(SWIGTYPE_p_unsigned_char.getCPtr(bytes), n); + } + + public static boolean check_openmp() { + return swigfaissJNI.check_openmp(); + } + + public static float kmeans_clustering(long d, long n, long k, SWIGTYPE_p_float x, SWIGTYPE_p_float centroids) { + return swigfaissJNI.kmeans_clustering(d, n, k, SWIGTYPE_p_float.getCPtr(x), SWIGTYPE_p_float.getCPtr(centroids)); + } + + public static void setIndexPQ_stats(IndexPQStats value) { + swigfaissJNI.indexPQ_stats_set(IndexPQStats.getCPtr(value), value); + } + + public static IndexPQStats getIndexPQ_stats() { + long cPtr = swigfaissJNI.indexPQ_stats_get(); + return (cPtr == 0) ? null : new IndexPQStats(cPtr, false); + } + + public static void setIndexIVF_stats(IndexIVFStats value) { + swigfaissJNI.indexIVF_stats_set(IndexIVFStats.getCPtr(value), value); + } + + public static IndexIVFStats getIndexIVF_stats() { + long cPtr = swigfaissJNI.indexIVF_stats_get(); + return (cPtr == 0) ? null : new IndexIVFStats(cPtr, false); + } + + public static short[] getHamdis_tab_ham_bytes() { + return swigfaissJNI.hamdis_tab_ham_bytes_get(); + } + + public static int generalized_hamming_64(long a) { + return swigfaissJNI.generalized_hamming_64(a); + } + + public static void generalized_hammings_knn_hc(SWIGTYPE_p_faiss__HeapArrayT_faiss__CMaxT_int_int64_t_t_t ha, SWIGTYPE_p_unsigned_char a, SWIGTYPE_p_unsigned_char b, long nb, long code_size, int ordered) { + swigfaissJNI.generalized_hammings_knn_hc__SWIG_0(SWIGTYPE_p_faiss__HeapArrayT_faiss__CMaxT_int_int64_t_t_t.getCPtr(ha), SWIGTYPE_p_unsigned_char.getCPtr(a), SWIGTYPE_p_unsigned_char.getCPtr(b), nb, code_size, ordered); + } + + public static void generalized_hammings_knn_hc(SWIGTYPE_p_faiss__HeapArrayT_faiss__CMaxT_int_int64_t_t_t ha, SWIGTYPE_p_unsigned_char a, SWIGTYPE_p_unsigned_char b, long nb, long code_size) { + swigfaissJNI.generalized_hammings_knn_hc__SWIG_1(SWIGTYPE_p_faiss__HeapArrayT_faiss__CMaxT_int_int64_t_t_t.getCPtr(ha), SWIGTYPE_p_unsigned_char.getCPtr(a), SWIGTYPE_p_unsigned_char.getCPtr(b), nb, code_size); + } + + public static void check_compatible_for_merge(Index index1, Index index2) { + swigfaissJNI.check_compatible_for_merge(Index.getCPtr(index1), index1, Index.getCPtr(index2), index2); + } + + public static IndexIVF extract_index_ivf(Index index) { + long cPtr = swigfaissJNI.extract_index_ivf__SWIG_0(Index.getCPtr(index), index); + return (cPtr == 0) ? null : new IndexIVF(cPtr, false); + } + + public static IndexIVF try_extract_index_ivf(Index index) { + long cPtr = swigfaissJNI.try_extract_index_ivf__SWIG_0(Index.getCPtr(index), index); + return (cPtr == 0) ? null : new IndexIVF(cPtr, false); + } + + public static void merge_into(Index index0, Index index1, boolean shift_ids) { + swigfaissJNI.merge_into(Index.getCPtr(index0), index0, Index.getCPtr(index1), index1, shift_ids); + } + + public static void search_centroid(Index index, SWIGTYPE_p_float x, int n, LongVector centroid_ids) { + swigfaissJNI.search_centroid(Index.getCPtr(index), index, SWIGTYPE_p_float.getCPtr(x), n, SWIGTYPE_p_long_long.getCPtr(centroid_ids.data()), centroid_ids); + } + + public static void search_and_return_centroids(Index index, long n, SWIGTYPE_p_float xin, int k, SWIGTYPE_p_float distances, LongVector labels, LongVector query_centroid_ids, LongVector result_centroid_ids) { + swigfaissJNI.search_and_return_centroids(Index.getCPtr(index), index, n, SWIGTYPE_p_float.getCPtr(xin), k, SWIGTYPE_p_float.getCPtr(distances), SWIGTYPE_p_long_long.getCPtr(labels.data()), labels, SWIGTYPE_p_long_long.getCPtr(query_centroid_ids.data()), query_centroid_ids, SWIGTYPE_p_long_long.getCPtr(result_centroid_ids.data()), result_centroid_ids); + } + + public static ArrayInvertedLists get_invlist_range(Index index, int i0, int i1) { + long cPtr = swigfaissJNI.get_invlist_range(Index.getCPtr(index), index, i0, i1); + return (cPtr == 0) ? null : new ArrayInvertedLists(cPtr, false); + } + + public static void set_invlist_range(Index index, int i0, int i1, ArrayInvertedLists src) { + swigfaissJNI.set_invlist_range(Index.getCPtr(index), index, i0, i1, ArrayInvertedLists.getCPtr(src), src); + } + + public static void search_with_parameters(Index index, long n, SWIGTYPE_p_float x, long k, SWIGTYPE_p_float distances, LongVector labels, IVFSearchParameters params, SWIGTYPE_p_unsigned_long nb_dis, SWIGTYPE_p_double ms_per_stage) { + swigfaissJNI.search_with_parameters__SWIG_0(Index.getCPtr(index), index, n, SWIGTYPE_p_float.getCPtr(x), k, SWIGTYPE_p_float.getCPtr(distances), SWIGTYPE_p_long_long.getCPtr(labels.data()), labels, IVFSearchParameters.getCPtr(params), params, SWIGTYPE_p_unsigned_long.getCPtr(nb_dis), SWIGTYPE_p_double.getCPtr(ms_per_stage)); + } + + public static void search_with_parameters(Index index, long n, SWIGTYPE_p_float x, long k, SWIGTYPE_p_float distances, LongVector labels, IVFSearchParameters params, SWIGTYPE_p_unsigned_long nb_dis) { + swigfaissJNI.search_with_parameters__SWIG_1(Index.getCPtr(index), index, n, SWIGTYPE_p_float.getCPtr(x), k, SWIGTYPE_p_float.getCPtr(distances), SWIGTYPE_p_long_long.getCPtr(labels.data()), labels, IVFSearchParameters.getCPtr(params), params, SWIGTYPE_p_unsigned_long.getCPtr(nb_dis)); + } + + public static void search_with_parameters(Index index, long n, SWIGTYPE_p_float x, long k, SWIGTYPE_p_float distances, LongVector labels, IVFSearchParameters params) { + swigfaissJNI.search_with_parameters__SWIG_2(Index.getCPtr(index), index, n, SWIGTYPE_p_float.getCPtr(x), k, SWIGTYPE_p_float.getCPtr(distances), SWIGTYPE_p_long_long.getCPtr(labels.data()), labels, IVFSearchParameters.getCPtr(params), params); + } + + public static void range_search_with_parameters(Index index, long n, SWIGTYPE_p_float x, float radius, RangeSearchResult result, IVFSearchParameters params, SWIGTYPE_p_unsigned_long nb_dis, SWIGTYPE_p_double ms_per_stage) { + swigfaissJNI.range_search_with_parameters__SWIG_0(Index.getCPtr(index), index, n, SWIGTYPE_p_float.getCPtr(x), radius, RangeSearchResult.getCPtr(result), result, IVFSearchParameters.getCPtr(params), params, SWIGTYPE_p_unsigned_long.getCPtr(nb_dis), SWIGTYPE_p_double.getCPtr(ms_per_stage)); + } + + public static void range_search_with_parameters(Index index, long n, SWIGTYPE_p_float x, float radius, RangeSearchResult result, IVFSearchParameters params, SWIGTYPE_p_unsigned_long nb_dis) { + swigfaissJNI.range_search_with_parameters__SWIG_1(Index.getCPtr(index), index, n, SWIGTYPE_p_float.getCPtr(x), radius, RangeSearchResult.getCPtr(result), result, IVFSearchParameters.getCPtr(params), params, SWIGTYPE_p_unsigned_long.getCPtr(nb_dis)); + } + + public static void range_search_with_parameters(Index index, long n, SWIGTYPE_p_float x, float radius, RangeSearchResult result, IVFSearchParameters params) { + swigfaissJNI.range_search_with_parameters__SWIG_2(Index.getCPtr(index), index, n, SWIGTYPE_p_float.getCPtr(x), radius, RangeSearchResult.getCPtr(result), result, IVFSearchParameters.getCPtr(params), params); + } + + public static void setHnsw_stats(HNSWStats value) { + swigfaissJNI.hnsw_stats_set(HNSWStats.getCPtr(value), value); + } + + public static HNSWStats getHnsw_stats() { + long cPtr = swigfaissJNI.hnsw_stats_get(); + return (cPtr == 0) ? null : new HNSWStats(cPtr, false); + } + + public static void setPrecomputed_table_max_bytes(long value) { + swigfaissJNI.precomputed_table_max_bytes_set(value); + } + + public static long getPrecomputed_table_max_bytes() { + return swigfaissJNI.precomputed_table_max_bytes_get(); + } + + public static void initialize_IVFPQ_precomputed_table(SWIGTYPE_p_int use_precomputed_table, Index quantizer, ProductQuantizer pq, SWIGTYPE_p_AlignedTableT_float_32_t precomputed_table, boolean verbose) { + swigfaissJNI.initialize_IVFPQ_precomputed_table(SWIGTYPE_p_int.getCPtr(use_precomputed_table), Index.getCPtr(quantizer), quantizer, ProductQuantizer.getCPtr(pq), pq, SWIGTYPE_p_AlignedTableT_float_32_t.getCPtr(precomputed_table), verbose); + } + + public static void setIndexIVFPQ_stats(IndexIVFPQStats value) { + swigfaissJNI.indexIVFPQ_stats_set(IndexIVFPQStats.getCPtr(value), value); + } + + public static IndexIVFPQStats getIndexIVFPQ_stats() { + long cPtr = swigfaissJNI.indexIVFPQ_stats_get(); + return (cPtr == 0) ? null : new IndexIVFPQStats(cPtr, false); + } + + public static Index downcast_index(Index index) { + long cPtr = swigfaissJNI.downcast_index(Index.getCPtr(index), index); + return (cPtr == 0) ? null : new Index(cPtr, false); + } + + public static VectorTransform downcast_VectorTransform(VectorTransform vt) { + long cPtr = swigfaissJNI.downcast_VectorTransform(VectorTransform.getCPtr(vt), vt); + return (cPtr == 0) ? null : new VectorTransform(cPtr, false); + } + + public static IndexBinary downcast_IndexBinary(IndexBinary index) { + long cPtr = swigfaissJNI.downcast_IndexBinary(IndexBinary.getCPtr(index), index); + return (cPtr == 0) ? null : new IndexBinary(cPtr, false); + } + + public static Index upcast_IndexShards(IndexShards index) { + long cPtr = swigfaissJNI.upcast_IndexShards(IndexShards.getCPtr(index), index); + return (cPtr == 0) ? null : new Index(cPtr, false); + } + + public static void write_index(Index idx, String fname) { + swigfaissJNI.write_index__SWIG_0(Index.getCPtr(idx), idx, fname); + } + + public static void write_index(Index idx, SWIGTYPE_p_FILE f) { + swigfaissJNI.write_index__SWIG_1(Index.getCPtr(idx), idx, SWIGTYPE_p_FILE.getCPtr(f)); + } + + public static void write_index(Index idx, SWIGTYPE_p_faiss__IOWriter writer) { + swigfaissJNI.write_index__SWIG_2(Index.getCPtr(idx), idx, SWIGTYPE_p_faiss__IOWriter.getCPtr(writer)); + } + + public static void write_index_binary(IndexBinary idx, String fname) { + swigfaissJNI.write_index_binary__SWIG_0(IndexBinary.getCPtr(idx), idx, fname); + } + + public static void write_index_binary(IndexBinary idx, SWIGTYPE_p_FILE f) { + swigfaissJNI.write_index_binary__SWIG_1(IndexBinary.getCPtr(idx), idx, SWIGTYPE_p_FILE.getCPtr(f)); + } + + public static void write_index_binary(IndexBinary idx, SWIGTYPE_p_faiss__IOWriter writer) { + swigfaissJNI.write_index_binary__SWIG_2(IndexBinary.getCPtr(idx), idx, SWIGTYPE_p_faiss__IOWriter.getCPtr(writer)); + } + + public static int getIO_FLAG_READ_ONLY() { + return swigfaissJNI.IO_FLAG_READ_ONLY_get(); + } + + public static int getIO_FLAG_ONDISK_SAME_DIR() { + return swigfaissJNI.IO_FLAG_ONDISK_SAME_DIR_get(); + } + + public static int getIO_FLAG_SKIP_IVF_DATA() { + return swigfaissJNI.IO_FLAG_SKIP_IVF_DATA_get(); + } + + public static int getIO_FLAG_MMAP() { + return swigfaissJNI.IO_FLAG_MMAP_get(); + } + + public static Index read_index(String fname, int io_flags) { + long cPtr = swigfaissJNI.read_index__SWIG_0(fname, io_flags); + return (cPtr == 0) ? null : new Index(cPtr, true); + } + + public static Index read_index(String fname) { + long cPtr = swigfaissJNI.read_index__SWIG_1(fname); + return (cPtr == 0) ? null : new Index(cPtr, true); + } + + public static Index read_index(SWIGTYPE_p_FILE f, int io_flags) { + long cPtr = swigfaissJNI.read_index__SWIG_2(SWIGTYPE_p_FILE.getCPtr(f), io_flags); + return (cPtr == 0) ? null : new Index(cPtr, true); + } + + public static Index read_index(SWIGTYPE_p_FILE f) { + long cPtr = swigfaissJNI.read_index__SWIG_3(SWIGTYPE_p_FILE.getCPtr(f)); + return (cPtr == 0) ? null : new Index(cPtr, true); + } + + public static Index read_index(SWIGTYPE_p_faiss__IOReader reader, int io_flags) { + long cPtr = swigfaissJNI.read_index__SWIG_4(SWIGTYPE_p_faiss__IOReader.getCPtr(reader), io_flags); + return (cPtr == 0) ? null : new Index(cPtr, true); + } + + public static Index read_index(SWIGTYPE_p_faiss__IOReader reader) { + long cPtr = swigfaissJNI.read_index__SWIG_5(SWIGTYPE_p_faiss__IOReader.getCPtr(reader)); + return (cPtr == 0) ? null : new Index(cPtr, true); + } + + public static IndexBinary read_index_binary(String fname, int io_flags) { + long cPtr = swigfaissJNI.read_index_binary__SWIG_0(fname, io_flags); + return (cPtr == 0) ? null : new IndexBinary(cPtr, true); + } + + public static IndexBinary read_index_binary(String fname) { + long cPtr = swigfaissJNI.read_index_binary__SWIG_1(fname); + return (cPtr == 0) ? null : new IndexBinary(cPtr, true); + } + + public static IndexBinary read_index_binary(SWIGTYPE_p_FILE f, int io_flags) { + long cPtr = swigfaissJNI.read_index_binary__SWIG_2(SWIGTYPE_p_FILE.getCPtr(f), io_flags); + return (cPtr == 0) ? null : new IndexBinary(cPtr, true); + } + + public static IndexBinary read_index_binary(SWIGTYPE_p_FILE f) { + long cPtr = swigfaissJNI.read_index_binary__SWIG_3(SWIGTYPE_p_FILE.getCPtr(f)); + return (cPtr == 0) ? null : new IndexBinary(cPtr, true); + } + + public static IndexBinary read_index_binary(SWIGTYPE_p_faiss__IOReader reader, int io_flags) { + long cPtr = swigfaissJNI.read_index_binary__SWIG_4(SWIGTYPE_p_faiss__IOReader.getCPtr(reader), io_flags); + return (cPtr == 0) ? null : new IndexBinary(cPtr, true); + } + + public static IndexBinary read_index_binary(SWIGTYPE_p_faiss__IOReader reader) { + long cPtr = swigfaissJNI.read_index_binary__SWIG_5(SWIGTYPE_p_faiss__IOReader.getCPtr(reader)); + return (cPtr == 0) ? null : new IndexBinary(cPtr, true); + } + + public static void write_VectorTransform(VectorTransform vt, String fname) { + swigfaissJNI.write_VectorTransform(VectorTransform.getCPtr(vt), vt, fname); + } + + public static VectorTransform read_VectorTransform(String fname) { + long cPtr = swigfaissJNI.read_VectorTransform(fname); + return (cPtr == 0) ? null : new VectorTransform(cPtr, true); + } + + public static ProductQuantizer read_ProductQuantizer(String fname) { + long cPtr = swigfaissJNI.read_ProductQuantizer__SWIG_0(fname); + return (cPtr == 0) ? null : new ProductQuantizer(cPtr, true); + } + + public static ProductQuantizer read_ProductQuantizer(SWIGTYPE_p_faiss__IOReader reader) { + long cPtr = swigfaissJNI.read_ProductQuantizer__SWIG_1(SWIGTYPE_p_faiss__IOReader.getCPtr(reader)); + return (cPtr == 0) ? null : new ProductQuantizer(cPtr, true); + } + + public static void write_ProductQuantizer(ProductQuantizer pq, String fname) { + swigfaissJNI.write_ProductQuantizer__SWIG_0(ProductQuantizer.getCPtr(pq), pq, fname); + } + + public static void write_ProductQuantizer(ProductQuantizer pq, SWIGTYPE_p_faiss__IOWriter f) { + swigfaissJNI.write_ProductQuantizer__SWIG_1(ProductQuantizer.getCPtr(pq), pq, SWIGTYPE_p_faiss__IOWriter.getCPtr(f)); + } + + public static void write_InvertedLists(InvertedLists ils, SWIGTYPE_p_faiss__IOWriter f) { + swigfaissJNI.write_InvertedLists(InvertedLists.getCPtr(ils), ils, SWIGTYPE_p_faiss__IOWriter.getCPtr(f)); + } + + public static InvertedLists read_InvertedLists(SWIGTYPE_p_faiss__IOReader reader, int io_flags) { + long cPtr = swigfaissJNI.read_InvertedLists__SWIG_0(SWIGTYPE_p_faiss__IOReader.getCPtr(reader), io_flags); + return (cPtr == 0) ? null : new InvertedLists(cPtr, false); + } + + public static InvertedLists read_InvertedLists(SWIGTYPE_p_faiss__IOReader reader) { + long cPtr = swigfaissJNI.read_InvertedLists__SWIG_1(SWIGTYPE_p_faiss__IOReader.getCPtr(reader)); + return (cPtr == 0) ? null : new InvertedLists(cPtr, false); + } + + public static Index index_factory(int d, String description, MetricType metric) { + long cPtr = swigfaissJNI.index_factory__SWIG_0(d, description, metric.swigValue()); + return (cPtr == 0) ? null : new Index(cPtr, true); + } + + public static Index index_factory(int d, String description) { + long cPtr = swigfaissJNI.index_factory__SWIG_1(d, description); + return (cPtr == 0) ? null : new Index(cPtr, true); + } + + public static void setIndex_factory_verbose(int value) { + swigfaissJNI.index_factory_verbose_set(value); + } + + public static int getIndex_factory_verbose() { + return swigfaissJNI.index_factory_verbose_get(); + } + + public static IndexBinary index_binary_factory(int d, String description) { + long cPtr = swigfaissJNI.index_binary_factory(d, description); + return (cPtr == 0) ? null : new IndexBinary(cPtr, true); + } + + public static void simd_histogram_8(SWIGTYPE_p_uint16_t data, int n, SWIGTYPE_p_uint16_t min, int shift, SWIGTYPE_p_int hist) { + swigfaissJNI.simd_histogram_8(SWIGTYPE_p_uint16_t.getCPtr(data), n, SWIGTYPE_p_uint16_t.getCPtr(min), shift, SWIGTYPE_p_int.getCPtr(hist)); + } + + public static void simd_histogram_16(SWIGTYPE_p_uint16_t data, int n, SWIGTYPE_p_uint16_t min, int shift, SWIGTYPE_p_int hist) { + swigfaissJNI.simd_histogram_16(SWIGTYPE_p_uint16_t.getCPtr(data), n, SWIGTYPE_p_uint16_t.getCPtr(min), shift, SWIGTYPE_p_int.getCPtr(hist)); + } + + public static void setPartition_stats(PartitionStats value) { + swigfaissJNI.partition_stats_set(PartitionStats.getCPtr(value), value); + } + + public static PartitionStats getPartition_stats() { + long cPtr = swigfaissJNI.partition_stats_get(); + return (cPtr == 0) ? null : new PartitionStats(cPtr, false); + } + + public static float CMin_float_partition_fuzzy(SWIGTYPE_p_float vals, LongVector ids, long n, long q_min, long q_max, SWIGTYPE_p_unsigned_long q_out) { + return swigfaissJNI.CMin_float_partition_fuzzy(SWIGTYPE_p_float.getCPtr(vals), SWIGTYPE_p_long_long.getCPtr(ids.data()), ids, n, q_min, q_max, SWIGTYPE_p_unsigned_long.getCPtr(q_out)); + } + + public static float CMax_float_partition_fuzzy(SWIGTYPE_p_float vals, LongVector ids, long n, long q_min, long q_max, SWIGTYPE_p_unsigned_long q_out) { + return swigfaissJNI.CMax_float_partition_fuzzy(SWIGTYPE_p_float.getCPtr(vals), SWIGTYPE_p_long_long.getCPtr(ids.data()), ids, n, q_min, q_max, SWIGTYPE_p_unsigned_long.getCPtr(q_out)); + } + + public static SWIGTYPE_p_uint16_t CMax_uint16_partition_fuzzy(SWIGTYPE_p_uint16_t vals, LongVector ids, long n, long q_min, long q_max, SWIGTYPE_p_unsigned_long q_out) { + return new SWIGTYPE_p_uint16_t(swigfaissJNI.CMax_uint16_partition_fuzzy__SWIG_0(SWIGTYPE_p_uint16_t.getCPtr(vals), SWIGTYPE_p_long_long.getCPtr(ids.data()), ids, n, q_min, q_max, SWIGTYPE_p_unsigned_long.getCPtr(q_out)), true); + } + + public static SWIGTYPE_p_uint16_t CMin_uint16_partition_fuzzy(SWIGTYPE_p_uint16_t vals, LongVector ids, long n, long q_min, long q_max, SWIGTYPE_p_unsigned_long q_out) { + return new SWIGTYPE_p_uint16_t(swigfaissJNI.CMin_uint16_partition_fuzzy__SWIG_0(SWIGTYPE_p_uint16_t.getCPtr(vals), SWIGTYPE_p_long_long.getCPtr(ids.data()), ids, n, q_min, q_max, SWIGTYPE_p_unsigned_long.getCPtr(q_out)), true); + } + + public static SWIGTYPE_p_uint16_t CMax_uint16_partition_fuzzy(SWIGTYPE_p_uint16_t vals, SWIGTYPE_p_int ids, long n, long q_min, long q_max, SWIGTYPE_p_unsigned_long q_out) { + return new SWIGTYPE_p_uint16_t(swigfaissJNI.CMax_uint16_partition_fuzzy__SWIG_1(SWIGTYPE_p_uint16_t.getCPtr(vals), SWIGTYPE_p_int.getCPtr(ids), n, q_min, q_max, SWIGTYPE_p_unsigned_long.getCPtr(q_out)), true); + } + + public static SWIGTYPE_p_uint16_t CMin_uint16_partition_fuzzy(SWIGTYPE_p_uint16_t vals, SWIGTYPE_p_int ids, long n, long q_min, long q_max, SWIGTYPE_p_unsigned_long q_out) { + return new SWIGTYPE_p_uint16_t(swigfaissJNI.CMin_uint16_partition_fuzzy__SWIG_1(SWIGTYPE_p_uint16_t.getCPtr(vals), SWIGTYPE_p_int.getCPtr(ids), n, q_min, q_max, SWIGTYPE_p_unsigned_long.getCPtr(q_out)), true); + } + + public static void omp_set_num_threads(int num_threads) { + swigfaissJNI.omp_set_num_threads(num_threads); + } + + public static int omp_get_max_threads() { + return swigfaissJNI.omp_get_max_threads(); + } + + public static SWIGTYPE_p_void memcpy(SWIGTYPE_p_void dest, SWIGTYPE_p_void src, long n) { + long cPtr = swigfaissJNI.memcpy(SWIGTYPE_p_void.getCPtr(dest), SWIGTYPE_p_void.getCPtr(src), n); + return (cPtr == 0) ? null : new SWIGTYPE_p_void(cPtr, false); + } + + public static SWIGTYPE_p_float cast_integer_to_float_ptr(int x) { + long cPtr = swigfaissJNI.cast_integer_to_float_ptr(x); + return (cPtr == 0) ? null : new SWIGTYPE_p_float(cPtr, false); + } + + public static SWIGTYPE_p_long cast_integer_to_long_ptr(int x) { + long cPtr = swigfaissJNI.cast_integer_to_long_ptr(x); + return (cPtr == 0) ? null : new SWIGTYPE_p_long(cPtr, false); + } + + public static SWIGTYPE_p_int cast_integer_to_int_ptr(int x) { + long cPtr = swigfaissJNI.cast_integer_to_int_ptr(x); + return (cPtr == 0) ? null : new SWIGTYPE_p_int(cPtr, false); + } + + public static void ignore_SIGTTIN() { + swigfaissJNI.ignore_SIGTTIN(); + } + +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/swigfaissConstants.java b/ann/src/main/java/com/twitter/ann/faiss/swig/swigfaissConstants.java new file mode 100644 index 0000000000..30ba6abc31 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/swigfaissConstants.java @@ -0,0 +1,15 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; + +public interface swigfaissConstants { + public final static int FAISS_VERSION_MAJOR = swigfaissJNI.FAISS_VERSION_MAJOR_get(); + public final static int FAISS_VERSION_MINOR = swigfaissJNI.FAISS_VERSION_MINOR_get(); + public final static int FAISS_VERSION_PATCH = swigfaissJNI.FAISS_VERSION_PATCH_get(); +} diff --git a/ann/src/main/java/com/twitter/ann/faiss/swig/swigfaissJNI.java b/ann/src/main/java/com/twitter/ann/faiss/swig/swigfaissJNI.java new file mode 100644 index 0000000000..9a5c19beae --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/faiss/swig/swigfaissJNI.java @@ -0,0 +1,2147 @@ +/* ---------------------------------------------------------------------------- + * This file was automatically generated by SWIG (http://www.swig.org). + * Version 4.0.2 + * + * Do not make changes to this file unless you know what you are doing--modify + * the SWIG interface file instead. + * ----------------------------------------------------------------------------- */ + +package com.twitter.ann.faiss; +import com.twitter.ann.faiss.NativeUtils; +public class swigfaissJNI { + + static { + try { + if (NativeUtils.getOperatingSystemType() == NativeUtils.OSType.MacOS) { + NativeUtils.loadLibraryFromJar("/com/twitter/ann/faiss/swig/resources/swigfaiss.dylib"); + } else { + NativeUtils.loadLibraryFromJar("/com/twitter/ann/faiss/swig/resources/libstdc++.so.6"); + NativeUtils.loadLibraryFromJar("/com/twitter/ann/faiss/swig/resources/libgcc_s.so.1"); + NativeUtils.loadLibraryFromJar("/com/twitter/ann/faiss/swig/resources/libgomp.so.1"); + NativeUtils.loadLibraryFromJar("/com/twitter/ann/faiss/swig/resources/libquadmath.so.0"); + NativeUtils.loadLibraryFromJar("/com/twitter/ann/faiss/swig/resources/libgfortran.so.5"); + NativeUtils.loadLibraryFromJar("/com/twitter/ann/faiss/swig/resources/swigfaiss.so"); + } + } catch (Exception e) { + System.err.println("Native code library failed to load. \n" + e); + System.exit(1); + } + } + + public final static native long new_intArray(int jarg1); + public final static native void delete_intArray(long jarg1); + public final static native int intArray_getitem(long jarg1, intArray jarg1_, int jarg2); + public final static native void intArray_setitem(long jarg1, intArray jarg1_, int jarg2, int jarg3); + public final static native long intArray_cast(long jarg1, intArray jarg1_); + public final static native long intArray_frompointer(long jarg1); + public final static native long new_floatArray(int jarg1); + public final static native void delete_floatArray(long jarg1); + public final static native float floatArray_getitem(long jarg1, floatArray jarg1_, int jarg2); + public final static native void floatArray_setitem(long jarg1, floatArray jarg1_, int jarg2, float jarg3); + public final static native long floatArray_cast(long jarg1, floatArray jarg1_); + public final static native long floatArray_frompointer(long jarg1); + public final static native long new_longArray(int jarg1); + public final static native void delete_longArray(long jarg1); + public final static native long longArray_getitem(long jarg1, longArray jarg1_, int jarg2); + public final static native void longArray_setitem(long jarg1, longArray jarg1_, int jarg2, long jarg3); + public final static native long longArray_cast(long jarg1, longArray jarg1_); + public final static native long longArray_frompointer(long jarg1); + public final static native long new_doubleArray(int jarg1); + public final static native void delete_doubleArray(long jarg1); + public final static native double doubleArray_getitem(long jarg1, doubleArray jarg1_, int jarg2); + public final static native void doubleArray_setitem(long jarg1, doubleArray jarg1_, int jarg2, double jarg3); + public final static native long doubleArray_cast(long jarg1, doubleArray jarg1_); + public final static native long doubleArray_frompointer(long jarg1); + public final static native long new_FloatVector(); + public final static native void FloatVector_push_back(long jarg1, FloatVector jarg1_, float jarg2); + public final static native void FloatVector_clear(long jarg1, FloatVector jarg1_); + public final static native long FloatVector_data(long jarg1, FloatVector jarg1_); + public final static native long FloatVector_size(long jarg1, FloatVector jarg1_); + public final static native float FloatVector_at(long jarg1, FloatVector jarg1_, long jarg2); + public final static native void FloatVector_resize(long jarg1, FloatVector jarg1_, long jarg2); + public final static native void FloatVector_reserve(long jarg1, FloatVector jarg1_, long jarg2); + public final static native void FloatVector_swap(long jarg1, FloatVector jarg1_, long jarg2, FloatVector jarg2_); + public final static native void delete_FloatVector(long jarg1); + public final static native long new_DoubleVector(); + public final static native void DoubleVector_push_back(long jarg1, DoubleVector jarg1_, double jarg2); + public final static native void DoubleVector_clear(long jarg1, DoubleVector jarg1_); + public final static native long DoubleVector_data(long jarg1, DoubleVector jarg1_); + public final static native long DoubleVector_size(long jarg1, DoubleVector jarg1_); + public final static native double DoubleVector_at(long jarg1, DoubleVector jarg1_, long jarg2); + public final static native void DoubleVector_resize(long jarg1, DoubleVector jarg1_, long jarg2); + public final static native void DoubleVector_reserve(long jarg1, DoubleVector jarg1_, long jarg2); + public final static native void DoubleVector_swap(long jarg1, DoubleVector jarg1_, long jarg2, DoubleVector jarg2_); + public final static native void delete_DoubleVector(long jarg1); + public final static native long new_ByteVector(); + public final static native void ByteVector_push_back(long jarg1, ByteVector jarg1_, short jarg2); + public final static native void ByteVector_clear(long jarg1, ByteVector jarg1_); + public final static native long ByteVector_data(long jarg1, ByteVector jarg1_); + public final static native long ByteVector_size(long jarg1, ByteVector jarg1_); + public final static native short ByteVector_at(long jarg1, ByteVector jarg1_, long jarg2); + public final static native void ByteVector_resize(long jarg1, ByteVector jarg1_, long jarg2); + public final static native void ByteVector_reserve(long jarg1, ByteVector jarg1_, long jarg2); + public final static native void ByteVector_swap(long jarg1, ByteVector jarg1_, long jarg2, ByteVector jarg2_); + public final static native void delete_ByteVector(long jarg1); + public final static native long new_CharVector(); + public final static native void CharVector_push_back(long jarg1, CharVector jarg1_, char jarg2); + public final static native void CharVector_clear(long jarg1, CharVector jarg1_); + public final static native String CharVector_data(long jarg1, CharVector jarg1_); + public final static native long CharVector_size(long jarg1, CharVector jarg1_); + public final static native char CharVector_at(long jarg1, CharVector jarg1_, long jarg2); + public final static native void CharVector_resize(long jarg1, CharVector jarg1_, long jarg2); + public final static native void CharVector_reserve(long jarg1, CharVector jarg1_, long jarg2); + public final static native void CharVector_swap(long jarg1, CharVector jarg1_, long jarg2, CharVector jarg2_); + public final static native void delete_CharVector(long jarg1); + public final static native long new_Uint64Vector(); + public final static native void Uint64Vector_push_back(long jarg1, Uint64Vector jarg1_, long jarg2); + public final static native void Uint64Vector_clear(long jarg1, Uint64Vector jarg1_); + public final static native long Uint64Vector_data(long jarg1, Uint64Vector jarg1_); + public final static native long Uint64Vector_size(long jarg1, Uint64Vector jarg1_); + public final static native long Uint64Vector_at(long jarg1, Uint64Vector jarg1_, long jarg2); + public final static native void Uint64Vector_resize(long jarg1, Uint64Vector jarg1_, long jarg2); + public final static native void Uint64Vector_reserve(long jarg1, Uint64Vector jarg1_, long jarg2); + public final static native void Uint64Vector_swap(long jarg1, Uint64Vector jarg1_, long jarg2, Uint64Vector jarg2_); + public final static native void delete_Uint64Vector(long jarg1); + public final static native long new_LongVector(); + public final static native void LongVector_push_back(long jarg1, LongVector jarg1_, long jarg2); + public final static native void LongVector_clear(long jarg1, LongVector jarg1_); + public final static native long LongVector_data(long jarg1, LongVector jarg1_); + public final static native long LongVector_size(long jarg1, LongVector jarg1_); + public final static native long LongVector_at(long jarg1, LongVector jarg1_, long jarg2); + public final static native void LongVector_resize(long jarg1, LongVector jarg1_, long jarg2); + public final static native void LongVector_reserve(long jarg1, LongVector jarg1_, long jarg2); + public final static native void LongVector_swap(long jarg1, LongVector jarg1_, long jarg2, LongVector jarg2_); + public final static native void delete_LongVector(long jarg1); + public final static native long new_IntVector(); + public final static native void IntVector_push_back(long jarg1, IntVector jarg1_, int jarg2); + public final static native void IntVector_clear(long jarg1, IntVector jarg1_); + public final static native long IntVector_data(long jarg1, IntVector jarg1_); + public final static native long IntVector_size(long jarg1, IntVector jarg1_); + public final static native int IntVector_at(long jarg1, IntVector jarg1_, long jarg2); + public final static native void IntVector_resize(long jarg1, IntVector jarg1_, long jarg2); + public final static native void IntVector_reserve(long jarg1, IntVector jarg1_, long jarg2); + public final static native void IntVector_swap(long jarg1, IntVector jarg1_, long jarg2, IntVector jarg2_); + public final static native void delete_IntVector(long jarg1); + public final static native long new_VectorTransformVector(); + public final static native void VectorTransformVector_push_back(long jarg1, VectorTransformVector jarg1_, long jarg2, VectorTransform jarg2_); + public final static native void VectorTransformVector_clear(long jarg1, VectorTransformVector jarg1_); + public final static native long VectorTransformVector_data(long jarg1, VectorTransformVector jarg1_); + public final static native long VectorTransformVector_size(long jarg1, VectorTransformVector jarg1_); + public final static native long VectorTransformVector_at(long jarg1, VectorTransformVector jarg1_, long jarg2); + public final static native void VectorTransformVector_resize(long jarg1, VectorTransformVector jarg1_, long jarg2); + public final static native void VectorTransformVector_reserve(long jarg1, VectorTransformVector jarg1_, long jarg2); + public final static native void VectorTransformVector_swap(long jarg1, VectorTransformVector jarg1_, long jarg2, VectorTransformVector jarg2_); + public final static native void delete_VectorTransformVector(long jarg1); + public final static native long new_OperatingPointVector(); + public final static native void OperatingPointVector_push_back(long jarg1, OperatingPointVector jarg1_, long jarg2, OperatingPoint jarg2_); + public final static native void OperatingPointVector_clear(long jarg1, OperatingPointVector jarg1_); + public final static native long OperatingPointVector_data(long jarg1, OperatingPointVector jarg1_); + public final static native long OperatingPointVector_size(long jarg1, OperatingPointVector jarg1_); + public final static native long OperatingPointVector_at(long jarg1, OperatingPointVector jarg1_, long jarg2); + public final static native void OperatingPointVector_resize(long jarg1, OperatingPointVector jarg1_, long jarg2); + public final static native void OperatingPointVector_reserve(long jarg1, OperatingPointVector jarg1_, long jarg2); + public final static native void OperatingPointVector_swap(long jarg1, OperatingPointVector jarg1_, long jarg2, OperatingPointVector jarg2_); + public final static native void delete_OperatingPointVector(long jarg1); + public final static native long new_InvertedListsPtrVector(); + public final static native void InvertedListsPtrVector_push_back(long jarg1, InvertedListsPtrVector jarg1_, long jarg2, InvertedLists jarg2_); + public final static native void InvertedListsPtrVector_clear(long jarg1, InvertedListsPtrVector jarg1_); + public final static native long InvertedListsPtrVector_data(long jarg1, InvertedListsPtrVector jarg1_); + public final static native long InvertedListsPtrVector_size(long jarg1, InvertedListsPtrVector jarg1_); + public final static native long InvertedListsPtrVector_at(long jarg1, InvertedListsPtrVector jarg1_, long jarg2); + public final static native void InvertedListsPtrVector_resize(long jarg1, InvertedListsPtrVector jarg1_, long jarg2); + public final static native void InvertedListsPtrVector_reserve(long jarg1, InvertedListsPtrVector jarg1_, long jarg2); + public final static native void InvertedListsPtrVector_swap(long jarg1, InvertedListsPtrVector jarg1_, long jarg2, InvertedListsPtrVector jarg2_); + public final static native void delete_InvertedListsPtrVector(long jarg1); + public final static native long new_FloatVectorVector(); + public final static native void FloatVectorVector_push_back(long jarg1, FloatVectorVector jarg1_, long jarg2, FloatVector jarg2_); + public final static native void FloatVectorVector_clear(long jarg1, FloatVectorVector jarg1_); + public final static native long FloatVectorVector_data(long jarg1, FloatVectorVector jarg1_); + public final static native long FloatVectorVector_size(long jarg1, FloatVectorVector jarg1_); + public final static native long FloatVectorVector_at(long jarg1, FloatVectorVector jarg1_, long jarg2); + public final static native void FloatVectorVector_resize(long jarg1, FloatVectorVector jarg1_, long jarg2); + public final static native void FloatVectorVector_reserve(long jarg1, FloatVectorVector jarg1_, long jarg2); + public final static native void FloatVectorVector_swap(long jarg1, FloatVectorVector jarg1_, long jarg2, FloatVectorVector jarg2_); + public final static native void delete_FloatVectorVector(long jarg1); + public final static native long new_ByteVectorVector(); + public final static native void ByteVectorVector_push_back(long jarg1, ByteVectorVector jarg1_, long jarg2, ByteVector jarg2_); + public final static native void ByteVectorVector_clear(long jarg1, ByteVectorVector jarg1_); + public final static native long ByteVectorVector_data(long jarg1, ByteVectorVector jarg1_); + public final static native long ByteVectorVector_size(long jarg1, ByteVectorVector jarg1_); + public final static native long ByteVectorVector_at(long jarg1, ByteVectorVector jarg1_, long jarg2); + public final static native void ByteVectorVector_resize(long jarg1, ByteVectorVector jarg1_, long jarg2); + public final static native void ByteVectorVector_reserve(long jarg1, ByteVectorVector jarg1_, long jarg2); + public final static native void ByteVectorVector_swap(long jarg1, ByteVectorVector jarg1_, long jarg2, ByteVectorVector jarg2_); + public final static native void delete_ByteVectorVector(long jarg1); + public final static native long new_LongVectorVector(); + public final static native void LongVectorVector_push_back(long jarg1, LongVectorVector jarg1_, long jarg2); + public final static native void LongVectorVector_clear(long jarg1, LongVectorVector jarg1_); + public final static native long LongVectorVector_data(long jarg1, LongVectorVector jarg1_); + public final static native long LongVectorVector_size(long jarg1, LongVectorVector jarg1_); + public final static native long LongVectorVector_at(long jarg1, LongVectorVector jarg1_, long jarg2); + public final static native void LongVectorVector_resize(long jarg1, LongVectorVector jarg1_, long jarg2); + public final static native void LongVectorVector_reserve(long jarg1, LongVectorVector jarg1_, long jarg2); + public final static native void LongVectorVector_swap(long jarg1, LongVectorVector jarg1_, long jarg2, LongVectorVector jarg2_); + public final static native void delete_LongVectorVector(long jarg1); + public final static native void bitvec_print(long jarg1, long jarg2); + public final static native void fvecs2bitvecs(long jarg1, long jarg2, long jarg3, long jarg4); + public final static native void bitvecs2fvecs(long jarg1, long jarg2, long jarg3, long jarg4); + public final static native void fvec2bitvec(long jarg1, long jarg2, long jarg3); + public final static native void bitvec_shuffle(long jarg1, long jarg2, long jarg3, long jarg4, long jarg5, long jarg6); + public final static native void BitstringWriter_code_set(long jarg1, BitstringWriter jarg1_, long jarg2); + public final static native long BitstringWriter_code_get(long jarg1, BitstringWriter jarg1_); + public final static native void BitstringWriter_code_size_set(long jarg1, BitstringWriter jarg1_, long jarg2); + public final static native long BitstringWriter_code_size_get(long jarg1, BitstringWriter jarg1_); + public final static native void BitstringWriter_i_set(long jarg1, BitstringWriter jarg1_, long jarg2); + public final static native long BitstringWriter_i_get(long jarg1, BitstringWriter jarg1_); + public final static native long new_BitstringWriter(long jarg1, long jarg2); + public final static native void BitstringWriter_write(long jarg1, BitstringWriter jarg1_, long jarg2, int jarg3); + public final static native void delete_BitstringWriter(long jarg1); + public final static native void BitstringReader_code_set(long jarg1, BitstringReader jarg1_, long jarg2); + public final static native long BitstringReader_code_get(long jarg1, BitstringReader jarg1_); + public final static native void BitstringReader_code_size_set(long jarg1, BitstringReader jarg1_, long jarg2); + public final static native long BitstringReader_code_size_get(long jarg1, BitstringReader jarg1_); + public final static native void BitstringReader_i_set(long jarg1, BitstringReader jarg1_, long jarg2); + public final static native long BitstringReader_i_get(long jarg1, BitstringReader jarg1_); + public final static native long new_BitstringReader(long jarg1, long jarg2); + public final static native long BitstringReader_read(long jarg1, BitstringReader jarg1_, int jarg2); + public final static native void delete_BitstringReader(long jarg1); + public final static native void hamming_batch_size_set(long jarg1); + public final static native long hamming_batch_size_get(); + public final static native int popcount64(long jarg1); + public final static native void hammings(long jarg1, long jarg2, long jarg3, long jarg4, long jarg5, long jarg6); + public final static native void hammings_knn_hc(long jarg1, long jarg2, long jarg3, long jarg4, long jarg5, int jarg6); + public final static native void hammings_knn(long jarg1, long jarg2, long jarg3, long jarg4, long jarg5, int jarg6); + public final static native void hammings_knn_mc(long jarg1, long jarg2, long jarg3, long jarg4, long jarg5, long jarg6, long jarg7, long jarg8, LongVector jarg8_); + public final static native void hamming_range_search(long jarg1, long jarg2, long jarg3, long jarg4, int jarg5, long jarg6, long jarg7, RangeSearchResult jarg7_); + public final static native void hamming_count_thres(long jarg1, long jarg2, long jarg3, long jarg4, int jarg5, long jarg6, long jarg7); + public final static native long match_hamming_thres(long jarg1, long jarg2, long jarg3, long jarg4, int jarg5, long jarg6, long jarg7, LongVector jarg7_, long jarg8); + public final static native void crosshamming_count_thres(long jarg1, long jarg2, int jarg3, long jarg4, long jarg5); + public final static native int get_num_gpus(); + public final static native int METRIC_INNER_PRODUCT_get(); + public final static native int METRIC_L2_get(); + public final static native int METRIC_Canberra_get(); + public final static native String get_compile_options(); + public final static native double getmillisecs(); + public final static native long get_mem_usage_kb(); + public final static native long get_cycles(); + public final static native void fvec_madd(long jarg1, long jarg2, float jarg3, long jarg4, long jarg5); + public final static native int fvec_madd_and_argmin(long jarg1, long jarg2, float jarg3, long jarg4, long jarg5); + public final static native void reflection(long jarg1, long jarg2, long jarg3, long jarg4, long jarg5); + public final static native void matrix_qr(int jarg1, int jarg2, long jarg3); + public final static native void ranklist_handle_ties(int jarg1, long jarg2, LongVector jarg2_, long jarg3); + public final static native long ranklist_intersection_size(long jarg1, long jarg2, LongVector jarg2_, long jarg3, long jarg4, LongVector jarg4_); + public final static native long merge_result_table_with__SWIG_0(long jarg1, long jarg2, long jarg3, LongVector jarg3_, long jarg4, long jarg5, LongVector jarg5_, long jarg6, boolean jarg7, long jarg8); + public final static native long merge_result_table_with__SWIG_1(long jarg1, long jarg2, long jarg3, LongVector jarg3_, long jarg4, long jarg5, LongVector jarg5_, long jarg6, boolean jarg7); + public final static native long merge_result_table_with__SWIG_2(long jarg1, long jarg2, long jarg3, LongVector jarg3_, long jarg4, long jarg5, LongVector jarg5_, long jarg6); + public final static native double imbalance_factor__SWIG_0(int jarg1, int jarg2, long jarg3, LongVector jarg3_); + public final static native double imbalance_factor__SWIG_1(int jarg1, long jarg2); + public final static native void fvec_argsort(long jarg1, long jarg2, long jarg3); + public final static native void fvec_argsort_parallel(long jarg1, long jarg2, long jarg3); + public final static native int ivec_hist(long jarg1, long jarg2, int jarg3, long jarg4); + public final static native void bincode_hist(long jarg1, long jarg2, long jarg3, long jarg4); + public final static native long ivec_checksum(long jarg1, long jarg2); + public final static native long fvecs_maybe_subsample__SWIG_0(long jarg1, long jarg2, long jarg3, long jarg4, boolean jarg5, long jarg6); + public final static native long fvecs_maybe_subsample__SWIG_1(long jarg1, long jarg2, long jarg3, long jarg4, boolean jarg5); + public final static native long fvecs_maybe_subsample__SWIG_2(long jarg1, long jarg2, long jarg3, long jarg4); + public final static native void binary_to_real(long jarg1, long jarg2, long jarg3); + public final static native void real_to_binary(long jarg1, long jarg2, long jarg3); + public final static native long hash_bytes(long jarg1, long jarg2); + public final static native boolean check_openmp(); + public final static native int FAISS_VERSION_MAJOR_get(); + public final static native int FAISS_VERSION_MINOR_get(); + public final static native int FAISS_VERSION_PATCH_get(); + public final static native void Index_d_set(long jarg1, Index jarg1_, int jarg2); + public final static native int Index_d_get(long jarg1, Index jarg1_); + public final static native void Index_ntotal_set(long jarg1, Index jarg1_, long jarg2); + public final static native long Index_ntotal_get(long jarg1, Index jarg1_); + public final static native void Index_verbose_set(long jarg1, Index jarg1_, boolean jarg2); + public final static native boolean Index_verbose_get(long jarg1, Index jarg1_); + public final static native void Index_is_trained_set(long jarg1, Index jarg1_, boolean jarg2); + public final static native boolean Index_is_trained_get(long jarg1, Index jarg1_); + public final static native void Index_metric_type_set(long jarg1, Index jarg1_, int jarg2); + public final static native int Index_metric_type_get(long jarg1, Index jarg1_); + public final static native void Index_metric_arg_set(long jarg1, Index jarg1_, float jarg2); + public final static native float Index_metric_arg_get(long jarg1, Index jarg1_); + public final static native void delete_Index(long jarg1); + public final static native void Index_train(long jarg1, Index jarg1_, long jarg2, long jarg3); + public final static native void Index_add(long jarg1, Index jarg1_, long jarg2, long jarg3); + public final static native void Index_add_with_ids(long jarg1, Index jarg1_, long jarg2, long jarg3, long jarg4, LongVector jarg4_); + public final static native void Index_search(long jarg1, Index jarg1_, long jarg2, long jarg3, long jarg4, long jarg5, long jarg6, LongVector jarg6_); + public final static native void Index_range_search(long jarg1, Index jarg1_, long jarg2, long jarg3, float jarg4, long jarg5, RangeSearchResult jarg5_); + public final static native void Index_assign__SWIG_0(long jarg1, Index jarg1_, long jarg2, long jarg3, long jarg4, LongVector jarg4_, long jarg5); + public final static native void Index_assign__SWIG_1(long jarg1, Index jarg1_, long jarg2, long jarg3, long jarg4, LongVector jarg4_); + public final static native void Index_reset(long jarg1, Index jarg1_); + public final static native long Index_remove_ids(long jarg1, Index jarg1_, long jarg2, IDSelector jarg2_); + public final static native void Index_reconstruct(long jarg1, Index jarg1_, long jarg2, long jarg3); + public final static native void Index_reconstruct_n(long jarg1, Index jarg1_, long jarg2, long jarg3, long jarg4); + public final static native void Index_search_and_reconstruct(long jarg1, Index jarg1_, long jarg2, long jarg3, long jarg4, long jarg5, long jarg6, LongVector jarg6_, long jarg7); + public final static native void Index_compute_residual(long jarg1, Index jarg1_, long jarg2, long jarg3, long jarg4); + public final static native void Index_compute_residual_n(long jarg1, Index jarg1_, long jarg2, long jarg3, long jarg4, long jarg5, LongVector jarg5_); + public final static native long Index_get_distance_computer(long jarg1, Index jarg1_); + public final static native long Index_sa_code_size(long jarg1, Index jarg1_); + public final static native void Index_sa_encode(long jarg1, Index jarg1_, long jarg2, long jarg3, long jarg4); + public final static native void Index_sa_decode(long jarg1, Index jarg1_, long jarg2, long jarg3, long jarg4); + public final static native long Index_toIVF(long jarg1, Index jarg1_); + public final static native void ClusteringParameters_niter_set(long jarg1, ClusteringParameters jarg1_, int jarg2); + public final static native int ClusteringParameters_niter_get(long jarg1, ClusteringParameters jarg1_); + public final static native void ClusteringParameters_nredo_set(long jarg1, ClusteringParameters jarg1_, int jarg2); + public final static native int ClusteringParameters_nredo_get(long jarg1, ClusteringParameters jarg1_); + public final static native void ClusteringParameters_verbose_set(long jarg1, ClusteringParameters jarg1_, boolean jarg2); + public final static native boolean ClusteringParameters_verbose_get(long jarg1, ClusteringParameters jarg1_); + public final static native void ClusteringParameters_spherical_set(long jarg1, ClusteringParameters jarg1_, boolean jarg2); + public final static native boolean ClusteringParameters_spherical_get(long jarg1, ClusteringParameters jarg1_); + public final static native void ClusteringParameters_int_centroids_set(long jarg1, ClusteringParameters jarg1_, boolean jarg2); + public final static native boolean ClusteringParameters_int_centroids_get(long jarg1, ClusteringParameters jarg1_); + public final static native void ClusteringParameters_update_index_set(long jarg1, ClusteringParameters jarg1_, boolean jarg2); + public final static native boolean ClusteringParameters_update_index_get(long jarg1, ClusteringParameters jarg1_); + public final static native void ClusteringParameters_frozen_centroids_set(long jarg1, ClusteringParameters jarg1_, boolean jarg2); + public final static native boolean ClusteringParameters_frozen_centroids_get(long jarg1, ClusteringParameters jarg1_); + public final static native void ClusteringParameters_min_points_per_centroid_set(long jarg1, ClusteringParameters jarg1_, int jarg2); + public final static native int ClusteringParameters_min_points_per_centroid_get(long jarg1, ClusteringParameters jarg1_); + public final static native void ClusteringParameters_max_points_per_centroid_set(long jarg1, ClusteringParameters jarg1_, int jarg2); + public final static native int ClusteringParameters_max_points_per_centroid_get(long jarg1, ClusteringParameters jarg1_); + public final static native void ClusteringParameters_seed_set(long jarg1, ClusteringParameters jarg1_, int jarg2); + public final static native int ClusteringParameters_seed_get(long jarg1, ClusteringParameters jarg1_); + public final static native void ClusteringParameters_decode_block_size_set(long jarg1, ClusteringParameters jarg1_, long jarg2); + public final static native long ClusteringParameters_decode_block_size_get(long jarg1, ClusteringParameters jarg1_); + public final static native long new_ClusteringParameters(); + public final static native void delete_ClusteringParameters(long jarg1); + public final static native void ClusteringIterationStats_obj_set(long jarg1, ClusteringIterationStats jarg1_, float jarg2); + public final static native float ClusteringIterationStats_obj_get(long jarg1, ClusteringIterationStats jarg1_); + public final static native void ClusteringIterationStats_time_set(long jarg1, ClusteringIterationStats jarg1_, double jarg2); + public final static native double ClusteringIterationStats_time_get(long jarg1, ClusteringIterationStats jarg1_); + public final static native void ClusteringIterationStats_time_search_set(long jarg1, ClusteringIterationStats jarg1_, double jarg2); + public final static native double ClusteringIterationStats_time_search_get(long jarg1, ClusteringIterationStats jarg1_); + public final static native void ClusteringIterationStats_imbalance_factor_set(long jarg1, ClusteringIterationStats jarg1_, double jarg2); + public final static native double ClusteringIterationStats_imbalance_factor_get(long jarg1, ClusteringIterationStats jarg1_); + public final static native void ClusteringIterationStats_nsplit_set(long jarg1, ClusteringIterationStats jarg1_, int jarg2); + public final static native int ClusteringIterationStats_nsplit_get(long jarg1, ClusteringIterationStats jarg1_); + public final static native long new_ClusteringIterationStats(); + public final static native void delete_ClusteringIterationStats(long jarg1); + public final static native void Clustering_d_set(long jarg1, Clustering jarg1_, long jarg2); + public final static native long Clustering_d_get(long jarg1, Clustering jarg1_); + public final static native void Clustering_k_set(long jarg1, Clustering jarg1_, long jarg2); + public final static native long Clustering_k_get(long jarg1, Clustering jarg1_); + public final static native void Clustering_centroids_set(long jarg1, Clustering jarg1_, long jarg2, FloatVector jarg2_); + public final static native long Clustering_centroids_get(long jarg1, Clustering jarg1_); + public final static native void Clustering_iteration_stats_set(long jarg1, Clustering jarg1_, long jarg2); + public final static native long Clustering_iteration_stats_get(long jarg1, Clustering jarg1_); + public final static native long new_Clustering__SWIG_0(int jarg1, int jarg2); + public final static native long new_Clustering__SWIG_1(int jarg1, int jarg2, long jarg3, ClusteringParameters jarg3_); + public final static native void Clustering_train__SWIG_0(long jarg1, Clustering jarg1_, long jarg2, long jarg3, long jarg4, Index jarg4_, long jarg5); + public final static native void Clustering_train__SWIG_1(long jarg1, Clustering jarg1_, long jarg2, long jarg3, long jarg4, Index jarg4_); + public final static native void Clustering_train_encoded__SWIG_0(long jarg1, Clustering jarg1_, long jarg2, long jarg3, long jarg4, Index jarg4_, long jarg5, Index jarg5_, long jarg6); + public final static native void Clustering_train_encoded__SWIG_1(long jarg1, Clustering jarg1_, long jarg2, long jarg3, long jarg4, Index jarg4_, long jarg5, Index jarg5_); + public final static native void Clustering_post_process_centroids(long jarg1, Clustering jarg1_); + public final static native void delete_Clustering(long jarg1); + public final static native long new_Clustering1D__SWIG_0(int jarg1); + public final static native long new_Clustering1D__SWIG_1(int jarg1, long jarg2, ClusteringParameters jarg2_); + public final static native void Clustering1D_train_exact(long jarg1, Clustering1D jarg1_, long jarg2, long jarg3); + public final static native void delete_Clustering1D(long jarg1); + public final static native void ProgressiveDimClusteringParameters_progressive_dim_steps_set(long jarg1, ProgressiveDimClusteringParameters jarg1_, int jarg2); + public final static native int ProgressiveDimClusteringParameters_progressive_dim_steps_get(long jarg1, ProgressiveDimClusteringParameters jarg1_); + public final static native void ProgressiveDimClusteringParameters_apply_pca_set(long jarg1, ProgressiveDimClusteringParameters jarg1_, boolean jarg2); + public final static native boolean ProgressiveDimClusteringParameters_apply_pca_get(long jarg1, ProgressiveDimClusteringParameters jarg1_); + public final static native long new_ProgressiveDimClusteringParameters(); + public final static native void delete_ProgressiveDimClusteringParameters(long jarg1); + public final static native void delete_ProgressiveDimIndexFactory(long jarg1); + public final static native long new_ProgressiveDimIndexFactory(); + public final static native void ProgressiveDimClustering_d_set(long jarg1, ProgressiveDimClustering jarg1_, long jarg2); + public final static native long ProgressiveDimClustering_d_get(long jarg1, ProgressiveDimClustering jarg1_); + public final static native void ProgressiveDimClustering_k_set(long jarg1, ProgressiveDimClustering jarg1_, long jarg2); + public final static native long ProgressiveDimClustering_k_get(long jarg1, ProgressiveDimClustering jarg1_); + public final static native void ProgressiveDimClustering_centroids_set(long jarg1, ProgressiveDimClustering jarg1_, long jarg2, FloatVector jarg2_); + public final static native long ProgressiveDimClustering_centroids_get(long jarg1, ProgressiveDimClustering jarg1_); + public final static native void ProgressiveDimClustering_iteration_stats_set(long jarg1, ProgressiveDimClustering jarg1_, long jarg2); + public final static native long ProgressiveDimClustering_iteration_stats_get(long jarg1, ProgressiveDimClustering jarg1_); + public final static native long new_ProgressiveDimClustering__SWIG_0(int jarg1, int jarg2); + public final static native long new_ProgressiveDimClustering__SWIG_1(int jarg1, int jarg2, long jarg3, ProgressiveDimClusteringParameters jarg3_); + public final static native void ProgressiveDimClustering_train(long jarg1, ProgressiveDimClustering jarg1_, long jarg2, long jarg3, long jarg4, ProgressiveDimIndexFactory jarg4_); + public final static native void delete_ProgressiveDimClustering(long jarg1); + public final static native float kmeans_clustering(long jarg1, long jarg2, long jarg3, long jarg4, long jarg5); + public final static native void ProductQuantizer_d_set(long jarg1, ProductQuantizer jarg1_, long jarg2); + public final static native long ProductQuantizer_d_get(long jarg1, ProductQuantizer jarg1_); + public final static native void ProductQuantizer_M_set(long jarg1, ProductQuantizer jarg1_, long jarg2); + public final static native long ProductQuantizer_M_get(long jarg1, ProductQuantizer jarg1_); + public final static native void ProductQuantizer_nbits_set(long jarg1, ProductQuantizer jarg1_, long jarg2); + public final static native long ProductQuantizer_nbits_get(long jarg1, ProductQuantizer jarg1_); + public final static native void ProductQuantizer_dsub_set(long jarg1, ProductQuantizer jarg1_, long jarg2); + public final static native long ProductQuantizer_dsub_get(long jarg1, ProductQuantizer jarg1_); + public final static native void ProductQuantizer_code_size_set(long jarg1, ProductQuantizer jarg1_, long jarg2); + public final static native long ProductQuantizer_code_size_get(long jarg1, ProductQuantizer jarg1_); + public final static native void ProductQuantizer_ksub_set(long jarg1, ProductQuantizer jarg1_, long jarg2); + public final static native long ProductQuantizer_ksub_get(long jarg1, ProductQuantizer jarg1_); + public final static native void ProductQuantizer_verbose_set(long jarg1, ProductQuantizer jarg1_, boolean jarg2); + public final static native boolean ProductQuantizer_verbose_get(long jarg1, ProductQuantizer jarg1_); + public final static native void ProductQuantizer_train_type_set(long jarg1, ProductQuantizer jarg1_, int jarg2); + public final static native int ProductQuantizer_train_type_get(long jarg1, ProductQuantizer jarg1_); + public final static native void ProductQuantizer_cp_set(long jarg1, ProductQuantizer jarg1_, long jarg2, ClusteringParameters jarg2_); + public final static native long ProductQuantizer_cp_get(long jarg1, ProductQuantizer jarg1_); + public final static native void ProductQuantizer_assign_index_set(long jarg1, ProductQuantizer jarg1_, long jarg2, Index jarg2_); + public final static native long ProductQuantizer_assign_index_get(long jarg1, ProductQuantizer jarg1_); + public final static native void ProductQuantizer_centroids_set(long jarg1, ProductQuantizer jarg1_, long jarg2, FloatVector jarg2_); + public final static native long ProductQuantizer_centroids_get(long jarg1, ProductQuantizer jarg1_); + public final static native long ProductQuantizer_get_centroids(long jarg1, ProductQuantizer jarg1_, long jarg2, long jarg3); + public final static native void ProductQuantizer_train(long jarg1, ProductQuantizer jarg1_, int jarg2, long jarg3); + public final static native long new_ProductQuantizer__SWIG_0(long jarg1, long jarg2, long jarg3); + public final static native long new_ProductQuantizer__SWIG_1(); + public final static native void ProductQuantizer_set_derived_values(long jarg1, ProductQuantizer jarg1_); + public final static native void ProductQuantizer_set_params(long jarg1, ProductQuantizer jarg1_, long jarg2, int jarg3); + public final static native void ProductQuantizer_compute_code(long jarg1, ProductQuantizer jarg1_, long jarg2, long jarg3); + public final static native void ProductQuantizer_compute_codes(long jarg1, ProductQuantizer jarg1_, long jarg2, long jarg3, long jarg4); + public final static native void ProductQuantizer_compute_codes_with_assign_index(long jarg1, ProductQuantizer jarg1_, long jarg2, long jarg3, long jarg4); + public final static native void ProductQuantizer_decode__SWIG_0(long jarg1, ProductQuantizer jarg1_, long jarg2, long jarg3); + public final static native void ProductQuantizer_decode__SWIG_1(long jarg1, ProductQuantizer jarg1_, long jarg2, long jarg3, long jarg4); + public final static native void ProductQuantizer_compute_code_from_distance_table(long jarg1, ProductQuantizer jarg1_, long jarg2, long jarg3); + public final static native void ProductQuantizer_compute_distance_table(long jarg1, ProductQuantizer jarg1_, long jarg2, long jarg3); + public final static native void ProductQuantizer_compute_inner_prod_table(long jarg1, ProductQuantizer jarg1_, long jarg2, long jarg3); + public final static native void ProductQuantizer_compute_distance_tables(long jarg1, ProductQuantizer jarg1_, long jarg2, long jarg3, long jarg4); + public final static native void ProductQuantizer_compute_inner_prod_tables(long jarg1, ProductQuantizer jarg1_, long jarg2, long jarg3, long jarg4); + public final static native void ProductQuantizer_search__SWIG_0(long jarg1, ProductQuantizer jarg1_, long jarg2, long jarg3, long jarg4, long jarg5, long jarg6, boolean jarg7); + public final static native void ProductQuantizer_search__SWIG_1(long jarg1, ProductQuantizer jarg1_, long jarg2, long jarg3, long jarg4, long jarg5, long jarg6); + public final static native void ProductQuantizer_search_ip__SWIG_0(long jarg1, ProductQuantizer jarg1_, long jarg2, long jarg3, long jarg4, long jarg5, long jarg6, boolean jarg7); + public final static native void ProductQuantizer_search_ip__SWIG_1(long jarg1, ProductQuantizer jarg1_, long jarg2, long jarg3, long jarg4, long jarg5, long jarg6); + public final static native void ProductQuantizer_sdc_table_set(long jarg1, ProductQuantizer jarg1_, long jarg2, FloatVector jarg2_); + public final static native long ProductQuantizer_sdc_table_get(long jarg1, ProductQuantizer jarg1_); + public final static native void ProductQuantizer_compute_sdc_table(long jarg1, ProductQuantizer jarg1_); + public final static native void ProductQuantizer_search_sdc__SWIG_0(long jarg1, ProductQuantizer jarg1_, long jarg2, long jarg3, long jarg4, long jarg5, long jarg6, boolean jarg7); + public final static native void ProductQuantizer_search_sdc__SWIG_1(long jarg1, ProductQuantizer jarg1_, long jarg2, long jarg3, long jarg4, long jarg5, long jarg6); + public final static native void delete_ProductQuantizer(long jarg1); + public final static native void PQEncoderGeneric_code_set(long jarg1, PQEncoderGeneric jarg1_, long jarg2); + public final static native long PQEncoderGeneric_code_get(long jarg1, PQEncoderGeneric jarg1_); + public final static native void PQEncoderGeneric_offset_set(long jarg1, PQEncoderGeneric jarg1_, short jarg2); + public final static native short PQEncoderGeneric_offset_get(long jarg1, PQEncoderGeneric jarg1_); + public final static native int PQEncoderGeneric_nbits_get(long jarg1, PQEncoderGeneric jarg1_); + public final static native void PQEncoderGeneric_reg_set(long jarg1, PQEncoderGeneric jarg1_, short jarg2); + public final static native short PQEncoderGeneric_reg_get(long jarg1, PQEncoderGeneric jarg1_); + public final static native long new_PQEncoderGeneric__SWIG_0(long jarg1, int jarg2, short jarg3); + public final static native long new_PQEncoderGeneric__SWIG_1(long jarg1, int jarg2); + public final static native void PQEncoderGeneric_encode(long jarg1, PQEncoderGeneric jarg1_, long jarg2); + public final static native void delete_PQEncoderGeneric(long jarg1); + public final static native void PQEncoder8_code_set(long jarg1, PQEncoder8 jarg1_, long jarg2); + public final static native long PQEncoder8_code_get(long jarg1, PQEncoder8 jarg1_); + public final static native long new_PQEncoder8(long jarg1, int jarg2); + public final static native void PQEncoder8_encode(long jarg1, PQEncoder8 jarg1_, long jarg2); + public final static native void delete_PQEncoder8(long jarg1); + public final static native void PQEncoder16_code_set(long jarg1, PQEncoder16 jarg1_, long jarg2); + public final static native long PQEncoder16_code_get(long jarg1, PQEncoder16 jarg1_); + public final static native long new_PQEncoder16(long jarg1, int jarg2); + public final static native void PQEncoder16_encode(long jarg1, PQEncoder16 jarg1_, long jarg2); + public final static native void delete_PQEncoder16(long jarg1); + public final static native void PQDecoderGeneric_code_set(long jarg1, PQDecoderGeneric jarg1_, long jarg2); + public final static native long PQDecoderGeneric_code_get(long jarg1, PQDecoderGeneric jarg1_); + public final static native void PQDecoderGeneric_offset_set(long jarg1, PQDecoderGeneric jarg1_, short jarg2); + public final static native short PQDecoderGeneric_offset_get(long jarg1, PQDecoderGeneric jarg1_); + public final static native int PQDecoderGeneric_nbits_get(long jarg1, PQDecoderGeneric jarg1_); + public final static native long PQDecoderGeneric_mask_get(long jarg1, PQDecoderGeneric jarg1_); + public final static native void PQDecoderGeneric_reg_set(long jarg1, PQDecoderGeneric jarg1_, short jarg2); + public final static native short PQDecoderGeneric_reg_get(long jarg1, PQDecoderGeneric jarg1_); + public final static native long new_PQDecoderGeneric(long jarg1, int jarg2); + public final static native long PQDecoderGeneric_decode(long jarg1, PQDecoderGeneric jarg1_); + public final static native void delete_PQDecoderGeneric(long jarg1); + public final static native int PQDecoder8_nbits_get(); + public final static native void PQDecoder8_code_set(long jarg1, PQDecoder8 jarg1_, long jarg2); + public final static native long PQDecoder8_code_get(long jarg1, PQDecoder8 jarg1_); + public final static native long new_PQDecoder8(long jarg1, int jarg2); + public final static native long PQDecoder8_decode(long jarg1, PQDecoder8 jarg1_); + public final static native void delete_PQDecoder8(long jarg1); + public final static native int PQDecoder16_nbits_get(); + public final static native void PQDecoder16_code_set(long jarg1, PQDecoder16 jarg1_, long jarg2); + public final static native long PQDecoder16_code_get(long jarg1, PQDecoder16 jarg1_); + public final static native long new_PQDecoder16(long jarg1, int jarg2); + public final static native long PQDecoder16_decode(long jarg1, PQDecoder16 jarg1_); + public final static native void delete_PQDecoder16(long jarg1); + public final static native void VectorTransform_d_in_set(long jarg1, VectorTransform jarg1_, int jarg2); + public final static native int VectorTransform_d_in_get(long jarg1, VectorTransform jarg1_); + public final static native void VectorTransform_d_out_set(long jarg1, VectorTransform jarg1_, int jarg2); + public final static native int VectorTransform_d_out_get(long jarg1, VectorTransform jarg1_); + public final static native void VectorTransform_is_trained_set(long jarg1, VectorTransform jarg1_, boolean jarg2); + public final static native boolean VectorTransform_is_trained_get(long jarg1, VectorTransform jarg1_); + public final static native void VectorTransform_train(long jarg1, VectorTransform jarg1_, long jarg2, long jarg3); + public final static native long VectorTransform_apply(long jarg1, VectorTransform jarg1_, long jarg2, long jarg3); + public final static native void VectorTransform_apply_noalloc(long jarg1, VectorTransform jarg1_, long jarg2, long jarg3, long jarg4); + public final static native void VectorTransform_reverse_transform(long jarg1, VectorTransform jarg1_, long jarg2, long jarg3, long jarg4); + public final static native void delete_VectorTransform(long jarg1); + public final static native void LinearTransform_have_bias_set(long jarg1, LinearTransform jarg1_, boolean jarg2); + public final static native boolean LinearTransform_have_bias_get(long jarg1, LinearTransform jarg1_); + public final static native void LinearTransform_is_orthonormal_set(long jarg1, LinearTransform jarg1_, boolean jarg2); + public final static native boolean LinearTransform_is_orthonormal_get(long jarg1, LinearTransform jarg1_); + public final static native void LinearTransform_A_set(long jarg1, LinearTransform jarg1_, long jarg2, FloatVector jarg2_); + public final static native long LinearTransform_A_get(long jarg1, LinearTransform jarg1_); + public final static native void LinearTransform_b_set(long jarg1, LinearTransform jarg1_, long jarg2, FloatVector jarg2_); + public final static native long LinearTransform_b_get(long jarg1, LinearTransform jarg1_); + public final static native long new_LinearTransform__SWIG_0(int jarg1, int jarg2, boolean jarg3); + public final static native long new_LinearTransform__SWIG_1(int jarg1, int jarg2); + public final static native long new_LinearTransform__SWIG_2(int jarg1); + public final static native long new_LinearTransform__SWIG_3(); + public final static native void LinearTransform_apply_noalloc(long jarg1, LinearTransform jarg1_, long jarg2, long jarg3, long jarg4); + public final static native void LinearTransform_transform_transpose(long jarg1, LinearTransform jarg1_, long jarg2, long jarg3, long jarg4); + public final static native void LinearTransform_reverse_transform(long jarg1, LinearTransform jarg1_, long jarg2, long jarg3, long jarg4); + public final static native void LinearTransform_set_is_orthonormal(long jarg1, LinearTransform jarg1_); + public final static native void LinearTransform_verbose_set(long jarg1, LinearTransform jarg1_, boolean jarg2); + public final static native boolean LinearTransform_verbose_get(long jarg1, LinearTransform jarg1_); + public final static native void LinearTransform_print_if_verbose(long jarg1, LinearTransform jarg1_, String jarg2, long jarg3, DoubleVector jarg3_, int jarg4, int jarg5); + public final static native void delete_LinearTransform(long jarg1); + public final static native long new_RandomRotationMatrix__SWIG_0(int jarg1, int jarg2); + public final static native void RandomRotationMatrix_init(long jarg1, RandomRotationMatrix jarg1_, int jarg2); + public final static native void RandomRotationMatrix_train(long jarg1, RandomRotationMatrix jarg1_, long jarg2, long jarg3); + public final static native long new_RandomRotationMatrix__SWIG_1(); + public final static native void delete_RandomRotationMatrix(long jarg1); + public final static native void PCAMatrix_eigen_power_set(long jarg1, PCAMatrix jarg1_, float jarg2); + public final static native float PCAMatrix_eigen_power_get(long jarg1, PCAMatrix jarg1_); + public final static native void PCAMatrix_epsilon_set(long jarg1, PCAMatrix jarg1_, float jarg2); + public final static native float PCAMatrix_epsilon_get(long jarg1, PCAMatrix jarg1_); + public final static native void PCAMatrix_random_rotation_set(long jarg1, PCAMatrix jarg1_, boolean jarg2); + public final static native boolean PCAMatrix_random_rotation_get(long jarg1, PCAMatrix jarg1_); + public final static native void PCAMatrix_max_points_per_d_set(long jarg1, PCAMatrix jarg1_, long jarg2); + public final static native long PCAMatrix_max_points_per_d_get(long jarg1, PCAMatrix jarg1_); + public final static native void PCAMatrix_balanced_bins_set(long jarg1, PCAMatrix jarg1_, int jarg2); + public final static native int PCAMatrix_balanced_bins_get(long jarg1, PCAMatrix jarg1_); + public final static native void PCAMatrix_mean_set(long jarg1, PCAMatrix jarg1_, long jarg2, FloatVector jarg2_); + public final static native long PCAMatrix_mean_get(long jarg1, PCAMatrix jarg1_); + public final static native void PCAMatrix_eigenvalues_set(long jarg1, PCAMatrix jarg1_, long jarg2, FloatVector jarg2_); + public final static native long PCAMatrix_eigenvalues_get(long jarg1, PCAMatrix jarg1_); + public final static native void PCAMatrix_PCAMat_set(long jarg1, PCAMatrix jarg1_, long jarg2, FloatVector jarg2_); + public final static native long PCAMatrix_PCAMat_get(long jarg1, PCAMatrix jarg1_); + public final static native long new_PCAMatrix__SWIG_0(int jarg1, int jarg2, float jarg3, boolean jarg4); + public final static native long new_PCAMatrix__SWIG_1(int jarg1, int jarg2, float jarg3); + public final static native long new_PCAMatrix__SWIG_2(int jarg1, int jarg2); + public final static native long new_PCAMatrix__SWIG_3(int jarg1); + public final static native long new_PCAMatrix__SWIG_4(); + public final static native void PCAMatrix_train(long jarg1, PCAMatrix jarg1_, long jarg2, long jarg3); + public final static native void PCAMatrix_copy_from(long jarg1, PCAMatrix jarg1_, long jarg2, PCAMatrix jarg2_); + public final static native void PCAMatrix_prepare_Ab(long jarg1, PCAMatrix jarg1_); + public final static native void delete_PCAMatrix(long jarg1); + public final static native void ITQMatrix_max_iter_set(long jarg1, ITQMatrix jarg1_, int jarg2); + public final static native int ITQMatrix_max_iter_get(long jarg1, ITQMatrix jarg1_); + public final static native void ITQMatrix_seed_set(long jarg1, ITQMatrix jarg1_, int jarg2); + public final static native int ITQMatrix_seed_get(long jarg1, ITQMatrix jarg1_); + public final static native void ITQMatrix_init_rotation_set(long jarg1, ITQMatrix jarg1_, long jarg2, DoubleVector jarg2_); + public final static native long ITQMatrix_init_rotation_get(long jarg1, ITQMatrix jarg1_); + public final static native long new_ITQMatrix__SWIG_0(int jarg1); + public final static native long new_ITQMatrix__SWIG_1(); + public final static native void ITQMatrix_train(long jarg1, ITQMatrix jarg1_, long jarg2, long jarg3); + public final static native void delete_ITQMatrix(long jarg1); + public final static native void ITQTransform_mean_set(long jarg1, ITQTransform jarg1_, long jarg2, FloatVector jarg2_); + public final static native long ITQTransform_mean_get(long jarg1, ITQTransform jarg1_); + public final static native void ITQTransform_do_pca_set(long jarg1, ITQTransform jarg1_, boolean jarg2); + public final static native boolean ITQTransform_do_pca_get(long jarg1, ITQTransform jarg1_); + public final static native void ITQTransform_itq_set(long jarg1, ITQTransform jarg1_, long jarg2, ITQMatrix jarg2_); + public final static native long ITQTransform_itq_get(long jarg1, ITQTransform jarg1_); + public final static native void ITQTransform_max_train_per_dim_set(long jarg1, ITQTransform jarg1_, int jarg2); + public final static native int ITQTransform_max_train_per_dim_get(long jarg1, ITQTransform jarg1_); + public final static native void ITQTransform_pca_then_itq_set(long jarg1, ITQTransform jarg1_, long jarg2, LinearTransform jarg2_); + public final static native long ITQTransform_pca_then_itq_get(long jarg1, ITQTransform jarg1_); + public final static native long new_ITQTransform__SWIG_0(int jarg1, int jarg2, boolean jarg3); + public final static native long new_ITQTransform__SWIG_1(int jarg1, int jarg2); + public final static native long new_ITQTransform__SWIG_2(int jarg1); + public final static native long new_ITQTransform__SWIG_3(); + public final static native void ITQTransform_train(long jarg1, ITQTransform jarg1_, long jarg2, long jarg3); + public final static native void ITQTransform_apply_noalloc(long jarg1, ITQTransform jarg1_, long jarg2, long jarg3, long jarg4); + public final static native void delete_ITQTransform(long jarg1); + public final static native void OPQMatrix_M_set(long jarg1, OPQMatrix jarg1_, int jarg2); + public final static native int OPQMatrix_M_get(long jarg1, OPQMatrix jarg1_); + public final static native void OPQMatrix_niter_set(long jarg1, OPQMatrix jarg1_, int jarg2); + public final static native int OPQMatrix_niter_get(long jarg1, OPQMatrix jarg1_); + public final static native void OPQMatrix_niter_pq_set(long jarg1, OPQMatrix jarg1_, int jarg2); + public final static native int OPQMatrix_niter_pq_get(long jarg1, OPQMatrix jarg1_); + public final static native void OPQMatrix_niter_pq_0_set(long jarg1, OPQMatrix jarg1_, int jarg2); + public final static native int OPQMatrix_niter_pq_0_get(long jarg1, OPQMatrix jarg1_); + public final static native void OPQMatrix_max_train_points_set(long jarg1, OPQMatrix jarg1_, long jarg2); + public final static native long OPQMatrix_max_train_points_get(long jarg1, OPQMatrix jarg1_); + public final static native void OPQMatrix_verbose_set(long jarg1, OPQMatrix jarg1_, boolean jarg2); + public final static native boolean OPQMatrix_verbose_get(long jarg1, OPQMatrix jarg1_); + public final static native void OPQMatrix_pq_set(long jarg1, OPQMatrix jarg1_, long jarg2, ProductQuantizer jarg2_); + public final static native long OPQMatrix_pq_get(long jarg1, OPQMatrix jarg1_); + public final static native long new_OPQMatrix__SWIG_0(int jarg1, int jarg2, int jarg3); + public final static native long new_OPQMatrix__SWIG_1(int jarg1, int jarg2); + public final static native long new_OPQMatrix__SWIG_2(int jarg1); + public final static native long new_OPQMatrix__SWIG_3(); + public final static native void OPQMatrix_train(long jarg1, OPQMatrix jarg1_, long jarg2, long jarg3); + public final static native void delete_OPQMatrix(long jarg1); + public final static native void RemapDimensionsTransform_map_set(long jarg1, RemapDimensionsTransform jarg1_, long jarg2, IntVector jarg2_); + public final static native long RemapDimensionsTransform_map_get(long jarg1, RemapDimensionsTransform jarg1_); + public final static native long new_RemapDimensionsTransform__SWIG_0(int jarg1, int jarg2, long jarg3); + public final static native long new_RemapDimensionsTransform__SWIG_1(int jarg1, int jarg2, boolean jarg3); + public final static native long new_RemapDimensionsTransform__SWIG_2(int jarg1, int jarg2); + public final static native void RemapDimensionsTransform_apply_noalloc(long jarg1, RemapDimensionsTransform jarg1_, long jarg2, long jarg3, long jarg4); + public final static native void RemapDimensionsTransform_reverse_transform(long jarg1, RemapDimensionsTransform jarg1_, long jarg2, long jarg3, long jarg4); + public final static native long new_RemapDimensionsTransform__SWIG_3(); + public final static native void delete_RemapDimensionsTransform(long jarg1); + public final static native void NormalizationTransform_norm_set(long jarg1, NormalizationTransform jarg1_, float jarg2); + public final static native float NormalizationTransform_norm_get(long jarg1, NormalizationTransform jarg1_); + public final static native long new_NormalizationTransform__SWIG_0(int jarg1, float jarg2); + public final static native long new_NormalizationTransform__SWIG_1(int jarg1); + public final static native long new_NormalizationTransform__SWIG_2(); + public final static native void NormalizationTransform_apply_noalloc(long jarg1, NormalizationTransform jarg1_, long jarg2, long jarg3, long jarg4); + public final static native void NormalizationTransform_reverse_transform(long jarg1, NormalizationTransform jarg1_, long jarg2, long jarg3, long jarg4); + public final static native void delete_NormalizationTransform(long jarg1); + public final static native void CenteringTransform_mean_set(long jarg1, CenteringTransform jarg1_, long jarg2, FloatVector jarg2_); + public final static native long CenteringTransform_mean_get(long jarg1, CenteringTransform jarg1_); + public final static native long new_CenteringTransform__SWIG_0(int jarg1); + public final static native long new_CenteringTransform__SWIG_1(); + public final static native void CenteringTransform_train(long jarg1, CenteringTransform jarg1_, long jarg2, long jarg3); + public final static native void CenteringTransform_apply_noalloc(long jarg1, CenteringTransform jarg1_, long jarg2, long jarg3, long jarg4); + public final static native void CenteringTransform_reverse_transform(long jarg1, CenteringTransform jarg1_, long jarg2, long jarg3, long jarg4); + public final static native void delete_CenteringTransform(long jarg1); + public final static native void IndexFlatCodes_code_size_set(long jarg1, IndexFlatCodes jarg1_, long jarg2); + public final static native long IndexFlatCodes_code_size_get(long jarg1, IndexFlatCodes jarg1_); + public final static native void IndexFlatCodes_codes_set(long jarg1, IndexFlatCodes jarg1_, long jarg2, ByteVector jarg2_); + public final static native long IndexFlatCodes_codes_get(long jarg1, IndexFlatCodes jarg1_); + public final static native void IndexFlatCodes_add(long jarg1, IndexFlatCodes jarg1_, long jarg2, long jarg3); + public final static native void IndexFlatCodes_reset(long jarg1, IndexFlatCodes jarg1_); + public final static native void IndexFlatCodes_reconstruct_n(long jarg1, IndexFlatCodes jarg1_, long jarg2, long jarg3, long jarg4); + public final static native void IndexFlatCodes_reconstruct(long jarg1, IndexFlatCodes jarg1_, long jarg2, long jarg3); + public final static native long IndexFlatCodes_sa_code_size(long jarg1, IndexFlatCodes jarg1_); + public final static native long IndexFlatCodes_remove_ids(long jarg1, IndexFlatCodes jarg1_, long jarg2, IDSelector jarg2_); + public final static native void delete_IndexFlatCodes(long jarg1); + public final static native long new_IndexFlat__SWIG_0(long jarg1, int jarg2); + public final static native long new_IndexFlat__SWIG_1(long jarg1); + public final static native void IndexFlat_search(long jarg1, IndexFlat jarg1_, long jarg2, long jarg3, long jarg4, long jarg5, long jarg6, LongVector jarg6_); + public final static native void IndexFlat_range_search(long jarg1, IndexFlat jarg1_, long jarg2, long jarg3, float jarg4, long jarg5, RangeSearchResult jarg5_); + public final static native void IndexFlat_reconstruct(long jarg1, IndexFlat jarg1_, long jarg2, long jarg3); + public final static native void IndexFlat_compute_distance_subset(long jarg1, IndexFlat jarg1_, long jarg2, long jarg3, long jarg4, long jarg5, long jarg6, LongVector jarg6_); + public final static native long IndexFlat_get_xb__SWIG_0(long jarg1, IndexFlat jarg1_); + public final static native long new_IndexFlat__SWIG_2(); + public final static native long IndexFlat_get_distance_computer(long jarg1, IndexFlat jarg1_); + public final static native void IndexFlat_sa_encode(long jarg1, IndexFlat jarg1_, long jarg2, long jarg3, long jarg4); + public final static native void IndexFlat_sa_decode(long jarg1, IndexFlat jarg1_, long jarg2, long jarg3, long jarg4); + public final static native void delete_IndexFlat(long jarg1); + public final static native long new_IndexFlatIP__SWIG_0(long jarg1); + public final static native long new_IndexFlatIP__SWIG_1(); + public final static native void delete_IndexFlatIP(long jarg1); + public final static native long new_IndexFlatL2__SWIG_0(long jarg1); + public final static native long new_IndexFlatL2__SWIG_1(); + public final static native void delete_IndexFlatL2(long jarg1); + public final static native void IndexFlat1D_continuous_update_set(long jarg1, IndexFlat1D jarg1_, boolean jarg2); + public final static native boolean IndexFlat1D_continuous_update_get(long jarg1, IndexFlat1D jarg1_); + public final static native void IndexFlat1D_perm_set(long jarg1, IndexFlat1D jarg1_, long jarg2); + public final static native long IndexFlat1D_perm_get(long jarg1, IndexFlat1D jarg1_); + public final static native long new_IndexFlat1D__SWIG_0(boolean jarg1); + public final static native long new_IndexFlat1D__SWIG_1(); + public final static native void IndexFlat1D_update_permutation(long jarg1, IndexFlat1D jarg1_); + public final static native void IndexFlat1D_add(long jarg1, IndexFlat1D jarg1_, long jarg2, long jarg3); + public final static native void IndexFlat1D_reset(long jarg1, IndexFlat1D jarg1_); + public final static native void IndexFlat1D_search(long jarg1, IndexFlat1D jarg1_, long jarg2, long jarg3, long jarg4, long jarg5, long jarg6, LongVector jarg6_); + public final static native void delete_IndexFlat1D(long jarg1); + public final static native void IndexLSH_nbits_set(long jarg1, IndexLSH jarg1_, int jarg2); + public final static native int IndexLSH_nbits_get(long jarg1, IndexLSH jarg1_); + public final static native void IndexLSH_rotate_data_set(long jarg1, IndexLSH jarg1_, boolean jarg2); + public final static native boolean IndexLSH_rotate_data_get(long jarg1, IndexLSH jarg1_); + public final static native void IndexLSH_train_thresholds_set(long jarg1, IndexLSH jarg1_, boolean jarg2); + public final static native boolean IndexLSH_train_thresholds_get(long jarg1, IndexLSH jarg1_); + public final static native void IndexLSH_rrot_set(long jarg1, IndexLSH jarg1_, long jarg2, RandomRotationMatrix jarg2_); + public final static native long IndexLSH_rrot_get(long jarg1, IndexLSH jarg1_); + public final static native void IndexLSH_thresholds_set(long jarg1, IndexLSH jarg1_, long jarg2, FloatVector jarg2_); + public final static native long IndexLSH_thresholds_get(long jarg1, IndexLSH jarg1_); + public final static native long new_IndexLSH__SWIG_0(long jarg1, int jarg2, boolean jarg3, boolean jarg4); + public final static native long new_IndexLSH__SWIG_1(long jarg1, int jarg2, boolean jarg3); + public final static native long new_IndexLSH__SWIG_2(long jarg1, int jarg2); + public final static native long IndexLSH_apply_preprocess(long jarg1, IndexLSH jarg1_, long jarg2, long jarg3); + public final static native void IndexLSH_train(long jarg1, IndexLSH jarg1_, long jarg2, long jarg3); + public final static native void IndexLSH_search(long jarg1, IndexLSH jarg1_, long jarg2, long jarg3, long jarg4, long jarg5, long jarg6, LongVector jarg6_); + public final static native void IndexLSH_transfer_thresholds(long jarg1, IndexLSH jarg1_, long jarg2, LinearTransform jarg2_); + public final static native void delete_IndexLSH(long jarg1); + public final static native long new_IndexLSH__SWIG_3(); + public final static native void IndexLSH_sa_encode(long jarg1, IndexLSH jarg1_, long jarg2, long jarg3, long jarg4); + public final static native void IndexLSH_sa_decode(long jarg1, IndexLSH jarg1_, long jarg2, long jarg3, long jarg4); + public final static native void SimulatedAnnealingParameters_init_temperature_set(long jarg1, SimulatedAnnealingParameters jarg1_, double jarg2); + public final static native double SimulatedAnnealingParameters_init_temperature_get(long jarg1, SimulatedAnnealingParameters jarg1_); + public final static native void SimulatedAnnealingParameters_temperature_decay_set(long jarg1, SimulatedAnnealingParameters jarg1_, double jarg2); + public final static native double SimulatedAnnealingParameters_temperature_decay_get(long jarg1, SimulatedAnnealingParameters jarg1_); + public final static native void SimulatedAnnealingParameters_n_iter_set(long jarg1, SimulatedAnnealingParameters jarg1_, int jarg2); + public final static native int SimulatedAnnealingParameters_n_iter_get(long jarg1, SimulatedAnnealingParameters jarg1_); + public final static native void SimulatedAnnealingParameters_n_redo_set(long jarg1, SimulatedAnnealingParameters jarg1_, int jarg2); + public final static native int SimulatedAnnealingParameters_n_redo_get(long jarg1, SimulatedAnnealingParameters jarg1_); + public final static native void SimulatedAnnealingParameters_seed_set(long jarg1, SimulatedAnnealingParameters jarg1_, int jarg2); + public final static native int SimulatedAnnealingParameters_seed_get(long jarg1, SimulatedAnnealingParameters jarg1_); + public final static native void SimulatedAnnealingParameters_verbose_set(long jarg1, SimulatedAnnealingParameters jarg1_, int jarg2); + public final static native int SimulatedAnnealingParameters_verbose_get(long jarg1, SimulatedAnnealingParameters jarg1_); + public final static native void SimulatedAnnealingParameters_only_bit_flips_set(long jarg1, SimulatedAnnealingParameters jarg1_, boolean jarg2); + public final static native boolean SimulatedAnnealingParameters_only_bit_flips_get(long jarg1, SimulatedAnnealingParameters jarg1_); + public final static native void SimulatedAnnealingParameters_init_random_set(long jarg1, SimulatedAnnealingParameters jarg1_, boolean jarg2); + public final static native boolean SimulatedAnnealingParameters_init_random_get(long jarg1, SimulatedAnnealingParameters jarg1_); + public final static native long new_SimulatedAnnealingParameters(); + public final static native void delete_SimulatedAnnealingParameters(long jarg1); + public final static native void PermutationObjective_n_set(long jarg1, PermutationObjective jarg1_, int jarg2); + public final static native int PermutationObjective_n_get(long jarg1, PermutationObjective jarg1_); + public final static native double PermutationObjective_compute_cost(long jarg1, PermutationObjective jarg1_, long jarg2); + public final static native double PermutationObjective_cost_update(long jarg1, PermutationObjective jarg1_, long jarg2, int jarg3, int jarg4); + public final static native void delete_PermutationObjective(long jarg1); + public final static native void ReproduceDistancesObjective_dis_weight_factor_set(long jarg1, ReproduceDistancesObjective jarg1_, double jarg2); + public final static native double ReproduceDistancesObjective_dis_weight_factor_get(long jarg1, ReproduceDistancesObjective jarg1_); + public final static native double ReproduceDistancesObjective_sqr(double jarg1); + public final static native double ReproduceDistancesObjective_dis_weight(long jarg1, ReproduceDistancesObjective jarg1_, double jarg2); + public final static native void ReproduceDistancesObjective_source_dis_set(long jarg1, ReproduceDistancesObjective jarg1_, long jarg2, DoubleVector jarg2_); + public final static native long ReproduceDistancesObjective_source_dis_get(long jarg1, ReproduceDistancesObjective jarg1_); + public final static native void ReproduceDistancesObjective_target_dis_set(long jarg1, ReproduceDistancesObjective jarg1_, long jarg2); + public final static native long ReproduceDistancesObjective_target_dis_get(long jarg1, ReproduceDistancesObjective jarg1_); + public final static native void ReproduceDistancesObjective_weights_set(long jarg1, ReproduceDistancesObjective jarg1_, long jarg2, DoubleVector jarg2_); + public final static native long ReproduceDistancesObjective_weights_get(long jarg1, ReproduceDistancesObjective jarg1_); + public final static native double ReproduceDistancesObjective_get_source_dis(long jarg1, ReproduceDistancesObjective jarg1_, int jarg2, int jarg3); + public final static native double ReproduceDistancesObjective_compute_cost(long jarg1, ReproduceDistancesObjective jarg1_, long jarg2); + public final static native double ReproduceDistancesObjective_cost_update(long jarg1, ReproduceDistancesObjective jarg1_, long jarg2, int jarg3, int jarg4); + public final static native long new_ReproduceDistancesObjective(int jarg1, long jarg2, long jarg3, double jarg4); + public final static native void ReproduceDistancesObjective_compute_mean_stdev(long jarg1, long jarg2, long jarg3, long jarg4); + public final static native void ReproduceDistancesObjective_set_affine_target_dis(long jarg1, ReproduceDistancesObjective jarg1_, long jarg2); + public final static native void delete_ReproduceDistancesObjective(long jarg1); + public final static native void SimulatedAnnealingOptimizer_obj_set(long jarg1, SimulatedAnnealingOptimizer jarg1_, long jarg2, PermutationObjective jarg2_); + public final static native long SimulatedAnnealingOptimizer_obj_get(long jarg1, SimulatedAnnealingOptimizer jarg1_); + public final static native void SimulatedAnnealingOptimizer_n_set(long jarg1, SimulatedAnnealingOptimizer jarg1_, int jarg2); + public final static native int SimulatedAnnealingOptimizer_n_get(long jarg1, SimulatedAnnealingOptimizer jarg1_); + public final static native void SimulatedAnnealingOptimizer_logfile_set(long jarg1, SimulatedAnnealingOptimizer jarg1_, long jarg2); + public final static native long SimulatedAnnealingOptimizer_logfile_get(long jarg1, SimulatedAnnealingOptimizer jarg1_); + public final static native long new_SimulatedAnnealingOptimizer(long jarg1, PermutationObjective jarg1_, long jarg2, SimulatedAnnealingParameters jarg2_); + public final static native void SimulatedAnnealingOptimizer_rnd_set(long jarg1, SimulatedAnnealingOptimizer jarg1_, long jarg2); + public final static native long SimulatedAnnealingOptimizer_rnd_get(long jarg1, SimulatedAnnealingOptimizer jarg1_); + public final static native void SimulatedAnnealingOptimizer_init_cost_set(long jarg1, SimulatedAnnealingOptimizer jarg1_, double jarg2); + public final static native double SimulatedAnnealingOptimizer_init_cost_get(long jarg1, SimulatedAnnealingOptimizer jarg1_); + public final static native double SimulatedAnnealingOptimizer_optimize(long jarg1, SimulatedAnnealingOptimizer jarg1_, long jarg2); + public final static native double SimulatedAnnealingOptimizer_run_optimization(long jarg1, SimulatedAnnealingOptimizer jarg1_, long jarg2); + public final static native void delete_SimulatedAnnealingOptimizer(long jarg1); + public final static native void PolysemousTraining_optimization_type_set(long jarg1, PolysemousTraining jarg1_, int jarg2); + public final static native int PolysemousTraining_optimization_type_get(long jarg1, PolysemousTraining jarg1_); + public final static native void PolysemousTraining_ntrain_permutation_set(long jarg1, PolysemousTraining jarg1_, int jarg2); + public final static native int PolysemousTraining_ntrain_permutation_get(long jarg1, PolysemousTraining jarg1_); + public final static native void PolysemousTraining_dis_weight_factor_set(long jarg1, PolysemousTraining jarg1_, double jarg2); + public final static native double PolysemousTraining_dis_weight_factor_get(long jarg1, PolysemousTraining jarg1_); + public final static native void PolysemousTraining_max_memory_set(long jarg1, PolysemousTraining jarg1_, long jarg2); + public final static native long PolysemousTraining_max_memory_get(long jarg1, PolysemousTraining jarg1_); + public final static native void PolysemousTraining_log_pattern_set(long jarg1, PolysemousTraining jarg1_, String jarg2); + public final static native String PolysemousTraining_log_pattern_get(long jarg1, PolysemousTraining jarg1_); + public final static native long new_PolysemousTraining(); + public final static native void PolysemousTraining_optimize_pq_for_hamming(long jarg1, PolysemousTraining jarg1_, long jarg2, ProductQuantizer jarg2_, long jarg3, long jarg4); + public final static native void PolysemousTraining_optimize_ranking(long jarg1, PolysemousTraining jarg1_, long jarg2, ProductQuantizer jarg2_, long jarg3, long jarg4); + public final static native void PolysemousTraining_optimize_reproduce_distances(long jarg1, PolysemousTraining jarg1_, long jarg2, ProductQuantizer jarg2_); + public final static native long PolysemousTraining_memory_usage_per_thread(long jarg1, PolysemousTraining jarg1_, long jarg2, ProductQuantizer jarg2_); + public final static native void delete_PolysemousTraining(long jarg1); + public final static native void IndexPQ_pq_set(long jarg1, IndexPQ jarg1_, long jarg2, ProductQuantizer jarg2_); + public final static native long IndexPQ_pq_get(long jarg1, IndexPQ jarg1_); + public final static native long new_IndexPQ__SWIG_0(int jarg1, long jarg2, long jarg3, int jarg4); + public final static native long new_IndexPQ__SWIG_1(int jarg1, long jarg2, long jarg3); + public final static native long new_IndexPQ__SWIG_2(); + public final static native void IndexPQ_train(long jarg1, IndexPQ jarg1_, long jarg2, long jarg3); + public final static native void IndexPQ_search(long jarg1, IndexPQ jarg1_, long jarg2, long jarg3, long jarg4, long jarg5, long jarg6, LongVector jarg6_); + public final static native void IndexPQ_sa_encode(long jarg1, IndexPQ jarg1_, long jarg2, long jarg3, long jarg4); + public final static native void IndexPQ_sa_decode(long jarg1, IndexPQ jarg1_, long jarg2, long jarg3, long jarg4); + public final static native long IndexPQ_get_distance_computer(long jarg1, IndexPQ jarg1_); + public final static native void IndexPQ_do_polysemous_training_set(long jarg1, IndexPQ jarg1_, boolean jarg2); + public final static native boolean IndexPQ_do_polysemous_training_get(long jarg1, IndexPQ jarg1_); + public final static native void IndexPQ_polysemous_training_set(long jarg1, IndexPQ jarg1_, long jarg2, PolysemousTraining jarg2_); + public final static native long IndexPQ_polysemous_training_get(long jarg1, IndexPQ jarg1_); + public final static native void IndexPQ_search_type_set(long jarg1, IndexPQ jarg1_, int jarg2); + public final static native int IndexPQ_search_type_get(long jarg1, IndexPQ jarg1_); + public final static native void IndexPQ_encode_signs_set(long jarg1, IndexPQ jarg1_, boolean jarg2); + public final static native boolean IndexPQ_encode_signs_get(long jarg1, IndexPQ jarg1_); + public final static native void IndexPQ_polysemous_ht_set(long jarg1, IndexPQ jarg1_, int jarg2); + public final static native int IndexPQ_polysemous_ht_get(long jarg1, IndexPQ jarg1_); + public final static native void IndexPQ_search_core_polysemous(long jarg1, IndexPQ jarg1_, long jarg2, long jarg3, long jarg4, long jarg5, long jarg6, LongVector jarg6_); + public final static native void IndexPQ_hamming_distance_histogram(long jarg1, IndexPQ jarg1_, long jarg2, long jarg3, long jarg4, long jarg5, long jarg6, LongVector jarg6_); + public final static native void IndexPQ_hamming_distance_table(long jarg1, IndexPQ jarg1_, long jarg2, long jarg3, long jarg4); + public final static native void delete_IndexPQ(long jarg1); + public final static native void IndexPQStats_nq_set(long jarg1, IndexPQStats jarg1_, long jarg2); + public final static native long IndexPQStats_nq_get(long jarg1, IndexPQStats jarg1_); + public final static native void IndexPQStats_ncode_set(long jarg1, IndexPQStats jarg1_, long jarg2); + public final static native long IndexPQStats_ncode_get(long jarg1, IndexPQStats jarg1_); + public final static native void IndexPQStats_n_hamming_pass_set(long jarg1, IndexPQStats jarg1_, long jarg2); + public final static native long IndexPQStats_n_hamming_pass_get(long jarg1, IndexPQStats jarg1_); + public final static native long new_IndexPQStats(); + public final static native void IndexPQStats_reset(long jarg1, IndexPQStats jarg1_); + public final static native void delete_IndexPQStats(long jarg1); + public final static native void indexPQ_stats_set(long jarg1, IndexPQStats jarg1_); + public final static native long indexPQ_stats_get(); + public final static native void MultiIndexQuantizer_pq_set(long jarg1, MultiIndexQuantizer jarg1_, long jarg2, ProductQuantizer jarg2_); + public final static native long MultiIndexQuantizer_pq_get(long jarg1, MultiIndexQuantizer jarg1_); + public final static native long new_MultiIndexQuantizer__SWIG_0(int jarg1, long jarg2, long jarg3); + public final static native void MultiIndexQuantizer_train(long jarg1, MultiIndexQuantizer jarg1_, long jarg2, long jarg3); + public final static native void MultiIndexQuantizer_search(long jarg1, MultiIndexQuantizer jarg1_, long jarg2, long jarg3, long jarg4, long jarg5, long jarg6, LongVector jarg6_); + public final static native void MultiIndexQuantizer_add(long jarg1, MultiIndexQuantizer jarg1_, long jarg2, long jarg3); + public final static native void MultiIndexQuantizer_reset(long jarg1, MultiIndexQuantizer jarg1_); + public final static native long new_MultiIndexQuantizer__SWIG_1(); + public final static native void MultiIndexQuantizer_reconstruct(long jarg1, MultiIndexQuantizer jarg1_, long jarg2, long jarg3); + public final static native void delete_MultiIndexQuantizer(long jarg1); + public final static native void MultiIndexQuantizer2_assign_indexes_set(long jarg1, MultiIndexQuantizer2 jarg1_, long jarg2); + public final static native long MultiIndexQuantizer2_assign_indexes_get(long jarg1, MultiIndexQuantizer2 jarg1_); + public final static native void MultiIndexQuantizer2_own_fields_set(long jarg1, MultiIndexQuantizer2 jarg1_, boolean jarg2); + public final static native boolean MultiIndexQuantizer2_own_fields_get(long jarg1, MultiIndexQuantizer2 jarg1_); + public final static native long new_MultiIndexQuantizer2__SWIG_0(int jarg1, long jarg2, long jarg3, long jarg4); + public final static native long new_MultiIndexQuantizer2__SWIG_1(int jarg1, long jarg2, long jarg3, Index jarg3_, long jarg4, Index jarg4_); + public final static native void MultiIndexQuantizer2_train(long jarg1, MultiIndexQuantizer2 jarg1_, long jarg2, long jarg3); + public final static native void MultiIndexQuantizer2_search(long jarg1, MultiIndexQuantizer2 jarg1_, long jarg2, long jarg3, long jarg4, long jarg5, long jarg6, LongVector jarg6_); + public final static native void delete_MultiIndexQuantizer2(long jarg1); + public final static native void InvertedLists_nlist_set(long jarg1, InvertedLists jarg1_, long jarg2); + public final static native long InvertedLists_nlist_get(long jarg1, InvertedLists jarg1_); + public final static native void InvertedLists_code_size_set(long jarg1, InvertedLists jarg1_, long jarg2); + public final static native long InvertedLists_code_size_get(long jarg1, InvertedLists jarg1_); + public final static native long InvertedLists_INVALID_CODE_SIZE_get(); + public final static native long InvertedLists_list_size(long jarg1, InvertedLists jarg1_, long jarg2); + public final static native long InvertedLists_get_codes(long jarg1, InvertedLists jarg1_, long jarg2); + public final static native long InvertedLists_get_ids(long jarg1, InvertedLists jarg1_, long jarg2); + public final static native void InvertedLists_release_codes(long jarg1, InvertedLists jarg1_, long jarg2, long jarg3); + public final static native void InvertedLists_release_ids(long jarg1, InvertedLists jarg1_, long jarg2, long jarg3, LongVector jarg3_); + public final static native long InvertedLists_get_single_id(long jarg1, InvertedLists jarg1_, long jarg2, long jarg3); + public final static native long InvertedLists_get_single_code(long jarg1, InvertedLists jarg1_, long jarg2, long jarg3); + public final static native void InvertedLists_prefetch_lists(long jarg1, InvertedLists jarg1_, long jarg2, LongVector jarg2_, int jarg3); + public final static native long InvertedLists_add_entry(long jarg1, InvertedLists jarg1_, long jarg2, long jarg3, long jarg4); + public final static native long InvertedLists_add_entries(long jarg1, InvertedLists jarg1_, long jarg2, long jarg3, long jarg4, LongVector jarg4_, long jarg5); + public final static native void InvertedLists_update_entry(long jarg1, InvertedLists jarg1_, long jarg2, long jarg3, long jarg4, long jarg5); + public final static native void InvertedLists_update_entries(long jarg1, InvertedLists jarg1_, long jarg2, long jarg3, long jarg4, long jarg5, LongVector jarg5_, long jarg6); + public final static native void InvertedLists_resize(long jarg1, InvertedLists jarg1_, long jarg2, long jarg3); + public final static native void InvertedLists_reset(long jarg1, InvertedLists jarg1_); + public final static native void InvertedLists_merge_from(long jarg1, InvertedLists jarg1_, long jarg2, InvertedLists jarg2_, long jarg3); + public final static native void delete_InvertedLists(long jarg1); + public final static native double InvertedLists_imbalance_factor(long jarg1, InvertedLists jarg1_); + public final static native void InvertedLists_print_stats(long jarg1, InvertedLists jarg1_); + public final static native long InvertedLists_compute_ntotal(long jarg1, InvertedLists jarg1_); + public final static native void InvertedLists_ScopedIds_il_set(long jarg1, InvertedLists.ScopedIds jarg1_, long jarg2, InvertedLists jarg2_); + public final static native long InvertedLists_ScopedIds_il_get(long jarg1, InvertedLists.ScopedIds jarg1_); + public final static native void InvertedLists_ScopedIds_ids_set(long jarg1, InvertedLists.ScopedIds jarg1_, long jarg2, LongVector jarg2_); + public final static native long InvertedLists_ScopedIds_ids_get(long jarg1, InvertedLists.ScopedIds jarg1_); + public final static native void InvertedLists_ScopedIds_list_no_set(long jarg1, InvertedLists.ScopedIds jarg1_, long jarg2); + public final static native long InvertedLists_ScopedIds_list_no_get(long jarg1, InvertedLists.ScopedIds jarg1_); + public final static native long new_InvertedLists_ScopedIds(long jarg1, InvertedLists jarg1_, long jarg2); + public final static native long InvertedLists_ScopedIds_get(long jarg1, InvertedLists.ScopedIds jarg1_); + public final static native void delete_InvertedLists_ScopedIds(long jarg1); + public final static native void InvertedLists_ScopedCodes_il_set(long jarg1, InvertedLists.ScopedCodes jarg1_, long jarg2, InvertedLists jarg2_); + public final static native long InvertedLists_ScopedCodes_il_get(long jarg1, InvertedLists.ScopedCodes jarg1_); + public final static native void InvertedLists_ScopedCodes_codes_set(long jarg1, InvertedLists.ScopedCodes jarg1_, long jarg2); + public final static native long InvertedLists_ScopedCodes_codes_get(long jarg1, InvertedLists.ScopedCodes jarg1_); + public final static native void InvertedLists_ScopedCodes_list_no_set(long jarg1, InvertedLists.ScopedCodes jarg1_, long jarg2); + public final static native long InvertedLists_ScopedCodes_list_no_get(long jarg1, InvertedLists.ScopedCodes jarg1_); + public final static native long new_InvertedLists_ScopedCodes__SWIG_0(long jarg1, InvertedLists jarg1_, long jarg2); + public final static native long new_InvertedLists_ScopedCodes__SWIG_1(long jarg1, InvertedLists jarg1_, long jarg2, long jarg3); + public final static native long InvertedLists_ScopedCodes_get(long jarg1, InvertedLists.ScopedCodes jarg1_); + public final static native void delete_InvertedLists_ScopedCodes(long jarg1); + public final static native void ArrayInvertedLists_codes_set(long jarg1, ArrayInvertedLists jarg1_, long jarg2, ByteVectorVector jarg2_); + public final static native long ArrayInvertedLists_codes_get(long jarg1, ArrayInvertedLists jarg1_); + public final static native void ArrayInvertedLists_ids_set(long jarg1, ArrayInvertedLists jarg1_, long jarg2); + public final static native long ArrayInvertedLists_ids_get(long jarg1, ArrayInvertedLists jarg1_); + public final static native long new_ArrayInvertedLists(long jarg1, long jarg2); + public final static native long ArrayInvertedLists_list_size(long jarg1, ArrayInvertedLists jarg1_, long jarg2); + public final static native long ArrayInvertedLists_get_codes(long jarg1, ArrayInvertedLists jarg1_, long jarg2); + public final static native long ArrayInvertedLists_get_ids(long jarg1, ArrayInvertedLists jarg1_, long jarg2); + public final static native long ArrayInvertedLists_add_entries(long jarg1, ArrayInvertedLists jarg1_, long jarg2, long jarg3, long jarg4, LongVector jarg4_, long jarg5); + public final static native void ArrayInvertedLists_update_entries(long jarg1, ArrayInvertedLists jarg1_, long jarg2, long jarg3, long jarg4, long jarg5, LongVector jarg5_, long jarg6); + public final static native void ArrayInvertedLists_resize(long jarg1, ArrayInvertedLists jarg1_, long jarg2, long jarg3); + public final static native void delete_ArrayInvertedLists(long jarg1); + public final static native long ReadOnlyInvertedLists_add_entries(long jarg1, ReadOnlyInvertedLists jarg1_, long jarg2, long jarg3, long jarg4, LongVector jarg4_, long jarg5); + public final static native void ReadOnlyInvertedLists_update_entries(long jarg1, ReadOnlyInvertedLists jarg1_, long jarg2, long jarg3, long jarg4, long jarg5, LongVector jarg5_, long jarg6); + public final static native void ReadOnlyInvertedLists_resize(long jarg1, ReadOnlyInvertedLists jarg1_, long jarg2, long jarg3); + public final static native void delete_ReadOnlyInvertedLists(long jarg1); + public final static native void HStackInvertedLists_ils_set(long jarg1, HStackInvertedLists jarg1_, long jarg2); + public final static native long HStackInvertedLists_ils_get(long jarg1, HStackInvertedLists jarg1_); + public final static native long new_HStackInvertedLists(int jarg1, long jarg2); + public final static native long HStackInvertedLists_list_size(long jarg1, HStackInvertedLists jarg1_, long jarg2); + public final static native long HStackInvertedLists_get_codes(long jarg1, HStackInvertedLists jarg1_, long jarg2); + public final static native long HStackInvertedLists_get_ids(long jarg1, HStackInvertedLists jarg1_, long jarg2); + public final static native void HStackInvertedLists_prefetch_lists(long jarg1, HStackInvertedLists jarg1_, long jarg2, LongVector jarg2_, int jarg3); + public final static native void HStackInvertedLists_release_codes(long jarg1, HStackInvertedLists jarg1_, long jarg2, long jarg3); + public final static native void HStackInvertedLists_release_ids(long jarg1, HStackInvertedLists jarg1_, long jarg2, long jarg3, LongVector jarg3_); + public final static native long HStackInvertedLists_get_single_id(long jarg1, HStackInvertedLists jarg1_, long jarg2, long jarg3); + public final static native long HStackInvertedLists_get_single_code(long jarg1, HStackInvertedLists jarg1_, long jarg2, long jarg3); + public final static native void delete_HStackInvertedLists(long jarg1); + public final static native void SliceInvertedLists_il_set(long jarg1, SliceInvertedLists jarg1_, long jarg2, InvertedLists jarg2_); + public final static native long SliceInvertedLists_il_get(long jarg1, SliceInvertedLists jarg1_); + public final static native void SliceInvertedLists_i0_set(long jarg1, SliceInvertedLists jarg1_, long jarg2); + public final static native long SliceInvertedLists_i0_get(long jarg1, SliceInvertedLists jarg1_); + public final static native void SliceInvertedLists_i1_set(long jarg1, SliceInvertedLists jarg1_, long jarg2); + public final static native long SliceInvertedLists_i1_get(long jarg1, SliceInvertedLists jarg1_); + public final static native long new_SliceInvertedLists(long jarg1, InvertedLists jarg1_, long jarg2, long jarg3); + public final static native long SliceInvertedLists_list_size(long jarg1, SliceInvertedLists jarg1_, long jarg2); + public final static native long SliceInvertedLists_get_codes(long jarg1, SliceInvertedLists jarg1_, long jarg2); + public final static native long SliceInvertedLists_get_ids(long jarg1, SliceInvertedLists jarg1_, long jarg2); + public final static native void SliceInvertedLists_release_codes(long jarg1, SliceInvertedLists jarg1_, long jarg2, long jarg3); + public final static native void SliceInvertedLists_release_ids(long jarg1, SliceInvertedLists jarg1_, long jarg2, long jarg3, LongVector jarg3_); + public final static native long SliceInvertedLists_get_single_id(long jarg1, SliceInvertedLists jarg1_, long jarg2, long jarg3); + public final static native long SliceInvertedLists_get_single_code(long jarg1, SliceInvertedLists jarg1_, long jarg2, long jarg3); + public final static native void SliceInvertedLists_prefetch_lists(long jarg1, SliceInvertedLists jarg1_, long jarg2, LongVector jarg2_, int jarg3); + public final static native void delete_SliceInvertedLists(long jarg1); + public final static native void VStackInvertedLists_ils_set(long jarg1, VStackInvertedLists jarg1_, long jarg2); + public final static native long VStackInvertedLists_ils_get(long jarg1, VStackInvertedLists jarg1_); + public final static native void VStackInvertedLists_cumsz_set(long jarg1, VStackInvertedLists jarg1_, long jarg2); + public final static native long VStackInvertedLists_cumsz_get(long jarg1, VStackInvertedLists jarg1_); + public final static native long new_VStackInvertedLists(int jarg1, long jarg2); + public final static native long VStackInvertedLists_list_size(long jarg1, VStackInvertedLists jarg1_, long jarg2); + public final static native long VStackInvertedLists_get_codes(long jarg1, VStackInvertedLists jarg1_, long jarg2); + public final static native long VStackInvertedLists_get_ids(long jarg1, VStackInvertedLists jarg1_, long jarg2); + public final static native void VStackInvertedLists_release_codes(long jarg1, VStackInvertedLists jarg1_, long jarg2, long jarg3); + public final static native void VStackInvertedLists_release_ids(long jarg1, VStackInvertedLists jarg1_, long jarg2, long jarg3, LongVector jarg3_); + public final static native long VStackInvertedLists_get_single_id(long jarg1, VStackInvertedLists jarg1_, long jarg2, long jarg3); + public final static native long VStackInvertedLists_get_single_code(long jarg1, VStackInvertedLists jarg1_, long jarg2, long jarg3); + public final static native void VStackInvertedLists_prefetch_lists(long jarg1, VStackInvertedLists jarg1_, long jarg2, LongVector jarg2_, int jarg3); + public final static native void delete_VStackInvertedLists(long jarg1); + public final static native void MaskedInvertedLists_il0_set(long jarg1, MaskedInvertedLists jarg1_, long jarg2, InvertedLists jarg2_); + public final static native long MaskedInvertedLists_il0_get(long jarg1, MaskedInvertedLists jarg1_); + public final static native void MaskedInvertedLists_il1_set(long jarg1, MaskedInvertedLists jarg1_, long jarg2, InvertedLists jarg2_); + public final static native long MaskedInvertedLists_il1_get(long jarg1, MaskedInvertedLists jarg1_); + public final static native long new_MaskedInvertedLists(long jarg1, InvertedLists jarg1_, long jarg2, InvertedLists jarg2_); + public final static native long MaskedInvertedLists_list_size(long jarg1, MaskedInvertedLists jarg1_, long jarg2); + public final static native long MaskedInvertedLists_get_codes(long jarg1, MaskedInvertedLists jarg1_, long jarg2); + public final static native long MaskedInvertedLists_get_ids(long jarg1, MaskedInvertedLists jarg1_, long jarg2); + public final static native void MaskedInvertedLists_release_codes(long jarg1, MaskedInvertedLists jarg1_, long jarg2, long jarg3); + public final static native void MaskedInvertedLists_release_ids(long jarg1, MaskedInvertedLists jarg1_, long jarg2, long jarg3, LongVector jarg3_); + public final static native long MaskedInvertedLists_get_single_id(long jarg1, MaskedInvertedLists jarg1_, long jarg2, long jarg3); + public final static native long MaskedInvertedLists_get_single_code(long jarg1, MaskedInvertedLists jarg1_, long jarg2, long jarg3); + public final static native void MaskedInvertedLists_prefetch_lists(long jarg1, MaskedInvertedLists jarg1_, long jarg2, LongVector jarg2_, int jarg3); + public final static native void delete_MaskedInvertedLists(long jarg1); + public final static native void StopWordsInvertedLists_il0_set(long jarg1, StopWordsInvertedLists jarg1_, long jarg2, InvertedLists jarg2_); + public final static native long StopWordsInvertedLists_il0_get(long jarg1, StopWordsInvertedLists jarg1_); + public final static native void StopWordsInvertedLists_maxsize_set(long jarg1, StopWordsInvertedLists jarg1_, long jarg2); + public final static native long StopWordsInvertedLists_maxsize_get(long jarg1, StopWordsInvertedLists jarg1_); + public final static native long new_StopWordsInvertedLists(long jarg1, InvertedLists jarg1_, long jarg2); + public final static native long StopWordsInvertedLists_list_size(long jarg1, StopWordsInvertedLists jarg1_, long jarg2); + public final static native long StopWordsInvertedLists_get_codes(long jarg1, StopWordsInvertedLists jarg1_, long jarg2); + public final static native long StopWordsInvertedLists_get_ids(long jarg1, StopWordsInvertedLists jarg1_, long jarg2); + public final static native void StopWordsInvertedLists_release_codes(long jarg1, StopWordsInvertedLists jarg1_, long jarg2, long jarg3); + public final static native void StopWordsInvertedLists_release_ids(long jarg1, StopWordsInvertedLists jarg1_, long jarg2, long jarg3, LongVector jarg3_); + public final static native long StopWordsInvertedLists_get_single_id(long jarg1, StopWordsInvertedLists jarg1_, long jarg2, long jarg3); + public final static native long StopWordsInvertedLists_get_single_code(long jarg1, StopWordsInvertedLists jarg1_, long jarg2, long jarg3); + public final static native void StopWordsInvertedLists_prefetch_lists(long jarg1, StopWordsInvertedLists jarg1_, long jarg2, LongVector jarg2_, int jarg3); + public final static native void delete_StopWordsInvertedLists(long jarg1); + public final static native void Level1Quantizer_quantizer_set(long jarg1, Level1Quantizer jarg1_, long jarg2, Index jarg2_); + public final static native long Level1Quantizer_quantizer_get(long jarg1, Level1Quantizer jarg1_); + public final static native void Level1Quantizer_nlist_set(long jarg1, Level1Quantizer jarg1_, long jarg2); + public final static native long Level1Quantizer_nlist_get(long jarg1, Level1Quantizer jarg1_); + public final static native void Level1Quantizer_quantizer_trains_alone_set(long jarg1, Level1Quantizer jarg1_, char jarg2); + public final static native char Level1Quantizer_quantizer_trains_alone_get(long jarg1, Level1Quantizer jarg1_); + public final static native void Level1Quantizer_own_fields_set(long jarg1, Level1Quantizer jarg1_, boolean jarg2); + public final static native boolean Level1Quantizer_own_fields_get(long jarg1, Level1Quantizer jarg1_); + public final static native void Level1Quantizer_cp_set(long jarg1, Level1Quantizer jarg1_, long jarg2, ClusteringParameters jarg2_); + public final static native long Level1Quantizer_cp_get(long jarg1, Level1Quantizer jarg1_); + public final static native void Level1Quantizer_clustering_index_set(long jarg1, Level1Quantizer jarg1_, long jarg2, Index jarg2_); + public final static native long Level1Quantizer_clustering_index_get(long jarg1, Level1Quantizer jarg1_); + public final static native void Level1Quantizer_train_q1(long jarg1, Level1Quantizer jarg1_, long jarg2, long jarg3, boolean jarg4, int jarg5); + public final static native long Level1Quantizer_coarse_code_size(long jarg1, Level1Quantizer jarg1_); + public final static native void Level1Quantizer_encode_listno(long jarg1, Level1Quantizer jarg1_, long jarg2, long jarg3); + public final static native long Level1Quantizer_decode_listno(long jarg1, Level1Quantizer jarg1_, long jarg2); + public final static native long new_Level1Quantizer__SWIG_0(long jarg1, Index jarg1_, long jarg2); + public final static native long new_Level1Quantizer__SWIG_1(); + public final static native void delete_Level1Quantizer(long jarg1); + public final static native void IVFSearchParameters_nprobe_set(long jarg1, IVFSearchParameters jarg1_, long jarg2); + public final static native long IVFSearchParameters_nprobe_get(long jarg1, IVFSearchParameters jarg1_); + public final static native void IVFSearchParameters_max_codes_set(long jarg1, IVFSearchParameters jarg1_, long jarg2); + public final static native long IVFSearchParameters_max_codes_get(long jarg1, IVFSearchParameters jarg1_); + public final static native long new_IVFSearchParameters(); + public final static native void delete_IVFSearchParameters(long jarg1); + public final static native void IndexIVF_invlists_set(long jarg1, IndexIVF jarg1_, long jarg2, InvertedLists jarg2_); + public final static native long IndexIVF_invlists_get(long jarg1, IndexIVF jarg1_); + public final static native void IndexIVF_own_invlists_set(long jarg1, IndexIVF jarg1_, boolean jarg2); + public final static native boolean IndexIVF_own_invlists_get(long jarg1, IndexIVF jarg1_); + public final static native void IndexIVF_code_size_set(long jarg1, IndexIVF jarg1_, long jarg2); + public final static native long IndexIVF_code_size_get(long jarg1, IndexIVF jarg1_); + public final static native void IndexIVF_nprobe_set(long jarg1, IndexIVF jarg1_, long jarg2); + public final static native long IndexIVF_nprobe_get(long jarg1, IndexIVF jarg1_); + public final static native void IndexIVF_max_codes_set(long jarg1, IndexIVF jarg1_, long jarg2); + public final static native long IndexIVF_max_codes_get(long jarg1, IndexIVF jarg1_); + public final static native void IndexIVF_parallel_mode_set(long jarg1, IndexIVF jarg1_, int jarg2); + public final static native int IndexIVF_parallel_mode_get(long jarg1, IndexIVF jarg1_); + public final static native int IndexIVF_PARALLEL_MODE_NO_HEAP_INIT_get(long jarg1, IndexIVF jarg1_); + public final static native void IndexIVF_direct_map_set(long jarg1, IndexIVF jarg1_, long jarg2); + public final static native long IndexIVF_direct_map_get(long jarg1, IndexIVF jarg1_); + public final static native void IndexIVF_reset(long jarg1, IndexIVF jarg1_); + public final static native void IndexIVF_train(long jarg1, IndexIVF jarg1_, long jarg2, long jarg3); + public final static native void IndexIVF_add(long jarg1, IndexIVF jarg1_, long jarg2, long jarg3); + public final static native void IndexIVF_add_with_ids(long jarg1, IndexIVF jarg1_, long jarg2, long jarg3, long jarg4, LongVector jarg4_); + public final static native void IndexIVF_add_core(long jarg1, IndexIVF jarg1_, long jarg2, long jarg3, long jarg4, LongVector jarg4_, long jarg5, LongVector jarg5_); + public final static native void IndexIVF_encode_vectors__SWIG_0(long jarg1, IndexIVF jarg1_, long jarg2, long jarg3, long jarg4, LongVector jarg4_, long jarg5, boolean jarg6); + public final static native void IndexIVF_encode_vectors__SWIG_1(long jarg1, IndexIVF jarg1_, long jarg2, long jarg3, long jarg4, LongVector jarg4_, long jarg5); + public final static native void IndexIVF_add_sa_codes(long jarg1, IndexIVF jarg1_, long jarg2, long jarg3, long jarg4, LongVector jarg4_); + public final static native void IndexIVF_train_residual(long jarg1, IndexIVF jarg1_, long jarg2, long jarg3); + public final static native void IndexIVF_search_preassigned__SWIG_0(long jarg1, IndexIVF jarg1_, long jarg2, long jarg3, long jarg4, long jarg5, LongVector jarg5_, long jarg6, long jarg7, long jarg8, LongVector jarg8_, boolean jarg9, long jarg10, IVFSearchParameters jarg10_, long jarg11, IndexIVFStats jarg11_); + public final static native void IndexIVF_search_preassigned__SWIG_1(long jarg1, IndexIVF jarg1_, long jarg2, long jarg3, long jarg4, long jarg5, LongVector jarg5_, long jarg6, long jarg7, long jarg8, LongVector jarg8_, boolean jarg9, long jarg10, IVFSearchParameters jarg10_); + public final static native void IndexIVF_search_preassigned__SWIG_2(long jarg1, IndexIVF jarg1_, long jarg2, long jarg3, long jarg4, long jarg5, LongVector jarg5_, long jarg6, long jarg7, long jarg8, LongVector jarg8_, boolean jarg9); + public final static native void IndexIVF_search(long jarg1, IndexIVF jarg1_, long jarg2, long jarg3, long jarg4, long jarg5, long jarg6, LongVector jarg6_); + public final static native void IndexIVF_range_search(long jarg1, IndexIVF jarg1_, long jarg2, long jarg3, float jarg4, long jarg5, RangeSearchResult jarg5_); + public final static native void IndexIVF_range_search_preassigned__SWIG_0(long jarg1, IndexIVF jarg1_, long jarg2, long jarg3, float jarg4, long jarg5, LongVector jarg5_, long jarg6, long jarg7, RangeSearchResult jarg7_, boolean jarg8, long jarg9, IVFSearchParameters jarg9_, long jarg10, IndexIVFStats jarg10_); + public final static native void IndexIVF_range_search_preassigned__SWIG_1(long jarg1, IndexIVF jarg1_, long jarg2, long jarg3, float jarg4, long jarg5, LongVector jarg5_, long jarg6, long jarg7, RangeSearchResult jarg7_, boolean jarg8, long jarg9, IVFSearchParameters jarg9_); + public final static native void IndexIVF_range_search_preassigned__SWIG_2(long jarg1, IndexIVF jarg1_, long jarg2, long jarg3, float jarg4, long jarg5, LongVector jarg5_, long jarg6, long jarg7, RangeSearchResult jarg7_, boolean jarg8); + public final static native void IndexIVF_range_search_preassigned__SWIG_3(long jarg1, IndexIVF jarg1_, long jarg2, long jarg3, float jarg4, long jarg5, LongVector jarg5_, long jarg6, long jarg7, RangeSearchResult jarg7_); + public final static native long IndexIVF_get_InvertedListScanner__SWIG_0(long jarg1, IndexIVF jarg1_, boolean jarg2); + public final static native long IndexIVF_get_InvertedListScanner__SWIG_1(long jarg1, IndexIVF jarg1_); + public final static native void IndexIVF_reconstruct(long jarg1, IndexIVF jarg1_, long jarg2, long jarg3); + public final static native void IndexIVF_update_vectors(long jarg1, IndexIVF jarg1_, int jarg2, long jarg3, LongVector jarg3_, long jarg4); + public final static native void IndexIVF_reconstruct_n(long jarg1, IndexIVF jarg1_, long jarg2, long jarg3, long jarg4); + public final static native void IndexIVF_search_and_reconstruct(long jarg1, IndexIVF jarg1_, long jarg2, long jarg3, long jarg4, long jarg5, long jarg6, LongVector jarg6_, long jarg7); + public final static native void IndexIVF_reconstruct_from_offset(long jarg1, IndexIVF jarg1_, long jarg2, long jarg3, long jarg4); + public final static native long IndexIVF_remove_ids(long jarg1, IndexIVF jarg1_, long jarg2, IDSelector jarg2_); + public final static native void IndexIVF_check_compatible_for_merge(long jarg1, IndexIVF jarg1_, long jarg2, IndexIVF jarg2_); + public final static native void IndexIVF_merge_from(long jarg1, IndexIVF jarg1_, long jarg2, IndexIVF jarg2_, long jarg3); + public final static native void IndexIVF_copy_subset_to(long jarg1, IndexIVF jarg1_, long jarg2, IndexIVF jarg2_, int jarg3, long jarg4, long jarg5); + public final static native void delete_IndexIVF(long jarg1); + public final static native long IndexIVF_get_list_size(long jarg1, IndexIVF jarg1_, long jarg2); + public final static native void IndexIVF_make_direct_map__SWIG_0(long jarg1, IndexIVF jarg1_, boolean jarg2); + public final static native void IndexIVF_make_direct_map__SWIG_1(long jarg1, IndexIVF jarg1_); + public final static native void IndexIVF_set_direct_map_type(long jarg1, IndexIVF jarg1_, long jarg2); + public final static native void IndexIVF_replace_invlists__SWIG_0(long jarg1, IndexIVF jarg1_, long jarg2, InvertedLists jarg2_, boolean jarg3); + public final static native void IndexIVF_replace_invlists__SWIG_1(long jarg1, IndexIVF jarg1_, long jarg2, InvertedLists jarg2_); + public final static native long IndexIVF_sa_code_size(long jarg1, IndexIVF jarg1_); + public final static native void IndexIVF_sa_encode(long jarg1, IndexIVF jarg1_, long jarg2, long jarg3, long jarg4); + public final static native void IndexIVFStats_nq_set(long jarg1, IndexIVFStats jarg1_, long jarg2); + public final static native long IndexIVFStats_nq_get(long jarg1, IndexIVFStats jarg1_); + public final static native void IndexIVFStats_nlist_set(long jarg1, IndexIVFStats jarg1_, long jarg2); + public final static native long IndexIVFStats_nlist_get(long jarg1, IndexIVFStats jarg1_); + public final static native void IndexIVFStats_ndis_set(long jarg1, IndexIVFStats jarg1_, long jarg2); + public final static native long IndexIVFStats_ndis_get(long jarg1, IndexIVFStats jarg1_); + public final static native void IndexIVFStats_nheap_updates_set(long jarg1, IndexIVFStats jarg1_, long jarg2); + public final static native long IndexIVFStats_nheap_updates_get(long jarg1, IndexIVFStats jarg1_); + public final static native void IndexIVFStats_quantization_time_set(long jarg1, IndexIVFStats jarg1_, double jarg2); + public final static native double IndexIVFStats_quantization_time_get(long jarg1, IndexIVFStats jarg1_); + public final static native void IndexIVFStats_search_time_set(long jarg1, IndexIVFStats jarg1_, double jarg2); + public final static native double IndexIVFStats_search_time_get(long jarg1, IndexIVFStats jarg1_); + public final static native long new_IndexIVFStats(); + public final static native void IndexIVFStats_reset(long jarg1, IndexIVFStats jarg1_); + public final static native void IndexIVFStats_add(long jarg1, IndexIVFStats jarg1_, long jarg2, IndexIVFStats jarg2_); + public final static native void delete_IndexIVFStats(long jarg1); + public final static native void indexIVF_stats_set(long jarg1, IndexIVFStats jarg1_); + public final static native long indexIVF_stats_get(); + public final static native short[] hamdis_tab_ham_bytes_get(); + public final static native void HammingComputer4_a0_set(long jarg1, HammingComputer4 jarg1_, long jarg2); + public final static native long HammingComputer4_a0_get(long jarg1, HammingComputer4 jarg1_); + public final static native long new_HammingComputer4__SWIG_0(); + public final static native long new_HammingComputer4__SWIG_1(long jarg1, int jarg2); + public final static native void HammingComputer4_set(long jarg1, HammingComputer4 jarg1_, long jarg2, int jarg3); + public final static native int HammingComputer4_hamming(long jarg1, HammingComputer4 jarg1_, long jarg2); + public final static native void delete_HammingComputer4(long jarg1); + public final static native void HammingComputer8_a0_set(long jarg1, HammingComputer8 jarg1_, long jarg2); + public final static native long HammingComputer8_a0_get(long jarg1, HammingComputer8 jarg1_); + public final static native long new_HammingComputer8__SWIG_0(); + public final static native long new_HammingComputer8__SWIG_1(long jarg1, int jarg2); + public final static native void HammingComputer8_set(long jarg1, HammingComputer8 jarg1_, long jarg2, int jarg3); + public final static native int HammingComputer8_hamming(long jarg1, HammingComputer8 jarg1_, long jarg2); + public final static native void delete_HammingComputer8(long jarg1); + public final static native void HammingComputer16_a0_set(long jarg1, HammingComputer16 jarg1_, long jarg2); + public final static native long HammingComputer16_a0_get(long jarg1, HammingComputer16 jarg1_); + public final static native void HammingComputer16_a1_set(long jarg1, HammingComputer16 jarg1_, long jarg2); + public final static native long HammingComputer16_a1_get(long jarg1, HammingComputer16 jarg1_); + public final static native long new_HammingComputer16__SWIG_0(); + public final static native long new_HammingComputer16__SWIG_1(long jarg1, int jarg2); + public final static native void HammingComputer16_set(long jarg1, HammingComputer16 jarg1_, long jarg2, int jarg3); + public final static native int HammingComputer16_hamming(long jarg1, HammingComputer16 jarg1_, long jarg2); + public final static native void delete_HammingComputer16(long jarg1); + public final static native void HammingComputer20_a0_set(long jarg1, HammingComputer20 jarg1_, long jarg2); + public final static native long HammingComputer20_a0_get(long jarg1, HammingComputer20 jarg1_); + public final static native void HammingComputer20_a1_set(long jarg1, HammingComputer20 jarg1_, long jarg2); + public final static native long HammingComputer20_a1_get(long jarg1, HammingComputer20 jarg1_); + public final static native void HammingComputer20_a2_set(long jarg1, HammingComputer20 jarg1_, long jarg2); + public final static native long HammingComputer20_a2_get(long jarg1, HammingComputer20 jarg1_); + public final static native long new_HammingComputer20__SWIG_0(); + public final static native long new_HammingComputer20__SWIG_1(long jarg1, int jarg2); + public final static native void HammingComputer20_set(long jarg1, HammingComputer20 jarg1_, long jarg2, int jarg3); + public final static native int HammingComputer20_hamming(long jarg1, HammingComputer20 jarg1_, long jarg2); + public final static native void delete_HammingComputer20(long jarg1); + public final static native void HammingComputer32_a0_set(long jarg1, HammingComputer32 jarg1_, long jarg2); + public final static native long HammingComputer32_a0_get(long jarg1, HammingComputer32 jarg1_); + public final static native void HammingComputer32_a1_set(long jarg1, HammingComputer32 jarg1_, long jarg2); + public final static native long HammingComputer32_a1_get(long jarg1, HammingComputer32 jarg1_); + public final static native void HammingComputer32_a2_set(long jarg1, HammingComputer32 jarg1_, long jarg2); + public final static native long HammingComputer32_a2_get(long jarg1, HammingComputer32 jarg1_); + public final static native void HammingComputer32_a3_set(long jarg1, HammingComputer32 jarg1_, long jarg2); + public final static native long HammingComputer32_a3_get(long jarg1, HammingComputer32 jarg1_); + public final static native long new_HammingComputer32__SWIG_0(); + public final static native long new_HammingComputer32__SWIG_1(long jarg1, int jarg2); + public final static native void HammingComputer32_set(long jarg1, HammingComputer32 jarg1_, long jarg2, int jarg3); + public final static native int HammingComputer32_hamming(long jarg1, HammingComputer32 jarg1_, long jarg2); + public final static native void delete_HammingComputer32(long jarg1); + public final static native void HammingComputer64_a0_set(long jarg1, HammingComputer64 jarg1_, long jarg2); + public final static native long HammingComputer64_a0_get(long jarg1, HammingComputer64 jarg1_); + public final static native void HammingComputer64_a1_set(long jarg1, HammingComputer64 jarg1_, long jarg2); + public final static native long HammingComputer64_a1_get(long jarg1, HammingComputer64 jarg1_); + public final static native void HammingComputer64_a2_set(long jarg1, HammingComputer64 jarg1_, long jarg2); + public final static native long HammingComputer64_a2_get(long jarg1, HammingComputer64 jarg1_); + public final static native void HammingComputer64_a3_set(long jarg1, HammingComputer64 jarg1_, long jarg2); + public final static native long HammingComputer64_a3_get(long jarg1, HammingComputer64 jarg1_); + public final static native void HammingComputer64_a4_set(long jarg1, HammingComputer64 jarg1_, long jarg2); + public final static native long HammingComputer64_a4_get(long jarg1, HammingComputer64 jarg1_); + public final static native void HammingComputer64_a5_set(long jarg1, HammingComputer64 jarg1_, long jarg2); + public final static native long HammingComputer64_a5_get(long jarg1, HammingComputer64 jarg1_); + public final static native void HammingComputer64_a6_set(long jarg1, HammingComputer64 jarg1_, long jarg2); + public final static native long HammingComputer64_a6_get(long jarg1, HammingComputer64 jarg1_); + public final static native void HammingComputer64_a7_set(long jarg1, HammingComputer64 jarg1_, long jarg2); + public final static native long HammingComputer64_a7_get(long jarg1, HammingComputer64 jarg1_); + public final static native long new_HammingComputer64__SWIG_0(); + public final static native long new_HammingComputer64__SWIG_1(long jarg1, int jarg2); + public final static native void HammingComputer64_set(long jarg1, HammingComputer64 jarg1_, long jarg2, int jarg3); + public final static native int HammingComputer64_hamming(long jarg1, HammingComputer64 jarg1_, long jarg2); + public final static native void delete_HammingComputer64(long jarg1); + public final static native void HammingComputerDefault_a8_set(long jarg1, HammingComputerDefault jarg1_, long jarg2); + public final static native long HammingComputerDefault_a8_get(long jarg1, HammingComputerDefault jarg1_); + public final static native void HammingComputerDefault_quotient8_set(long jarg1, HammingComputerDefault jarg1_, int jarg2); + public final static native int HammingComputerDefault_quotient8_get(long jarg1, HammingComputerDefault jarg1_); + public final static native void HammingComputerDefault_remainder8_set(long jarg1, HammingComputerDefault jarg1_, int jarg2); + public final static native int HammingComputerDefault_remainder8_get(long jarg1, HammingComputerDefault jarg1_); + public final static native long new_HammingComputerDefault__SWIG_0(); + public final static native long new_HammingComputerDefault__SWIG_1(long jarg1, int jarg2); + public final static native void HammingComputerDefault_set(long jarg1, HammingComputerDefault jarg1_, long jarg2, int jarg3); + public final static native int HammingComputerDefault_hamming(long jarg1, HammingComputerDefault jarg1_, long jarg2); + public final static native void delete_HammingComputerDefault(long jarg1); + public final static native void HammingComputerM8_a_set(long jarg1, HammingComputerM8 jarg1_, long jarg2); + public final static native long HammingComputerM8_a_get(long jarg1, HammingComputerM8 jarg1_); + public final static native void HammingComputerM8_n_set(long jarg1, HammingComputerM8 jarg1_, int jarg2); + public final static native int HammingComputerM8_n_get(long jarg1, HammingComputerM8 jarg1_); + public final static native long new_HammingComputerM8__SWIG_0(); + public final static native long new_HammingComputerM8__SWIG_1(long jarg1, int jarg2); + public final static native void HammingComputerM8_set(long jarg1, HammingComputerM8 jarg1_, long jarg2, int jarg3); + public final static native int HammingComputerM8_hamming(long jarg1, HammingComputerM8 jarg1_, long jarg2); + public final static native void delete_HammingComputerM8(long jarg1); + public final static native void HammingComputerM4_a_set(long jarg1, HammingComputerM4 jarg1_, long jarg2); + public final static native long HammingComputerM4_a_get(long jarg1, HammingComputerM4 jarg1_); + public final static native void HammingComputerM4_n_set(long jarg1, HammingComputerM4 jarg1_, int jarg2); + public final static native int HammingComputerM4_n_get(long jarg1, HammingComputerM4 jarg1_); + public final static native long new_HammingComputerM4__SWIG_0(); + public final static native long new_HammingComputerM4__SWIG_1(long jarg1, int jarg2); + public final static native void HammingComputerM4_set(long jarg1, HammingComputerM4 jarg1_, long jarg2, int jarg3); + public final static native int HammingComputerM4_hamming(long jarg1, HammingComputerM4 jarg1_, long jarg2); + public final static native void delete_HammingComputerM4(long jarg1); + public final static native int generalized_hamming_64(long jarg1); + public final static native void GenHammingComputer8_a0_set(long jarg1, GenHammingComputer8 jarg1_, long jarg2); + public final static native long GenHammingComputer8_a0_get(long jarg1, GenHammingComputer8 jarg1_); + public final static native long new_GenHammingComputer8(long jarg1, int jarg2); + public final static native int GenHammingComputer8_hamming(long jarg1, GenHammingComputer8 jarg1_, long jarg2); + public final static native void delete_GenHammingComputer8(long jarg1); + public final static native void GenHammingComputer16_a0_set(long jarg1, GenHammingComputer16 jarg1_, long jarg2); + public final static native long GenHammingComputer16_a0_get(long jarg1, GenHammingComputer16 jarg1_); + public final static native void GenHammingComputer16_a1_set(long jarg1, GenHammingComputer16 jarg1_, long jarg2); + public final static native long GenHammingComputer16_a1_get(long jarg1, GenHammingComputer16 jarg1_); + public final static native long new_GenHammingComputer16(long jarg1, int jarg2); + public final static native int GenHammingComputer16_hamming(long jarg1, GenHammingComputer16 jarg1_, long jarg2); + public final static native void delete_GenHammingComputer16(long jarg1); + public final static native void GenHammingComputer32_a0_set(long jarg1, GenHammingComputer32 jarg1_, long jarg2); + public final static native long GenHammingComputer32_a0_get(long jarg1, GenHammingComputer32 jarg1_); + public final static native void GenHammingComputer32_a1_set(long jarg1, GenHammingComputer32 jarg1_, long jarg2); + public final static native long GenHammingComputer32_a1_get(long jarg1, GenHammingComputer32 jarg1_); + public final static native void GenHammingComputer32_a2_set(long jarg1, GenHammingComputer32 jarg1_, long jarg2); + public final static native long GenHammingComputer32_a2_get(long jarg1, GenHammingComputer32 jarg1_); + public final static native void GenHammingComputer32_a3_set(long jarg1, GenHammingComputer32 jarg1_, long jarg2); + public final static native long GenHammingComputer32_a3_get(long jarg1, GenHammingComputer32 jarg1_); + public final static native long new_GenHammingComputer32(long jarg1, int jarg2); + public final static native int GenHammingComputer32_hamming(long jarg1, GenHammingComputer32 jarg1_, long jarg2); + public final static native void delete_GenHammingComputer32(long jarg1); + public final static native void GenHammingComputerM8_a_set(long jarg1, GenHammingComputerM8 jarg1_, long jarg2); + public final static native long GenHammingComputerM8_a_get(long jarg1, GenHammingComputerM8 jarg1_); + public final static native void GenHammingComputerM8_n_set(long jarg1, GenHammingComputerM8 jarg1_, int jarg2); + public final static native int GenHammingComputerM8_n_get(long jarg1, GenHammingComputerM8 jarg1_); + public final static native long new_GenHammingComputerM8(long jarg1, int jarg2); + public final static native int GenHammingComputerM8_hamming(long jarg1, GenHammingComputerM8 jarg1_, long jarg2); + public final static native void delete_GenHammingComputerM8(long jarg1); + public final static native void generalized_hammings_knn_hc__SWIG_0(long jarg1, long jarg2, long jarg3, long jarg4, long jarg5, int jarg6); + public final static native void generalized_hammings_knn_hc__SWIG_1(long jarg1, long jarg2, long jarg3, long jarg4, long jarg5); + public final static native void check_compatible_for_merge(long jarg1, Index jarg1_, long jarg2, Index jarg2_); + public final static native long extract_index_ivf__SWIG_0(long jarg1, Index jarg1_); + public final static native long try_extract_index_ivf__SWIG_0(long jarg1, Index jarg1_); + public final static native void merge_into(long jarg1, Index jarg1_, long jarg2, Index jarg2_, boolean jarg3); + public final static native void search_centroid(long jarg1, Index jarg1_, long jarg2, int jarg3, long jarg4, LongVector jarg4_); + public final static native void search_and_return_centroids(long jarg1, Index jarg1_, long jarg2, long jarg3, int jarg4, long jarg5, long jarg6, LongVector jarg6_, long jarg7, LongVector jarg7_, long jarg8, LongVector jarg8_); + public final static native void SlidingIndexWindow_index_set(long jarg1, SlidingIndexWindow jarg1_, long jarg2, Index jarg2_); + public final static native long SlidingIndexWindow_index_get(long jarg1, SlidingIndexWindow jarg1_); + public final static native void SlidingIndexWindow_ils_set(long jarg1, SlidingIndexWindow jarg1_, long jarg2, ArrayInvertedLists jarg2_); + public final static native long SlidingIndexWindow_ils_get(long jarg1, SlidingIndexWindow jarg1_); + public final static native void SlidingIndexWindow_n_slice_set(long jarg1, SlidingIndexWindow jarg1_, int jarg2); + public final static native int SlidingIndexWindow_n_slice_get(long jarg1, SlidingIndexWindow jarg1_); + public final static native void SlidingIndexWindow_nlist_set(long jarg1, SlidingIndexWindow jarg1_, long jarg2); + public final static native long SlidingIndexWindow_nlist_get(long jarg1, SlidingIndexWindow jarg1_); + public final static native void SlidingIndexWindow_sizes_set(long jarg1, SlidingIndexWindow jarg1_, long jarg2); + public final static native long SlidingIndexWindow_sizes_get(long jarg1, SlidingIndexWindow jarg1_); + public final static native long new_SlidingIndexWindow(long jarg1, Index jarg1_); + public final static native void SlidingIndexWindow_step(long jarg1, SlidingIndexWindow jarg1_, long jarg2, Index jarg2_, boolean jarg3); + public final static native void delete_SlidingIndexWindow(long jarg1); + public final static native long get_invlist_range(long jarg1, Index jarg1_, int jarg2, int jarg3); + public final static native void set_invlist_range(long jarg1, Index jarg1_, int jarg2, int jarg3, long jarg4, ArrayInvertedLists jarg4_); + public final static native void search_with_parameters__SWIG_0(long jarg1, Index jarg1_, long jarg2, long jarg3, long jarg4, long jarg5, long jarg6, LongVector jarg6_, long jarg7, IVFSearchParameters jarg7_, long jarg8, long jarg9); + public final static native void search_with_parameters__SWIG_1(long jarg1, Index jarg1_, long jarg2, long jarg3, long jarg4, long jarg5, long jarg6, LongVector jarg6_, long jarg7, IVFSearchParameters jarg7_, long jarg8); + public final static native void search_with_parameters__SWIG_2(long jarg1, Index jarg1_, long jarg2, long jarg3, long jarg4, long jarg5, long jarg6, LongVector jarg6_, long jarg7, IVFSearchParameters jarg7_); + public final static native void range_search_with_parameters__SWIG_0(long jarg1, Index jarg1_, long jarg2, long jarg3, float jarg4, long jarg5, RangeSearchResult jarg5_, long jarg6, IVFSearchParameters jarg6_, long jarg7, long jarg8); + public final static native void range_search_with_parameters__SWIG_1(long jarg1, Index jarg1_, long jarg2, long jarg3, float jarg4, long jarg5, RangeSearchResult jarg5_, long jarg6, IVFSearchParameters jarg6_, long jarg7); + public final static native void range_search_with_parameters__SWIG_2(long jarg1, Index jarg1_, long jarg2, long jarg3, float jarg4, long jarg5, RangeSearchResult jarg5_, long jarg6, IVFSearchParameters jarg6_); + public final static native void IndexScalarQuantizer_sq_set(long jarg1, IndexScalarQuantizer jarg1_, long jarg2); + public final static native long IndexScalarQuantizer_sq_get(long jarg1, IndexScalarQuantizer jarg1_); + public final static native long new_IndexScalarQuantizer__SWIG_0(int jarg1, long jarg2, int jarg3); + public final static native long new_IndexScalarQuantizer__SWIG_1(int jarg1, long jarg2); + public final static native long new_IndexScalarQuantizer__SWIG_2(); + public final static native void IndexScalarQuantizer_train(long jarg1, IndexScalarQuantizer jarg1_, long jarg2, long jarg3); + public final static native void IndexScalarQuantizer_search(long jarg1, IndexScalarQuantizer jarg1_, long jarg2, long jarg3, long jarg4, long jarg5, long jarg6, LongVector jarg6_); + public final static native long IndexScalarQuantizer_get_distance_computer(long jarg1, IndexScalarQuantizer jarg1_); + public final static native void IndexScalarQuantizer_sa_encode(long jarg1, IndexScalarQuantizer jarg1_, long jarg2, long jarg3, long jarg4); + public final static native void IndexScalarQuantizer_sa_decode(long jarg1, IndexScalarQuantizer jarg1_, long jarg2, long jarg3, long jarg4); + public final static native void delete_IndexScalarQuantizer(long jarg1); + public final static native void IndexIVFScalarQuantizer_sq_set(long jarg1, IndexIVFScalarQuantizer jarg1_, long jarg2); + public final static native long IndexIVFScalarQuantizer_sq_get(long jarg1, IndexIVFScalarQuantizer jarg1_); + public final static native void IndexIVFScalarQuantizer_by_residual_set(long jarg1, IndexIVFScalarQuantizer jarg1_, boolean jarg2); + public final static native boolean IndexIVFScalarQuantizer_by_residual_get(long jarg1, IndexIVFScalarQuantizer jarg1_); + public final static native long new_IndexIVFScalarQuantizer__SWIG_0(long jarg1, Index jarg1_, long jarg2, long jarg3, long jarg4, int jarg5, boolean jarg6); + public final static native long new_IndexIVFScalarQuantizer__SWIG_1(long jarg1, Index jarg1_, long jarg2, long jarg3, long jarg4, int jarg5); + public final static native long new_IndexIVFScalarQuantizer__SWIG_2(long jarg1, Index jarg1_, long jarg2, long jarg3, long jarg4); + public final static native long new_IndexIVFScalarQuantizer__SWIG_3(); + public final static native void IndexIVFScalarQuantizer_train_residual(long jarg1, IndexIVFScalarQuantizer jarg1_, long jarg2, long jarg3); + public final static native void IndexIVFScalarQuantizer_encode_vectors__SWIG_0(long jarg1, IndexIVFScalarQuantizer jarg1_, long jarg2, long jarg3, long jarg4, LongVector jarg4_, long jarg5, boolean jarg6); + public final static native void IndexIVFScalarQuantizer_encode_vectors__SWIG_1(long jarg1, IndexIVFScalarQuantizer jarg1_, long jarg2, long jarg3, long jarg4, LongVector jarg4_, long jarg5); + public final static native void IndexIVFScalarQuantizer_add_core(long jarg1, IndexIVFScalarQuantizer jarg1_, long jarg2, long jarg3, long jarg4, LongVector jarg4_, long jarg5, LongVector jarg5_); + public final static native long IndexIVFScalarQuantizer_get_InvertedListScanner(long jarg1, IndexIVFScalarQuantizer jarg1_, boolean jarg2); + public final static native void IndexIVFScalarQuantizer_reconstruct_from_offset(long jarg1, IndexIVFScalarQuantizer jarg1_, long jarg2, long jarg3, long jarg4); + public final static native void IndexIVFScalarQuantizer_sa_decode(long jarg1, IndexIVFScalarQuantizer jarg1_, long jarg2, long jarg3, long jarg4); + public final static native void delete_IndexIVFScalarQuantizer(long jarg1); + public final static native void HNSW_MinimaxHeap_n_set(long jarg1, HNSW.MinimaxHeap jarg1_, int jarg2); + public final static native int HNSW_MinimaxHeap_n_get(long jarg1, HNSW.MinimaxHeap jarg1_); + public final static native void HNSW_MinimaxHeap_k_set(long jarg1, HNSW.MinimaxHeap jarg1_, int jarg2); + public final static native int HNSW_MinimaxHeap_k_get(long jarg1, HNSW.MinimaxHeap jarg1_); + public final static native void HNSW_MinimaxHeap_nvalid_set(long jarg1, HNSW.MinimaxHeap jarg1_, int jarg2); + public final static native int HNSW_MinimaxHeap_nvalid_get(long jarg1, HNSW.MinimaxHeap jarg1_); + public final static native void HNSW_MinimaxHeap_ids_set(long jarg1, HNSW.MinimaxHeap jarg1_, long jarg2, IntVector jarg2_); + public final static native long HNSW_MinimaxHeap_ids_get(long jarg1, HNSW.MinimaxHeap jarg1_); + public final static native void HNSW_MinimaxHeap_dis_set(long jarg1, HNSW.MinimaxHeap jarg1_, long jarg2, FloatVector jarg2_); + public final static native long HNSW_MinimaxHeap_dis_get(long jarg1, HNSW.MinimaxHeap jarg1_); + public final static native long new_HNSW_MinimaxHeap(int jarg1); + public final static native void HNSW_MinimaxHeap_push(long jarg1, HNSW.MinimaxHeap jarg1_, int jarg2, float jarg3); + public final static native float HNSW_MinimaxHeap_max(long jarg1, HNSW.MinimaxHeap jarg1_); + public final static native int HNSW_MinimaxHeap_size(long jarg1, HNSW.MinimaxHeap jarg1_); + public final static native void HNSW_MinimaxHeap_clear(long jarg1, HNSW.MinimaxHeap jarg1_); + public final static native int HNSW_MinimaxHeap_pop_min__SWIG_0(long jarg1, HNSW.MinimaxHeap jarg1_, long jarg2); + public final static native int HNSW_MinimaxHeap_pop_min__SWIG_1(long jarg1, HNSW.MinimaxHeap jarg1_); + public final static native int HNSW_MinimaxHeap_count_below(long jarg1, HNSW.MinimaxHeap jarg1_, float jarg2); + public final static native void delete_HNSW_MinimaxHeap(long jarg1); + public final static native void HNSW_NodeDistCloser_d_set(long jarg1, HNSW.NodeDistCloser jarg1_, float jarg2); + public final static native float HNSW_NodeDistCloser_d_get(long jarg1, HNSW.NodeDistCloser jarg1_); + public final static native void HNSW_NodeDistCloser_id_set(long jarg1, HNSW.NodeDistCloser jarg1_, int jarg2); + public final static native int HNSW_NodeDistCloser_id_get(long jarg1, HNSW.NodeDistCloser jarg1_); + public final static native long new_HNSW_NodeDistCloser(float jarg1, int jarg2); + public final static native void delete_HNSW_NodeDistCloser(long jarg1); + public final static native void HNSW_NodeDistFarther_d_set(long jarg1, HNSW.NodeDistFarther jarg1_, float jarg2); + public final static native float HNSW_NodeDistFarther_d_get(long jarg1, HNSW.NodeDistFarther jarg1_); + public final static native void HNSW_NodeDistFarther_id_set(long jarg1, HNSW.NodeDistFarther jarg1_, int jarg2); + public final static native int HNSW_NodeDistFarther_id_get(long jarg1, HNSW.NodeDistFarther jarg1_); + public final static native long new_HNSW_NodeDistFarther(float jarg1, int jarg2); + public final static native void delete_HNSW_NodeDistFarther(long jarg1); + public final static native void HNSW_assign_probas_set(long jarg1, HNSW jarg1_, long jarg2, DoubleVector jarg2_); + public final static native long HNSW_assign_probas_get(long jarg1, HNSW jarg1_); + public final static native void HNSW_cum_nneighbor_per_level_set(long jarg1, HNSW jarg1_, long jarg2, IntVector jarg2_); + public final static native long HNSW_cum_nneighbor_per_level_get(long jarg1, HNSW jarg1_); + public final static native void HNSW_levels_set(long jarg1, HNSW jarg1_, long jarg2, IntVector jarg2_); + public final static native long HNSW_levels_get(long jarg1, HNSW jarg1_); + public final static native void HNSW_offsets_set(long jarg1, HNSW jarg1_, long jarg2, Uint64Vector jarg2_); + public final static native long HNSW_offsets_get(long jarg1, HNSW jarg1_); + public final static native void HNSW_neighbors_set(long jarg1, HNSW jarg1_, long jarg2, IntVector jarg2_); + public final static native long HNSW_neighbors_get(long jarg1, HNSW jarg1_); + public final static native void HNSW_entry_point_set(long jarg1, HNSW jarg1_, int jarg2); + public final static native int HNSW_entry_point_get(long jarg1, HNSW jarg1_); + public final static native void HNSW_rng_set(long jarg1, HNSW jarg1_, long jarg2); + public final static native long HNSW_rng_get(long jarg1, HNSW jarg1_); + public final static native void HNSW_max_level_set(long jarg1, HNSW jarg1_, int jarg2); + public final static native int HNSW_max_level_get(long jarg1, HNSW jarg1_); + public final static native void HNSW_efConstruction_set(long jarg1, HNSW jarg1_, int jarg2); + public final static native int HNSW_efConstruction_get(long jarg1, HNSW jarg1_); + public final static native void HNSW_efSearch_set(long jarg1, HNSW jarg1_, int jarg2); + public final static native int HNSW_efSearch_get(long jarg1, HNSW jarg1_); + public final static native void HNSW_check_relative_distance_set(long jarg1, HNSW jarg1_, boolean jarg2); + public final static native boolean HNSW_check_relative_distance_get(long jarg1, HNSW jarg1_); + public final static native void HNSW_upper_beam_set(long jarg1, HNSW jarg1_, int jarg2); + public final static native int HNSW_upper_beam_get(long jarg1, HNSW jarg1_); + public final static native void HNSW_search_bounded_queue_set(long jarg1, HNSW jarg1_, boolean jarg2); + public final static native boolean HNSW_search_bounded_queue_get(long jarg1, HNSW jarg1_); + public final static native void HNSW_set_default_probas(long jarg1, HNSW jarg1_, int jarg2, float jarg3); + public final static native void HNSW_set_nb_neighbors(long jarg1, HNSW jarg1_, int jarg2, int jarg3); + public final static native int HNSW_nb_neighbors(long jarg1, HNSW jarg1_, int jarg2); + public final static native int HNSW_cum_nb_neighbors(long jarg1, HNSW jarg1_, int jarg2); + public final static native void HNSW_neighbor_range(long jarg1, HNSW jarg1_, long jarg2, int jarg3, long jarg4, long jarg5); + public final static native long new_HNSW__SWIG_0(int jarg1); + public final static native long new_HNSW__SWIG_1(); + public final static native int HNSW_random_level(long jarg1, HNSW jarg1_); + public final static native void HNSW_fill_with_random_links(long jarg1, HNSW jarg1_, long jarg2); + public final static native void HNSW_add_links_starting_from(long jarg1, HNSW jarg1_, long jarg2, DistanceComputer jarg2_, int jarg3, int jarg4, float jarg5, int jarg6, long jarg7, long jarg8, VisitedTable jarg8_); + public final static native void HNSW_add_with_locks(long jarg1, HNSW jarg1_, long jarg2, DistanceComputer jarg2_, int jarg3, int jarg4, long jarg5, long jarg6, VisitedTable jarg6_); + public final static native int HNSW_search_from_candidates__SWIG_0(long jarg1, HNSW jarg1_, long jarg2, DistanceComputer jarg2_, int jarg3, long jarg4, LongVector jarg4_, long jarg5, long jarg6, HNSW.MinimaxHeap jarg6_, long jarg7, VisitedTable jarg7_, long jarg8, HNSWStats jarg8_, int jarg9, int jarg10); + public final static native int HNSW_search_from_candidates__SWIG_1(long jarg1, HNSW jarg1_, long jarg2, DistanceComputer jarg2_, int jarg3, long jarg4, LongVector jarg4_, long jarg5, long jarg6, HNSW.MinimaxHeap jarg6_, long jarg7, VisitedTable jarg7_, long jarg8, HNSWStats jarg8_, int jarg9); + public final static native long HNSW_search_from_candidate_unbounded(long jarg1, HNSW jarg1_, long jarg2, long jarg3, DistanceComputer jarg3_, int jarg4, long jarg5, VisitedTable jarg5_, long jarg6, HNSWStats jarg6_); + public final static native long HNSW_search(long jarg1, HNSW jarg1_, long jarg2, DistanceComputer jarg2_, int jarg3, long jarg4, LongVector jarg4_, long jarg5, long jarg6, VisitedTable jarg6_); + public final static native void HNSW_reset(long jarg1, HNSW jarg1_); + public final static native void HNSW_clear_neighbor_tables(long jarg1, HNSW jarg1_, int jarg2); + public final static native void HNSW_print_neighbor_stats(long jarg1, HNSW jarg1_, int jarg2); + public final static native int HNSW_prepare_level_tab__SWIG_0(long jarg1, HNSW jarg1_, long jarg2, boolean jarg3); + public final static native int HNSW_prepare_level_tab__SWIG_1(long jarg1, HNSW jarg1_, long jarg2); + public final static native void HNSW_shrink_neighbor_list(long jarg1, DistanceComputer jarg1_, long jarg2, long jarg3, int jarg4); + public final static native void delete_HNSW(long jarg1); + public final static native void HNSWStats_n1_set(long jarg1, HNSWStats jarg1_, long jarg2); + public final static native long HNSWStats_n1_get(long jarg1, HNSWStats jarg1_); + public final static native void HNSWStats_n2_set(long jarg1, HNSWStats jarg1_, long jarg2); + public final static native long HNSWStats_n2_get(long jarg1, HNSWStats jarg1_); + public final static native void HNSWStats_n3_set(long jarg1, HNSWStats jarg1_, long jarg2); + public final static native long HNSWStats_n3_get(long jarg1, HNSWStats jarg1_); + public final static native void HNSWStats_ndis_set(long jarg1, HNSWStats jarg1_, long jarg2); + public final static native long HNSWStats_ndis_get(long jarg1, HNSWStats jarg1_); + public final static native void HNSWStats_nreorder_set(long jarg1, HNSWStats jarg1_, long jarg2); + public final static native long HNSWStats_nreorder_get(long jarg1, HNSWStats jarg1_); + public final static native long new_HNSWStats__SWIG_0(long jarg1, long jarg2, long jarg3, long jarg4, long jarg5); + public final static native long new_HNSWStats__SWIG_1(long jarg1, long jarg2, long jarg3, long jarg4); + public final static native long new_HNSWStats__SWIG_2(long jarg1, long jarg2, long jarg3); + public final static native long new_HNSWStats__SWIG_3(long jarg1, long jarg2); + public final static native long new_HNSWStats__SWIG_4(long jarg1); + public final static native long new_HNSWStats__SWIG_5(); + public final static native void HNSWStats_reset(long jarg1, HNSWStats jarg1_); + public final static native void HNSWStats_combine(long jarg1, HNSWStats jarg1_, long jarg2, HNSWStats jarg2_); + public final static native void delete_HNSWStats(long jarg1); + public final static native void hnsw_stats_set(long jarg1, HNSWStats jarg1_); + public final static native long hnsw_stats_get(); + public final static native long ReconstructFromNeighbors_index_get(long jarg1, ReconstructFromNeighbors jarg1_); + public final static native void ReconstructFromNeighbors_M_set(long jarg1, ReconstructFromNeighbors jarg1_, long jarg2); + public final static native long ReconstructFromNeighbors_M_get(long jarg1, ReconstructFromNeighbors jarg1_); + public final static native void ReconstructFromNeighbors_k_set(long jarg1, ReconstructFromNeighbors jarg1_, long jarg2); + public final static native long ReconstructFromNeighbors_k_get(long jarg1, ReconstructFromNeighbors jarg1_); + public final static native void ReconstructFromNeighbors_nsq_set(long jarg1, ReconstructFromNeighbors jarg1_, long jarg2); + public final static native long ReconstructFromNeighbors_nsq_get(long jarg1, ReconstructFromNeighbors jarg1_); + public final static native void ReconstructFromNeighbors_code_size_set(long jarg1, ReconstructFromNeighbors jarg1_, long jarg2); + public final static native long ReconstructFromNeighbors_code_size_get(long jarg1, ReconstructFromNeighbors jarg1_); + public final static native void ReconstructFromNeighbors_k_reorder_set(long jarg1, ReconstructFromNeighbors jarg1_, int jarg2); + public final static native int ReconstructFromNeighbors_k_reorder_get(long jarg1, ReconstructFromNeighbors jarg1_); + public final static native void ReconstructFromNeighbors_codebook_set(long jarg1, ReconstructFromNeighbors jarg1_, long jarg2, FloatVector jarg2_); + public final static native long ReconstructFromNeighbors_codebook_get(long jarg1, ReconstructFromNeighbors jarg1_); + public final static native void ReconstructFromNeighbors_codes_set(long jarg1, ReconstructFromNeighbors jarg1_, long jarg2, ByteVector jarg2_); + public final static native long ReconstructFromNeighbors_codes_get(long jarg1, ReconstructFromNeighbors jarg1_); + public final static native void ReconstructFromNeighbors_ntotal_set(long jarg1, ReconstructFromNeighbors jarg1_, long jarg2); + public final static native long ReconstructFromNeighbors_ntotal_get(long jarg1, ReconstructFromNeighbors jarg1_); + public final static native void ReconstructFromNeighbors_d_set(long jarg1, ReconstructFromNeighbors jarg1_, long jarg2); + public final static native long ReconstructFromNeighbors_d_get(long jarg1, ReconstructFromNeighbors jarg1_); + public final static native void ReconstructFromNeighbors_dsub_set(long jarg1, ReconstructFromNeighbors jarg1_, long jarg2); + public final static native long ReconstructFromNeighbors_dsub_get(long jarg1, ReconstructFromNeighbors jarg1_); + public final static native long new_ReconstructFromNeighbors__SWIG_0(long jarg1, IndexHNSW jarg1_, long jarg2, long jarg3); + public final static native long new_ReconstructFromNeighbors__SWIG_1(long jarg1, IndexHNSW jarg1_, long jarg2); + public final static native long new_ReconstructFromNeighbors__SWIG_2(long jarg1, IndexHNSW jarg1_); + public final static native void ReconstructFromNeighbors_add_codes(long jarg1, ReconstructFromNeighbors jarg1_, long jarg2, long jarg3); + public final static native long ReconstructFromNeighbors_compute_distances(long jarg1, ReconstructFromNeighbors jarg1_, long jarg2, long jarg3, LongVector jarg3_, long jarg4, long jarg5); + public final static native void ReconstructFromNeighbors_estimate_code(long jarg1, ReconstructFromNeighbors jarg1_, long jarg2, int jarg3, long jarg4); + public final static native void ReconstructFromNeighbors_reconstruct(long jarg1, ReconstructFromNeighbors jarg1_, int jarg2, long jarg3, long jarg4); + public final static native void ReconstructFromNeighbors_reconstruct_n(long jarg1, ReconstructFromNeighbors jarg1_, int jarg2, int jarg3, long jarg4); + public final static native void ReconstructFromNeighbors_get_neighbor_table(long jarg1, ReconstructFromNeighbors jarg1_, int jarg2, long jarg3); + public final static native void delete_ReconstructFromNeighbors(long jarg1); + public final static native void IndexHNSW_hnsw_set(long jarg1, IndexHNSW jarg1_, long jarg2, HNSW jarg2_); + public final static native long IndexHNSW_hnsw_get(long jarg1, IndexHNSW jarg1_); + public final static native void IndexHNSW_own_fields_set(long jarg1, IndexHNSW jarg1_, boolean jarg2); + public final static native boolean IndexHNSW_own_fields_get(long jarg1, IndexHNSW jarg1_); + public final static native void IndexHNSW_storage_set(long jarg1, IndexHNSW jarg1_, long jarg2, Index jarg2_); + public final static native long IndexHNSW_storage_get(long jarg1, IndexHNSW jarg1_); + public final static native void IndexHNSW_reconstruct_from_neighbors_set(long jarg1, IndexHNSW jarg1_, long jarg2, ReconstructFromNeighbors jarg2_); + public final static native long IndexHNSW_reconstruct_from_neighbors_get(long jarg1, IndexHNSW jarg1_); + public final static native long new_IndexHNSW__SWIG_0(int jarg1, int jarg2, int jarg3); + public final static native long new_IndexHNSW__SWIG_1(int jarg1, int jarg2); + public final static native long new_IndexHNSW__SWIG_2(int jarg1); + public final static native long new_IndexHNSW__SWIG_3(); + public final static native long new_IndexHNSW__SWIG_4(long jarg1, Index jarg1_, int jarg2); + public final static native long new_IndexHNSW__SWIG_5(long jarg1, Index jarg1_); + public final static native void delete_IndexHNSW(long jarg1); + public final static native void IndexHNSW_add(long jarg1, IndexHNSW jarg1_, long jarg2, long jarg3); + public final static native void IndexHNSW_train(long jarg1, IndexHNSW jarg1_, long jarg2, long jarg3); + public final static native void IndexHNSW_search(long jarg1, IndexHNSW jarg1_, long jarg2, long jarg3, long jarg4, long jarg5, long jarg6, LongVector jarg6_); + public final static native void IndexHNSW_reconstruct(long jarg1, IndexHNSW jarg1_, long jarg2, long jarg3); + public final static native void IndexHNSW_reset(long jarg1, IndexHNSW jarg1_); + public final static native void IndexHNSW_shrink_level_0_neighbors(long jarg1, IndexHNSW jarg1_, int jarg2); + public final static native void IndexHNSW_search_level_0__SWIG_0(long jarg1, IndexHNSW jarg1_, long jarg2, long jarg3, long jarg4, long jarg5, long jarg6, long jarg7, long jarg8, LongVector jarg8_, int jarg9, int jarg10); + public final static native void IndexHNSW_search_level_0__SWIG_1(long jarg1, IndexHNSW jarg1_, long jarg2, long jarg3, long jarg4, long jarg5, long jarg6, long jarg7, long jarg8, LongVector jarg8_, int jarg9); + public final static native void IndexHNSW_search_level_0__SWIG_2(long jarg1, IndexHNSW jarg1_, long jarg2, long jarg3, long jarg4, long jarg5, long jarg6, long jarg7, long jarg8, LongVector jarg8_); + public final static native void IndexHNSW_init_level_0_from_knngraph(long jarg1, IndexHNSW jarg1_, int jarg2, long jarg3, long jarg4, LongVector jarg4_); + public final static native void IndexHNSW_init_level_0_from_entry_points(long jarg1, IndexHNSW jarg1_, int jarg2, long jarg3, long jarg4); + public final static native void IndexHNSW_reorder_links(long jarg1, IndexHNSW jarg1_); + public final static native void IndexHNSW_link_singletons(long jarg1, IndexHNSW jarg1_); + public final static native long new_IndexHNSWFlat__SWIG_0(); + public final static native long new_IndexHNSWFlat__SWIG_1(int jarg1, int jarg2, int jarg3); + public final static native long new_IndexHNSWFlat__SWIG_2(int jarg1, int jarg2); + public final static native void delete_IndexHNSWFlat(long jarg1); + public final static native long new_IndexHNSWPQ__SWIG_0(); + public final static native long new_IndexHNSWPQ__SWIG_1(int jarg1, int jarg2, int jarg3); + public final static native void IndexHNSWPQ_train(long jarg1, IndexHNSWPQ jarg1_, long jarg2, long jarg3); + public final static native void delete_IndexHNSWPQ(long jarg1); + public final static native long new_IndexHNSWSQ__SWIG_0(); + public final static native long new_IndexHNSWSQ__SWIG_1(int jarg1, long jarg2, int jarg3, int jarg4); + public final static native long new_IndexHNSWSQ__SWIG_2(int jarg1, long jarg2, int jarg3); + public final static native void delete_IndexHNSWSQ(long jarg1); + public final static native long new_IndexHNSW2Level__SWIG_0(); + public final static native long new_IndexHNSW2Level__SWIG_1(long jarg1, Index jarg1_, long jarg2, int jarg3, int jarg4); + public final static native void IndexHNSW2Level_flip_to_ivf(long jarg1, IndexHNSW2Level jarg1_); + public final static native void IndexHNSW2Level_search(long jarg1, IndexHNSW2Level jarg1_, long jarg2, long jarg3, long jarg4, long jarg5, long jarg6, LongVector jarg6_); + public final static native void delete_IndexHNSW2Level(long jarg1); + public final static native long new_IndexIVFFlat__SWIG_0(long jarg1, Index jarg1_, long jarg2, long jarg3, int jarg4); + public final static native long new_IndexIVFFlat__SWIG_1(long jarg1, Index jarg1_, long jarg2, long jarg3); + public final static native void IndexIVFFlat_add_core(long jarg1, IndexIVFFlat jarg1_, long jarg2, long jarg3, long jarg4, LongVector jarg4_, long jarg5, LongVector jarg5_); + public final static native void IndexIVFFlat_encode_vectors__SWIG_0(long jarg1, IndexIVFFlat jarg1_, long jarg2, long jarg3, long jarg4, LongVector jarg4_, long jarg5, boolean jarg6); + public final static native void IndexIVFFlat_encode_vectors__SWIG_1(long jarg1, IndexIVFFlat jarg1_, long jarg2, long jarg3, long jarg4, LongVector jarg4_, long jarg5); + public final static native long IndexIVFFlat_get_InvertedListScanner(long jarg1, IndexIVFFlat jarg1_, boolean jarg2); + public final static native void IndexIVFFlat_reconstruct_from_offset(long jarg1, IndexIVFFlat jarg1_, long jarg2, long jarg3, long jarg4); + public final static native void IndexIVFFlat_sa_decode(long jarg1, IndexIVFFlat jarg1_, long jarg2, long jarg3, long jarg4); + public final static native long new_IndexIVFFlat__SWIG_2(); + public final static native void delete_IndexIVFFlat(long jarg1); + public final static native void IndexIVFFlatDedup_instances_set(long jarg1, IndexIVFFlatDedup jarg1_, long jarg2); + public final static native long IndexIVFFlatDedup_instances_get(long jarg1, IndexIVFFlatDedup jarg1_); + public final static native long new_IndexIVFFlatDedup__SWIG_0(long jarg1, Index jarg1_, long jarg2, long jarg3, int jarg4); + public final static native long new_IndexIVFFlatDedup__SWIG_1(long jarg1, Index jarg1_, long jarg2, long jarg3); + public final static native void IndexIVFFlatDedup_train(long jarg1, IndexIVFFlatDedup jarg1_, long jarg2, long jarg3); + public final static native void IndexIVFFlatDedup_add_with_ids(long jarg1, IndexIVFFlatDedup jarg1_, long jarg2, long jarg3, long jarg4, LongVector jarg4_); + public final static native void IndexIVFFlatDedup_search_preassigned__SWIG_0(long jarg1, IndexIVFFlatDedup jarg1_, long jarg2, long jarg3, long jarg4, long jarg5, LongVector jarg5_, long jarg6, long jarg7, long jarg8, LongVector jarg8_, boolean jarg9, long jarg10, IVFSearchParameters jarg10_, long jarg11, IndexIVFStats jarg11_); + public final static native void IndexIVFFlatDedup_search_preassigned__SWIG_1(long jarg1, IndexIVFFlatDedup jarg1_, long jarg2, long jarg3, long jarg4, long jarg5, LongVector jarg5_, long jarg6, long jarg7, long jarg8, LongVector jarg8_, boolean jarg9, long jarg10, IVFSearchParameters jarg10_); + public final static native void IndexIVFFlatDedup_search_preassigned__SWIG_2(long jarg1, IndexIVFFlatDedup jarg1_, long jarg2, long jarg3, long jarg4, long jarg5, LongVector jarg5_, long jarg6, long jarg7, long jarg8, LongVector jarg8_, boolean jarg9); + public final static native long IndexIVFFlatDedup_remove_ids(long jarg1, IndexIVFFlatDedup jarg1_, long jarg2, IDSelector jarg2_); + public final static native void IndexIVFFlatDedup_range_search(long jarg1, IndexIVFFlatDedup jarg1_, long jarg2, long jarg3, float jarg4, long jarg5, RangeSearchResult jarg5_); + public final static native void IndexIVFFlatDedup_update_vectors(long jarg1, IndexIVFFlatDedup jarg1_, int jarg2, long jarg3, LongVector jarg3_, long jarg4); + public final static native void IndexIVFFlatDedup_reconstruct_from_offset(long jarg1, IndexIVFFlatDedup jarg1_, long jarg2, long jarg3, long jarg4); + public final static native long new_IndexIVFFlatDedup__SWIG_2(); + public final static native void delete_IndexIVFFlatDedup(long jarg1); + public final static native void OnDiskOneList_size_set(long jarg1, OnDiskOneList jarg1_, long jarg2); + public final static native long OnDiskOneList_size_get(long jarg1, OnDiskOneList jarg1_); + public final static native void OnDiskOneList_capacity_set(long jarg1, OnDiskOneList jarg1_, long jarg2); + public final static native long OnDiskOneList_capacity_get(long jarg1, OnDiskOneList jarg1_); + public final static native void OnDiskOneList_offset_set(long jarg1, OnDiskOneList jarg1_, long jarg2); + public final static native long OnDiskOneList_offset_get(long jarg1, OnDiskOneList jarg1_); + public final static native long new_OnDiskOneList(); + public final static native void delete_OnDiskOneList(long jarg1); + public final static native void OnDiskInvertedLists_lists_set(long jarg1, OnDiskInvertedLists jarg1_, long jarg2); + public final static native long OnDiskInvertedLists_lists_get(long jarg1, OnDiskInvertedLists jarg1_); + public final static native void OnDiskInvertedLists_Slot_offset_set(long jarg1, OnDiskInvertedLists.Slot jarg1_, long jarg2); + public final static native long OnDiskInvertedLists_Slot_offset_get(long jarg1, OnDiskInvertedLists.Slot jarg1_); + public final static native void OnDiskInvertedLists_Slot_capacity_set(long jarg1, OnDiskInvertedLists.Slot jarg1_, long jarg2); + public final static native long OnDiskInvertedLists_Slot_capacity_get(long jarg1, OnDiskInvertedLists.Slot jarg1_); + public final static native long new_OnDiskInvertedLists_Slot__SWIG_0(long jarg1, long jarg2); + public final static native long new_OnDiskInvertedLists_Slot__SWIG_1(); + public final static native void delete_OnDiskInvertedLists_Slot(long jarg1); + public final static native void OnDiskInvertedLists_slots_set(long jarg1, OnDiskInvertedLists jarg1_, long jarg2); + public final static native long OnDiskInvertedLists_slots_get(long jarg1, OnDiskInvertedLists jarg1_); + public final static native void OnDiskInvertedLists_filename_set(long jarg1, OnDiskInvertedLists jarg1_, String jarg2); + public final static native String OnDiskInvertedLists_filename_get(long jarg1, OnDiskInvertedLists jarg1_); + public final static native void OnDiskInvertedLists_totsize_set(long jarg1, OnDiskInvertedLists jarg1_, long jarg2); + public final static native long OnDiskInvertedLists_totsize_get(long jarg1, OnDiskInvertedLists jarg1_); + public final static native void OnDiskInvertedLists_ptr_set(long jarg1, OnDiskInvertedLists jarg1_, long jarg2); + public final static native long OnDiskInvertedLists_ptr_get(long jarg1, OnDiskInvertedLists jarg1_); + public final static native void OnDiskInvertedLists_read_only_set(long jarg1, OnDiskInvertedLists jarg1_, boolean jarg2); + public final static native boolean OnDiskInvertedLists_read_only_get(long jarg1, OnDiskInvertedLists jarg1_); + public final static native long new_OnDiskInvertedLists__SWIG_0(long jarg1, long jarg2, String jarg3); + public final static native long OnDiskInvertedLists_list_size(long jarg1, OnDiskInvertedLists jarg1_, long jarg2); + public final static native long OnDiskInvertedLists_get_codes(long jarg1, OnDiskInvertedLists jarg1_, long jarg2); + public final static native long OnDiskInvertedLists_get_ids(long jarg1, OnDiskInvertedLists jarg1_, long jarg2); + public final static native long OnDiskInvertedLists_add_entries(long jarg1, OnDiskInvertedLists jarg1_, long jarg2, long jarg3, long jarg4, LongVector jarg4_, long jarg5); + public final static native void OnDiskInvertedLists_update_entries(long jarg1, OnDiskInvertedLists jarg1_, long jarg2, long jarg3, long jarg4, long jarg5, LongVector jarg5_, long jarg6); + public final static native void OnDiskInvertedLists_resize(long jarg1, OnDiskInvertedLists jarg1_, long jarg2, long jarg3); + public final static native long OnDiskInvertedLists_merge_from__SWIG_0(long jarg1, OnDiskInvertedLists jarg1_, long jarg2, int jarg3, boolean jarg4); + public final static native long OnDiskInvertedLists_merge_from__SWIG_1(long jarg1, OnDiskInvertedLists jarg1_, long jarg2, int jarg3); + public final static native long OnDiskInvertedLists_merge_from_1__SWIG_0(long jarg1, OnDiskInvertedLists jarg1_, long jarg2, InvertedLists jarg2_, boolean jarg3); + public final static native long OnDiskInvertedLists_merge_from_1__SWIG_1(long jarg1, OnDiskInvertedLists jarg1_, long jarg2, InvertedLists jarg2_); + public final static native void OnDiskInvertedLists_crop_invlists(long jarg1, OnDiskInvertedLists jarg1_, long jarg2, long jarg3); + public final static native void OnDiskInvertedLists_prefetch_lists(long jarg1, OnDiskInvertedLists jarg1_, long jarg2, LongVector jarg2_, int jarg3); + public final static native void delete_OnDiskInvertedLists(long jarg1); + public final static native void OnDiskInvertedLists_locks_set(long jarg1, OnDiskInvertedLists jarg1_, long jarg2); + public final static native long OnDiskInvertedLists_locks_get(long jarg1, OnDiskInvertedLists jarg1_); + public final static native void OnDiskInvertedLists_pf_set(long jarg1, OnDiskInvertedLists jarg1_, long jarg2); + public final static native long OnDiskInvertedLists_pf_get(long jarg1, OnDiskInvertedLists jarg1_); + public final static native void OnDiskInvertedLists_prefetch_nthread_set(long jarg1, OnDiskInvertedLists jarg1_, int jarg2); + public final static native int OnDiskInvertedLists_prefetch_nthread_get(long jarg1, OnDiskInvertedLists jarg1_); + public final static native void OnDiskInvertedLists_do_mmap(long jarg1, OnDiskInvertedLists jarg1_); + public final static native void OnDiskInvertedLists_update_totsize(long jarg1, OnDiskInvertedLists jarg1_, long jarg2); + public final static native void OnDiskInvertedLists_resize_locked(long jarg1, OnDiskInvertedLists jarg1_, long jarg2, long jarg3); + public final static native long OnDiskInvertedLists_allocate_slot(long jarg1, OnDiskInvertedLists jarg1_, long jarg2); + public final static native void OnDiskInvertedLists_free_slot(long jarg1, OnDiskInvertedLists jarg1_, long jarg2, long jarg3); + public final static native void OnDiskInvertedLists_set_all_lists_sizes(long jarg1, OnDiskInvertedLists jarg1_, long jarg2); + public final static native long new_OnDiskInvertedLists__SWIG_1(); + public final static native long new_OnDiskInvertedListsIOHook(); + public final static native void OnDiskInvertedListsIOHook_write(long jarg1, OnDiskInvertedListsIOHook jarg1_, long jarg2, InvertedLists jarg2_, long jarg3); + public final static native long OnDiskInvertedListsIOHook_read(long jarg1, OnDiskInvertedListsIOHook jarg1_, long jarg2, int jarg3); + public final static native long OnDiskInvertedListsIOHook_read_ArrayInvertedLists(long jarg1, OnDiskInvertedListsIOHook jarg1_, long jarg2, int jarg3, long jarg4, long jarg5, long jarg6, Uint64Vector jarg6_); + public final static native void delete_OnDiskInvertedListsIOHook(long jarg1); + public final static native void IVFPQSearchParameters_scan_table_threshold_set(long jarg1, IVFPQSearchParameters jarg1_, long jarg2); + public final static native long IVFPQSearchParameters_scan_table_threshold_get(long jarg1, IVFPQSearchParameters jarg1_); + public final static native void IVFPQSearchParameters_polysemous_ht_set(long jarg1, IVFPQSearchParameters jarg1_, int jarg2); + public final static native int IVFPQSearchParameters_polysemous_ht_get(long jarg1, IVFPQSearchParameters jarg1_); + public final static native long new_IVFPQSearchParameters(); + public final static native void delete_IVFPQSearchParameters(long jarg1); + public final static native void precomputed_table_max_bytes_set(long jarg1); + public final static native long precomputed_table_max_bytes_get(); + public final static native void IndexIVFPQ_by_residual_set(long jarg1, IndexIVFPQ jarg1_, boolean jarg2); + public final static native boolean IndexIVFPQ_by_residual_get(long jarg1, IndexIVFPQ jarg1_); + public final static native void IndexIVFPQ_pq_set(long jarg1, IndexIVFPQ jarg1_, long jarg2, ProductQuantizer jarg2_); + public final static native long IndexIVFPQ_pq_get(long jarg1, IndexIVFPQ jarg1_); + public final static native void IndexIVFPQ_do_polysemous_training_set(long jarg1, IndexIVFPQ jarg1_, boolean jarg2); + public final static native boolean IndexIVFPQ_do_polysemous_training_get(long jarg1, IndexIVFPQ jarg1_); + public final static native void IndexIVFPQ_polysemous_training_set(long jarg1, IndexIVFPQ jarg1_, long jarg2, PolysemousTraining jarg2_); + public final static native long IndexIVFPQ_polysemous_training_get(long jarg1, IndexIVFPQ jarg1_); + public final static native void IndexIVFPQ_scan_table_threshold_set(long jarg1, IndexIVFPQ jarg1_, long jarg2); + public final static native long IndexIVFPQ_scan_table_threshold_get(long jarg1, IndexIVFPQ jarg1_); + public final static native void IndexIVFPQ_polysemous_ht_set(long jarg1, IndexIVFPQ jarg1_, int jarg2); + public final static native int IndexIVFPQ_polysemous_ht_get(long jarg1, IndexIVFPQ jarg1_); + public final static native void IndexIVFPQ_use_precomputed_table_set(long jarg1, IndexIVFPQ jarg1_, int jarg2); + public final static native int IndexIVFPQ_use_precomputed_table_get(long jarg1, IndexIVFPQ jarg1_); + public final static native void IndexIVFPQ_precomputed_table_set(long jarg1, IndexIVFPQ jarg1_, long jarg2); + public final static native long IndexIVFPQ_precomputed_table_get(long jarg1, IndexIVFPQ jarg1_); + public final static native long new_IndexIVFPQ__SWIG_0(long jarg1, Index jarg1_, long jarg2, long jarg3, long jarg4, long jarg5, int jarg6); + public final static native long new_IndexIVFPQ__SWIG_1(long jarg1, Index jarg1_, long jarg2, long jarg3, long jarg4, long jarg5); + public final static native void IndexIVFPQ_encode_vectors__SWIG_0(long jarg1, IndexIVFPQ jarg1_, long jarg2, long jarg3, long jarg4, LongVector jarg4_, long jarg5, boolean jarg6); + public final static native void IndexIVFPQ_encode_vectors__SWIG_1(long jarg1, IndexIVFPQ jarg1_, long jarg2, long jarg3, long jarg4, LongVector jarg4_, long jarg5); + public final static native void IndexIVFPQ_sa_decode(long jarg1, IndexIVFPQ jarg1_, long jarg2, long jarg3, long jarg4); + public final static native void IndexIVFPQ_add_core(long jarg1, IndexIVFPQ jarg1_, long jarg2, long jarg3, long jarg4, LongVector jarg4_, long jarg5, LongVector jarg5_); + public final static native void IndexIVFPQ_add_core_o__SWIG_0(long jarg1, IndexIVFPQ jarg1_, long jarg2, long jarg3, long jarg4, LongVector jarg4_, long jarg5, long jarg6, LongVector jarg6_); + public final static native void IndexIVFPQ_add_core_o__SWIG_1(long jarg1, IndexIVFPQ jarg1_, long jarg2, long jarg3, long jarg4, LongVector jarg4_, long jarg5); + public final static native void IndexIVFPQ_train_residual(long jarg1, IndexIVFPQ jarg1_, long jarg2, long jarg3); + public final static native void IndexIVFPQ_train_residual_o(long jarg1, IndexIVFPQ jarg1_, long jarg2, long jarg3, long jarg4); + public final static native void IndexIVFPQ_reconstruct_from_offset(long jarg1, IndexIVFPQ jarg1_, long jarg2, long jarg3, long jarg4); + public final static native long IndexIVFPQ_find_duplicates(long jarg1, IndexIVFPQ jarg1_, long jarg2, LongVector jarg2_, long jarg3); + public final static native void IndexIVFPQ_encode(long jarg1, IndexIVFPQ jarg1_, long jarg2, long jarg3, long jarg4); + public final static native void IndexIVFPQ_encode_multiple__SWIG_0(long jarg1, IndexIVFPQ jarg1_, long jarg2, long jarg3, LongVector jarg3_, long jarg4, long jarg5, boolean jarg6); + public final static native void IndexIVFPQ_encode_multiple__SWIG_1(long jarg1, IndexIVFPQ jarg1_, long jarg2, long jarg3, LongVector jarg3_, long jarg4, long jarg5); + public final static native void IndexIVFPQ_decode_multiple(long jarg1, IndexIVFPQ jarg1_, long jarg2, long jarg3, LongVector jarg3_, long jarg4, long jarg5); + public final static native long IndexIVFPQ_get_InvertedListScanner(long jarg1, IndexIVFPQ jarg1_, boolean jarg2); + public final static native void IndexIVFPQ_precompute_table(long jarg1, IndexIVFPQ jarg1_); + public final static native long new_IndexIVFPQ__SWIG_2(); + public final static native void delete_IndexIVFPQ(long jarg1); + public final static native void initialize_IVFPQ_precomputed_table(long jarg1, long jarg2, Index jarg2_, long jarg3, ProductQuantizer jarg3_, long jarg4, boolean jarg5); + public final static native void IndexIVFPQStats_nrefine_set(long jarg1, IndexIVFPQStats jarg1_, long jarg2); + public final static native long IndexIVFPQStats_nrefine_get(long jarg1, IndexIVFPQStats jarg1_); + public final static native void IndexIVFPQStats_n_hamming_pass_set(long jarg1, IndexIVFPQStats jarg1_, long jarg2); + public final static native long IndexIVFPQStats_n_hamming_pass_get(long jarg1, IndexIVFPQStats jarg1_); + public final static native void IndexIVFPQStats_search_cycles_set(long jarg1, IndexIVFPQStats jarg1_, long jarg2); + public final static native long IndexIVFPQStats_search_cycles_get(long jarg1, IndexIVFPQStats jarg1_); + public final static native void IndexIVFPQStats_refine_cycles_set(long jarg1, IndexIVFPQStats jarg1_, long jarg2); + public final static native long IndexIVFPQStats_refine_cycles_get(long jarg1, IndexIVFPQStats jarg1_); + public final static native long new_IndexIVFPQStats(); + public final static native void IndexIVFPQStats_reset(long jarg1, IndexIVFPQStats jarg1_); + public final static native void delete_IndexIVFPQStats(long jarg1); + public final static native void indexIVFPQ_stats_set(long jarg1, IndexIVFPQStats jarg1_); + public final static native long indexIVFPQ_stats_get(); + public final static native void IndexBinary_d_set(long jarg1, IndexBinary jarg1_, int jarg2); + public final static native int IndexBinary_d_get(long jarg1, IndexBinary jarg1_); + public final static native void IndexBinary_code_size_set(long jarg1, IndexBinary jarg1_, int jarg2); + public final static native int IndexBinary_code_size_get(long jarg1, IndexBinary jarg1_); + public final static native void IndexBinary_ntotal_set(long jarg1, IndexBinary jarg1_, long jarg2); + public final static native long IndexBinary_ntotal_get(long jarg1, IndexBinary jarg1_); + public final static native void IndexBinary_verbose_set(long jarg1, IndexBinary jarg1_, boolean jarg2); + public final static native boolean IndexBinary_verbose_get(long jarg1, IndexBinary jarg1_); + public final static native void IndexBinary_is_trained_set(long jarg1, IndexBinary jarg1_, boolean jarg2); + public final static native boolean IndexBinary_is_trained_get(long jarg1, IndexBinary jarg1_); + public final static native void IndexBinary_metric_type_set(long jarg1, IndexBinary jarg1_, int jarg2); + public final static native int IndexBinary_metric_type_get(long jarg1, IndexBinary jarg1_); + public final static native void delete_IndexBinary(long jarg1); + public final static native void IndexBinary_train(long jarg1, IndexBinary jarg1_, long jarg2, long jarg3); + public final static native void IndexBinary_add(long jarg1, IndexBinary jarg1_, long jarg2, long jarg3); + public final static native void IndexBinary_add_with_ids(long jarg1, IndexBinary jarg1_, long jarg2, long jarg3, long jarg4, LongVector jarg4_); + public final static native void IndexBinary_search(long jarg1, IndexBinary jarg1_, long jarg2, long jarg3, long jarg4, long jarg5, long jarg6, LongVector jarg6_); + public final static native void IndexBinary_range_search(long jarg1, IndexBinary jarg1_, long jarg2, long jarg3, int jarg4, long jarg5, RangeSearchResult jarg5_); + public final static native void IndexBinary_assign__SWIG_0(long jarg1, IndexBinary jarg1_, long jarg2, long jarg3, long jarg4, LongVector jarg4_, long jarg5); + public final static native void IndexBinary_assign__SWIG_1(long jarg1, IndexBinary jarg1_, long jarg2, long jarg3, long jarg4, LongVector jarg4_); + public final static native void IndexBinary_reset(long jarg1, IndexBinary jarg1_); + public final static native long IndexBinary_remove_ids(long jarg1, IndexBinary jarg1_, long jarg2, IDSelector jarg2_); + public final static native void IndexBinary_reconstruct(long jarg1, IndexBinary jarg1_, long jarg2, long jarg3); + public final static native void IndexBinary_reconstruct_n(long jarg1, IndexBinary jarg1_, long jarg2, long jarg3, long jarg4); + public final static native void IndexBinary_search_and_reconstruct(long jarg1, IndexBinary jarg1_, long jarg2, long jarg3, long jarg4, long jarg5, long jarg6, LongVector jarg6_, long jarg7); + public final static native void IndexBinary_display(long jarg1, IndexBinary jarg1_); + public final static native void Index2Layer_q1_set(long jarg1, Index2Layer jarg1_, long jarg2, Level1Quantizer jarg2_); + public final static native long Index2Layer_q1_get(long jarg1, Index2Layer jarg1_); + public final static native void Index2Layer_pq_set(long jarg1, Index2Layer jarg1_, long jarg2, ProductQuantizer jarg2_); + public final static native long Index2Layer_pq_get(long jarg1, Index2Layer jarg1_); + public final static native void Index2Layer_code_size_1_set(long jarg1, Index2Layer jarg1_, long jarg2); + public final static native long Index2Layer_code_size_1_get(long jarg1, Index2Layer jarg1_); + public final static native void Index2Layer_code_size_2_set(long jarg1, Index2Layer jarg1_, long jarg2); + public final static native long Index2Layer_code_size_2_get(long jarg1, Index2Layer jarg1_); + public final static native long new_Index2Layer__SWIG_0(long jarg1, Index jarg1_, long jarg2, int jarg3, int jarg4, int jarg5); + public final static native long new_Index2Layer__SWIG_1(long jarg1, Index jarg1_, long jarg2, int jarg3, int jarg4); + public final static native long new_Index2Layer__SWIG_2(long jarg1, Index jarg1_, long jarg2, int jarg3); + public final static native long new_Index2Layer__SWIG_3(); + public final static native void delete_Index2Layer(long jarg1); + public final static native void Index2Layer_train(long jarg1, Index2Layer jarg1_, long jarg2, long jarg3); + public final static native void Index2Layer_search(long jarg1, Index2Layer jarg1_, long jarg2, long jarg3, long jarg4, long jarg5, long jarg6, LongVector jarg6_); + public final static native long Index2Layer_get_distance_computer(long jarg1, Index2Layer jarg1_); + public final static native void Index2Layer_transfer_to_IVFPQ(long jarg1, Index2Layer jarg1_, long jarg2, IndexIVFPQ jarg2_); + public final static native void Index2Layer_sa_encode(long jarg1, Index2Layer jarg1_, long jarg2, long jarg3, long jarg4); + public final static native void Index2Layer_sa_decode(long jarg1, Index2Layer jarg1_, long jarg2, long jarg3, long jarg4); + public final static native void IndexBinaryFlat_xb_set(long jarg1, IndexBinaryFlat jarg1_, long jarg2, ByteVector jarg2_); + public final static native long IndexBinaryFlat_xb_get(long jarg1, IndexBinaryFlat jarg1_); + public final static native void IndexBinaryFlat_use_heap_set(long jarg1, IndexBinaryFlat jarg1_, boolean jarg2); + public final static native boolean IndexBinaryFlat_use_heap_get(long jarg1, IndexBinaryFlat jarg1_); + public final static native void IndexBinaryFlat_query_batch_size_set(long jarg1, IndexBinaryFlat jarg1_, long jarg2); + public final static native long IndexBinaryFlat_query_batch_size_get(long jarg1, IndexBinaryFlat jarg1_); + public final static native long new_IndexBinaryFlat__SWIG_0(long jarg1); + public final static native void IndexBinaryFlat_add(long jarg1, IndexBinaryFlat jarg1_, long jarg2, long jarg3); + public final static native void IndexBinaryFlat_reset(long jarg1, IndexBinaryFlat jarg1_); + public final static native void IndexBinaryFlat_search(long jarg1, IndexBinaryFlat jarg1_, long jarg2, long jarg3, long jarg4, long jarg5, long jarg6, LongVector jarg6_); + public final static native void IndexBinaryFlat_range_search(long jarg1, IndexBinaryFlat jarg1_, long jarg2, long jarg3, int jarg4, long jarg5, RangeSearchResult jarg5_); + public final static native void IndexBinaryFlat_reconstruct(long jarg1, IndexBinaryFlat jarg1_, long jarg2, long jarg3); + public final static native long IndexBinaryFlat_remove_ids(long jarg1, IndexBinaryFlat jarg1_, long jarg2, IDSelector jarg2_); + public final static native long new_IndexBinaryFlat__SWIG_1(); + public final static native void delete_IndexBinaryFlat(long jarg1); + public final static native void IndexBinaryIVF_invlists_set(long jarg1, IndexBinaryIVF jarg1_, long jarg2, InvertedLists jarg2_); + public final static native long IndexBinaryIVF_invlists_get(long jarg1, IndexBinaryIVF jarg1_); + public final static native void IndexBinaryIVF_own_invlists_set(long jarg1, IndexBinaryIVF jarg1_, boolean jarg2); + public final static native boolean IndexBinaryIVF_own_invlists_get(long jarg1, IndexBinaryIVF jarg1_); + public final static native void IndexBinaryIVF_nprobe_set(long jarg1, IndexBinaryIVF jarg1_, long jarg2); + public final static native long IndexBinaryIVF_nprobe_get(long jarg1, IndexBinaryIVF jarg1_); + public final static native void IndexBinaryIVF_max_codes_set(long jarg1, IndexBinaryIVF jarg1_, long jarg2); + public final static native long IndexBinaryIVF_max_codes_get(long jarg1, IndexBinaryIVF jarg1_); + public final static native void IndexBinaryIVF_use_heap_set(long jarg1, IndexBinaryIVF jarg1_, boolean jarg2); + public final static native boolean IndexBinaryIVF_use_heap_get(long jarg1, IndexBinaryIVF jarg1_); + public final static native void IndexBinaryIVF_direct_map_set(long jarg1, IndexBinaryIVF jarg1_, long jarg2); + public final static native long IndexBinaryIVF_direct_map_get(long jarg1, IndexBinaryIVF jarg1_); + public final static native void IndexBinaryIVF_quantizer_set(long jarg1, IndexBinaryIVF jarg1_, long jarg2, IndexBinary jarg2_); + public final static native long IndexBinaryIVF_quantizer_get(long jarg1, IndexBinaryIVF jarg1_); + public final static native void IndexBinaryIVF_nlist_set(long jarg1, IndexBinaryIVF jarg1_, long jarg2); + public final static native long IndexBinaryIVF_nlist_get(long jarg1, IndexBinaryIVF jarg1_); + public final static native void IndexBinaryIVF_own_fields_set(long jarg1, IndexBinaryIVF jarg1_, boolean jarg2); + public final static native boolean IndexBinaryIVF_own_fields_get(long jarg1, IndexBinaryIVF jarg1_); + public final static native void IndexBinaryIVF_cp_set(long jarg1, IndexBinaryIVF jarg1_, long jarg2, ClusteringParameters jarg2_); + public final static native long IndexBinaryIVF_cp_get(long jarg1, IndexBinaryIVF jarg1_); + public final static native void IndexBinaryIVF_clustering_index_set(long jarg1, IndexBinaryIVF jarg1_, long jarg2, Index jarg2_); + public final static native long IndexBinaryIVF_clustering_index_get(long jarg1, IndexBinaryIVF jarg1_); + public final static native long new_IndexBinaryIVF__SWIG_0(long jarg1, IndexBinary jarg1_, long jarg2, long jarg3); + public final static native long new_IndexBinaryIVF__SWIG_1(); + public final static native void delete_IndexBinaryIVF(long jarg1); + public final static native void IndexBinaryIVF_reset(long jarg1, IndexBinaryIVF jarg1_); + public final static native void IndexBinaryIVF_train(long jarg1, IndexBinaryIVF jarg1_, long jarg2, long jarg3); + public final static native void IndexBinaryIVF_add(long jarg1, IndexBinaryIVF jarg1_, long jarg2, long jarg3); + public final static native void IndexBinaryIVF_add_with_ids(long jarg1, IndexBinaryIVF jarg1_, long jarg2, long jarg3, long jarg4, LongVector jarg4_); + public final static native void IndexBinaryIVF_add_core(long jarg1, IndexBinaryIVF jarg1_, long jarg2, long jarg3, long jarg4, LongVector jarg4_, long jarg5, LongVector jarg5_); + public final static native void IndexBinaryIVF_search_preassigned__SWIG_0(long jarg1, IndexBinaryIVF jarg1_, long jarg2, long jarg3, long jarg4, long jarg5, LongVector jarg5_, long jarg6, long jarg7, long jarg8, LongVector jarg8_, boolean jarg9, long jarg10, IVFSearchParameters jarg10_); + public final static native void IndexBinaryIVF_search_preassigned__SWIG_1(long jarg1, IndexBinaryIVF jarg1_, long jarg2, long jarg3, long jarg4, long jarg5, LongVector jarg5_, long jarg6, long jarg7, long jarg8, LongVector jarg8_, boolean jarg9); + public final static native long IndexBinaryIVF_get_InvertedListScanner__SWIG_0(long jarg1, IndexBinaryIVF jarg1_, boolean jarg2); + public final static native long IndexBinaryIVF_get_InvertedListScanner__SWIG_1(long jarg1, IndexBinaryIVF jarg1_); + public final static native void IndexBinaryIVF_search(long jarg1, IndexBinaryIVF jarg1_, long jarg2, long jarg3, long jarg4, long jarg5, long jarg6, LongVector jarg6_); + public final static native void IndexBinaryIVF_range_search(long jarg1, IndexBinaryIVF jarg1_, long jarg2, long jarg3, int jarg4, long jarg5, RangeSearchResult jarg5_); + public final static native void IndexBinaryIVF_range_search_preassigned(long jarg1, IndexBinaryIVF jarg1_, long jarg2, long jarg3, int jarg4, long jarg5, LongVector jarg5_, long jarg6, long jarg7, RangeSearchResult jarg7_); + public final static native void IndexBinaryIVF_reconstruct(long jarg1, IndexBinaryIVF jarg1_, long jarg2, long jarg3); + public final static native void IndexBinaryIVF_reconstruct_n(long jarg1, IndexBinaryIVF jarg1_, long jarg2, long jarg3, long jarg4); + public final static native void IndexBinaryIVF_search_and_reconstruct(long jarg1, IndexBinaryIVF jarg1_, long jarg2, long jarg3, long jarg4, long jarg5, long jarg6, LongVector jarg6_, long jarg7); + public final static native void IndexBinaryIVF_reconstruct_from_offset(long jarg1, IndexBinaryIVF jarg1_, long jarg2, long jarg3, long jarg4); + public final static native long IndexBinaryIVF_remove_ids(long jarg1, IndexBinaryIVF jarg1_, long jarg2, IDSelector jarg2_); + public final static native void IndexBinaryIVF_merge_from(long jarg1, IndexBinaryIVF jarg1_, long jarg2, IndexBinaryIVF jarg2_, long jarg3); + public final static native long IndexBinaryIVF_get_list_size(long jarg1, IndexBinaryIVF jarg1_, long jarg2); + public final static native void IndexBinaryIVF_make_direct_map__SWIG_0(long jarg1, IndexBinaryIVF jarg1_, boolean jarg2); + public final static native void IndexBinaryIVF_make_direct_map__SWIG_1(long jarg1, IndexBinaryIVF jarg1_); + public final static native void IndexBinaryIVF_set_direct_map_type(long jarg1, IndexBinaryIVF jarg1_, long jarg2); + public final static native void IndexBinaryIVF_replace_invlists__SWIG_0(long jarg1, IndexBinaryIVF jarg1_, long jarg2, InvertedLists jarg2_, boolean jarg3); + public final static native void IndexBinaryIVF_replace_invlists__SWIG_1(long jarg1, IndexBinaryIVF jarg1_, long jarg2, InvertedLists jarg2_); + public final static native void IndexBinaryFromFloat_index_set(long jarg1, IndexBinaryFromFloat jarg1_, long jarg2, Index jarg2_); + public final static native long IndexBinaryFromFloat_index_get(long jarg1, IndexBinaryFromFloat jarg1_); + public final static native void IndexBinaryFromFloat_own_fields_set(long jarg1, IndexBinaryFromFloat jarg1_, boolean jarg2); + public final static native boolean IndexBinaryFromFloat_own_fields_get(long jarg1, IndexBinaryFromFloat jarg1_); + public final static native long new_IndexBinaryFromFloat__SWIG_0(); + public final static native long new_IndexBinaryFromFloat__SWIG_1(long jarg1, Index jarg1_); + public final static native void delete_IndexBinaryFromFloat(long jarg1); + public final static native void IndexBinaryFromFloat_add(long jarg1, IndexBinaryFromFloat jarg1_, long jarg2, long jarg3); + public final static native void IndexBinaryFromFloat_reset(long jarg1, IndexBinaryFromFloat jarg1_); + public final static native void IndexBinaryFromFloat_search(long jarg1, IndexBinaryFromFloat jarg1_, long jarg2, long jarg3, long jarg4, long jarg5, long jarg6, LongVector jarg6_); + public final static native void IndexBinaryFromFloat_train(long jarg1, IndexBinaryFromFloat jarg1_, long jarg2, long jarg3); + public final static native void IndexBinaryHNSW_hnsw_set(long jarg1, IndexBinaryHNSW jarg1_, long jarg2, HNSW jarg2_); + public final static native long IndexBinaryHNSW_hnsw_get(long jarg1, IndexBinaryHNSW jarg1_); + public final static native void IndexBinaryHNSW_own_fields_set(long jarg1, IndexBinaryHNSW jarg1_, boolean jarg2); + public final static native boolean IndexBinaryHNSW_own_fields_get(long jarg1, IndexBinaryHNSW jarg1_); + public final static native void IndexBinaryHNSW_storage_set(long jarg1, IndexBinaryHNSW jarg1_, long jarg2, IndexBinary jarg2_); + public final static native long IndexBinaryHNSW_storage_get(long jarg1, IndexBinaryHNSW jarg1_); + public final static native long new_IndexBinaryHNSW__SWIG_0(); + public final static native long new_IndexBinaryHNSW__SWIG_1(int jarg1, int jarg2); + public final static native long new_IndexBinaryHNSW__SWIG_2(int jarg1); + public final static native long new_IndexBinaryHNSW__SWIG_3(long jarg1, IndexBinary jarg1_, int jarg2); + public final static native long new_IndexBinaryHNSW__SWIG_4(long jarg1, IndexBinary jarg1_); + public final static native void delete_IndexBinaryHNSW(long jarg1); + public final static native long IndexBinaryHNSW_get_distance_computer(long jarg1, IndexBinaryHNSW jarg1_); + public final static native void IndexBinaryHNSW_add(long jarg1, IndexBinaryHNSW jarg1_, long jarg2, long jarg3); + public final static native void IndexBinaryHNSW_train(long jarg1, IndexBinaryHNSW jarg1_, long jarg2, long jarg3); + public final static native void IndexBinaryHNSW_search(long jarg1, IndexBinaryHNSW jarg1_, long jarg2, long jarg3, long jarg4, long jarg5, long jarg6, LongVector jarg6_); + public final static native void IndexBinaryHNSW_reconstruct(long jarg1, IndexBinaryHNSW jarg1_, long jarg2, long jarg3); + public final static native void IndexBinaryHNSW_reset(long jarg1, IndexBinaryHNSW jarg1_); + public final static native void IndexRefine_base_index_set(long jarg1, IndexRefine jarg1_, long jarg2, Index jarg2_); + public final static native long IndexRefine_base_index_get(long jarg1, IndexRefine jarg1_); + public final static native void IndexRefine_refine_index_set(long jarg1, IndexRefine jarg1_, long jarg2, Index jarg2_); + public final static native long IndexRefine_refine_index_get(long jarg1, IndexRefine jarg1_); + public final static native void IndexRefine_own_fields_set(long jarg1, IndexRefine jarg1_, boolean jarg2); + public final static native boolean IndexRefine_own_fields_get(long jarg1, IndexRefine jarg1_); + public final static native void IndexRefine_own_refine_index_set(long jarg1, IndexRefine jarg1_, boolean jarg2); + public final static native boolean IndexRefine_own_refine_index_get(long jarg1, IndexRefine jarg1_); + public final static native void IndexRefine_k_factor_set(long jarg1, IndexRefine jarg1_, float jarg2); + public final static native float IndexRefine_k_factor_get(long jarg1, IndexRefine jarg1_); + public final static native long new_IndexRefine__SWIG_0(long jarg1, Index jarg1_, long jarg2, Index jarg2_); + public final static native long new_IndexRefine__SWIG_1(); + public final static native void IndexRefine_train(long jarg1, IndexRefine jarg1_, long jarg2, long jarg3); + public final static native void IndexRefine_add(long jarg1, IndexRefine jarg1_, long jarg2, long jarg3); + public final static native void IndexRefine_reset(long jarg1, IndexRefine jarg1_); + public final static native void IndexRefine_search(long jarg1, IndexRefine jarg1_, long jarg2, long jarg3, long jarg4, long jarg5, long jarg6, LongVector jarg6_); + public final static native void IndexRefine_reconstruct(long jarg1, IndexRefine jarg1_, long jarg2, long jarg3); + public final static native long IndexRefine_sa_code_size(long jarg1, IndexRefine jarg1_); + public final static native void IndexRefine_sa_encode(long jarg1, IndexRefine jarg1_, long jarg2, long jarg3, long jarg4); + public final static native void IndexRefine_sa_decode(long jarg1, IndexRefine jarg1_, long jarg2, long jarg3, long jarg4); + public final static native void delete_IndexRefine(long jarg1); + public final static native long new_IndexRefineFlat__SWIG_0(long jarg1, Index jarg1_); + public final static native long new_IndexRefineFlat__SWIG_1(long jarg1, Index jarg1_, long jarg2); + public final static native long new_IndexRefineFlat__SWIG_2(); + public final static native void IndexRefineFlat_search(long jarg1, IndexRefineFlat jarg1_, long jarg2, long jarg3, long jarg4, long jarg5, long jarg6, LongVector jarg6_); + public final static native void delete_IndexRefineFlat(long jarg1); + public final static native void IndexSplitVectors_own_fields_set(long jarg1, IndexSplitVectors jarg1_, boolean jarg2); + public final static native boolean IndexSplitVectors_own_fields_get(long jarg1, IndexSplitVectors jarg1_); + public final static native void IndexSplitVectors_threaded_set(long jarg1, IndexSplitVectors jarg1_, boolean jarg2); + public final static native boolean IndexSplitVectors_threaded_get(long jarg1, IndexSplitVectors jarg1_); + public final static native void IndexSplitVectors_sub_indexes_set(long jarg1, IndexSplitVectors jarg1_, long jarg2); + public final static native long IndexSplitVectors_sub_indexes_get(long jarg1, IndexSplitVectors jarg1_); + public final static native void IndexSplitVectors_sum_d_set(long jarg1, IndexSplitVectors jarg1_, long jarg2); + public final static native long IndexSplitVectors_sum_d_get(long jarg1, IndexSplitVectors jarg1_); + public final static native void IndexSplitVectors_add_sub_index(long jarg1, IndexSplitVectors jarg1_, long jarg2, Index jarg2_); + public final static native void IndexSplitVectors_sync_with_sub_indexes(long jarg1, IndexSplitVectors jarg1_); + public final static native void IndexSplitVectors_add(long jarg1, IndexSplitVectors jarg1_, long jarg2, long jarg3); + public final static native void IndexSplitVectors_search(long jarg1, IndexSplitVectors jarg1_, long jarg2, long jarg3, long jarg4, long jarg5, long jarg6, LongVector jarg6_); + public final static native void IndexSplitVectors_train(long jarg1, IndexSplitVectors jarg1_, long jarg2, long jarg3); + public final static native void IndexSplitVectors_reset(long jarg1, IndexSplitVectors jarg1_); + public final static native void delete_IndexSplitVectors(long jarg1); + public final static native void IndexIDMap_index_set(long jarg1, IndexIDMap jarg1_, long jarg2, Index jarg2_); + public final static native long IndexIDMap_index_get(long jarg1, IndexIDMap jarg1_); + public final static native void IndexIDMap_own_fields_set(long jarg1, IndexIDMap jarg1_, boolean jarg2); + public final static native boolean IndexIDMap_own_fields_get(long jarg1, IndexIDMap jarg1_); + public final static native void IndexIDMap_id_map_set(long jarg1, IndexIDMap jarg1_, long jarg2); + public final static native long IndexIDMap_id_map_get(long jarg1, IndexIDMap jarg1_); + public final static native long new_IndexIDMap__SWIG_0(long jarg1, Index jarg1_); + public final static native void IndexIDMap_add_with_ids(long jarg1, IndexIDMap jarg1_, long jarg2, long jarg3, long jarg4, LongVector jarg4_); + public final static native void IndexIDMap_add(long jarg1, IndexIDMap jarg1_, long jarg2, long jarg3); + public final static native void IndexIDMap_search(long jarg1, IndexIDMap jarg1_, long jarg2, long jarg3, long jarg4, long jarg5, long jarg6, LongVector jarg6_); + public final static native void IndexIDMap_train(long jarg1, IndexIDMap jarg1_, long jarg2, long jarg3); + public final static native void IndexIDMap_reset(long jarg1, IndexIDMap jarg1_); + public final static native long IndexIDMap_remove_ids(long jarg1, IndexIDMap jarg1_, long jarg2, IDSelector jarg2_); + public final static native void IndexIDMap_range_search(long jarg1, IndexIDMap jarg1_, long jarg2, long jarg3, float jarg4, long jarg5, RangeSearchResult jarg5_); + public final static native void delete_IndexIDMap(long jarg1); + public final static native long new_IndexIDMap__SWIG_1(); + public final static native long new_IndexShards__SWIG_0(boolean jarg1, boolean jarg2); + public final static native long new_IndexShards__SWIG_1(boolean jarg1); + public final static native long new_IndexShards__SWIG_2(); + public final static native long new_IndexShards__SWIG_3(int jarg1, boolean jarg2, boolean jarg3); + public final static native long new_IndexShards__SWIG_4(int jarg1, boolean jarg2); + public final static native long new_IndexShards__SWIG_5(int jarg1); + public final static native void IndexShards_add_shard(long jarg1, IndexShards jarg1_, long jarg2, Index jarg2_); + public final static native void IndexShards_remove_shard(long jarg1, IndexShards jarg1_, long jarg2, Index jarg2_); + public final static native void IndexShards_add(long jarg1, IndexShards jarg1_, long jarg2, long jarg3); + public final static native void IndexShards_add_with_ids(long jarg1, IndexShards jarg1_, long jarg2, long jarg3, long jarg4, LongVector jarg4_); + public final static native void IndexShards_search(long jarg1, IndexShards jarg1_, long jarg2, long jarg3, long jarg4, long jarg5, long jarg6, LongVector jarg6_); + public final static native void IndexShards_train(long jarg1, IndexShards jarg1_, long jarg2, long jarg3); + public final static native void IndexShards_successive_ids_set(long jarg1, IndexShards jarg1_, boolean jarg2); + public final static native boolean IndexShards_successive_ids_get(long jarg1, IndexShards jarg1_); + public final static native void IndexShards_syncWithSubIndexes(long jarg1, IndexShards jarg1_); + public final static native void delete_IndexShards(long jarg1); + public final static native long downcast_index(long jarg1, Index jarg1_); + public final static native long downcast_VectorTransform(long jarg1, VectorTransform jarg1_); + public final static native long downcast_IndexBinary(long jarg1, IndexBinary jarg1_); + public final static native long upcast_IndexShards(long jarg1, IndexShards jarg1_); + public final static native void write_index__SWIG_0(long jarg1, Index jarg1_, String jarg2); + public final static native void write_index__SWIG_1(long jarg1, Index jarg1_, long jarg2); + public final static native void write_index__SWIG_2(long jarg1, Index jarg1_, long jarg2); + public final static native void write_index_binary__SWIG_0(long jarg1, IndexBinary jarg1_, String jarg2); + public final static native void write_index_binary__SWIG_1(long jarg1, IndexBinary jarg1_, long jarg2); + public final static native void write_index_binary__SWIG_2(long jarg1, IndexBinary jarg1_, long jarg2); + public final static native int IO_FLAG_READ_ONLY_get(); + public final static native int IO_FLAG_ONDISK_SAME_DIR_get(); + public final static native int IO_FLAG_SKIP_IVF_DATA_get(); + public final static native int IO_FLAG_MMAP_get(); + public final static native long read_index__SWIG_0(String jarg1, int jarg2); + public final static native long read_index__SWIG_1(String jarg1); + public final static native long read_index__SWIG_2(long jarg1, int jarg2); + public final static native long read_index__SWIG_3(long jarg1); + public final static native long read_index__SWIG_4(long jarg1, int jarg2); + public final static native long read_index__SWIG_5(long jarg1); + public final static native long read_index_binary__SWIG_0(String jarg1, int jarg2); + public final static native long read_index_binary__SWIG_1(String jarg1); + public final static native long read_index_binary__SWIG_2(long jarg1, int jarg2); + public final static native long read_index_binary__SWIG_3(long jarg1); + public final static native long read_index_binary__SWIG_4(long jarg1, int jarg2); + public final static native long read_index_binary__SWIG_5(long jarg1); + public final static native void write_VectorTransform(long jarg1, VectorTransform jarg1_, String jarg2); + public final static native long read_VectorTransform(String jarg1); + public final static native long read_ProductQuantizer__SWIG_0(String jarg1); + public final static native long read_ProductQuantizer__SWIG_1(long jarg1); + public final static native void write_ProductQuantizer__SWIG_0(long jarg1, ProductQuantizer jarg1_, String jarg2); + public final static native void write_ProductQuantizer__SWIG_1(long jarg1, ProductQuantizer jarg1_, long jarg2); + public final static native void write_InvertedLists(long jarg1, InvertedLists jarg1_, long jarg2); + public final static native long read_InvertedLists__SWIG_0(long jarg1, int jarg2); + public final static native long read_InvertedLists__SWIG_1(long jarg1); + public final static native void AutoTuneCriterion_nq_set(long jarg1, AutoTuneCriterion jarg1_, long jarg2); + public final static native long AutoTuneCriterion_nq_get(long jarg1, AutoTuneCriterion jarg1_); + public final static native void AutoTuneCriterion_nnn_set(long jarg1, AutoTuneCriterion jarg1_, long jarg2); + public final static native long AutoTuneCriterion_nnn_get(long jarg1, AutoTuneCriterion jarg1_); + public final static native void AutoTuneCriterion_gt_nnn_set(long jarg1, AutoTuneCriterion jarg1_, long jarg2); + public final static native long AutoTuneCriterion_gt_nnn_get(long jarg1, AutoTuneCriterion jarg1_); + public final static native void AutoTuneCriterion_gt_D_set(long jarg1, AutoTuneCriterion jarg1_, long jarg2, FloatVector jarg2_); + public final static native long AutoTuneCriterion_gt_D_get(long jarg1, AutoTuneCriterion jarg1_); + public final static native void AutoTuneCriterion_gt_I_set(long jarg1, AutoTuneCriterion jarg1_, long jarg2); + public final static native long AutoTuneCriterion_gt_I_get(long jarg1, AutoTuneCriterion jarg1_); + public final static native void AutoTuneCriterion_set_groundtruth(long jarg1, AutoTuneCriterion jarg1_, int jarg2, long jarg3, long jarg4, LongVector jarg4_); + public final static native double AutoTuneCriterion_evaluate(long jarg1, AutoTuneCriterion jarg1_, long jarg2, long jarg3, LongVector jarg3_); + public final static native void delete_AutoTuneCriterion(long jarg1); + public final static native void OneRecallAtRCriterion_R_set(long jarg1, OneRecallAtRCriterion jarg1_, long jarg2); + public final static native long OneRecallAtRCriterion_R_get(long jarg1, OneRecallAtRCriterion jarg1_); + public final static native long new_OneRecallAtRCriterion(long jarg1, long jarg2); + public final static native double OneRecallAtRCriterion_evaluate(long jarg1, OneRecallAtRCriterion jarg1_, long jarg2, long jarg3, LongVector jarg3_); + public final static native void delete_OneRecallAtRCriterion(long jarg1); + public final static native void IntersectionCriterion_R_set(long jarg1, IntersectionCriterion jarg1_, long jarg2); + public final static native long IntersectionCriterion_R_get(long jarg1, IntersectionCriterion jarg1_); + public final static native long new_IntersectionCriterion(long jarg1, long jarg2); + public final static native double IntersectionCriterion_evaluate(long jarg1, IntersectionCriterion jarg1_, long jarg2, long jarg3, LongVector jarg3_); + public final static native void delete_IntersectionCriterion(long jarg1); + public final static native void OperatingPoint_perf_set(long jarg1, OperatingPoint jarg1_, double jarg2); + public final static native double OperatingPoint_perf_get(long jarg1, OperatingPoint jarg1_); + public final static native void OperatingPoint_t_set(long jarg1, OperatingPoint jarg1_, double jarg2); + public final static native double OperatingPoint_t_get(long jarg1, OperatingPoint jarg1_); + public final static native void OperatingPoint_key_set(long jarg1, OperatingPoint jarg1_, String jarg2); + public final static native String OperatingPoint_key_get(long jarg1, OperatingPoint jarg1_); + public final static native void OperatingPoint_cno_set(long jarg1, OperatingPoint jarg1_, long jarg2); + public final static native long OperatingPoint_cno_get(long jarg1, OperatingPoint jarg1_); + public final static native long new_OperatingPoint(); + public final static native void delete_OperatingPoint(long jarg1); + public final static native void OperatingPoints_all_pts_set(long jarg1, OperatingPoints jarg1_, long jarg2, OperatingPointVector jarg2_); + public final static native long OperatingPoints_all_pts_get(long jarg1, OperatingPoints jarg1_); + public final static native void OperatingPoints_optimal_pts_set(long jarg1, OperatingPoints jarg1_, long jarg2, OperatingPointVector jarg2_); + public final static native long OperatingPoints_optimal_pts_get(long jarg1, OperatingPoints jarg1_); + public final static native long new_OperatingPoints(); + public final static native int OperatingPoints_merge_with__SWIG_0(long jarg1, OperatingPoints jarg1_, long jarg2, OperatingPoints jarg2_, String jarg3); + public final static native int OperatingPoints_merge_with__SWIG_1(long jarg1, OperatingPoints jarg1_, long jarg2, OperatingPoints jarg2_); + public final static native void OperatingPoints_clear(long jarg1, OperatingPoints jarg1_); + public final static native boolean OperatingPoints_add__SWIG_0(long jarg1, OperatingPoints jarg1_, double jarg2, double jarg3, String jarg4, long jarg5); + public final static native boolean OperatingPoints_add__SWIG_1(long jarg1, OperatingPoints jarg1_, double jarg2, double jarg3, String jarg4); + public final static native double OperatingPoints_t_for_perf(long jarg1, OperatingPoints jarg1_, double jarg2); + public final static native void OperatingPoints_display__SWIG_0(long jarg1, OperatingPoints jarg1_, boolean jarg2); + public final static native void OperatingPoints_display__SWIG_1(long jarg1, OperatingPoints jarg1_); + public final static native void OperatingPoints_all_to_gnuplot(long jarg1, OperatingPoints jarg1_, String jarg2); + public final static native void OperatingPoints_optimal_to_gnuplot(long jarg1, OperatingPoints jarg1_, String jarg2); + public final static native void delete_OperatingPoints(long jarg1); + public final static native void ParameterRange_name_set(long jarg1, ParameterRange jarg1_, String jarg2); + public final static native String ParameterRange_name_get(long jarg1, ParameterRange jarg1_); + public final static native void ParameterRange_values_set(long jarg1, ParameterRange jarg1_, long jarg2, DoubleVector jarg2_); + public final static native long ParameterRange_values_get(long jarg1, ParameterRange jarg1_); + public final static native long new_ParameterRange(); + public final static native void delete_ParameterRange(long jarg1); + public final static native void ParameterSpace_parameter_ranges_set(long jarg1, ParameterSpace jarg1_, long jarg2); + public final static native long ParameterSpace_parameter_ranges_get(long jarg1, ParameterSpace jarg1_); + public final static native void ParameterSpace_verbose_set(long jarg1, ParameterSpace jarg1_, int jarg2); + public final static native int ParameterSpace_verbose_get(long jarg1, ParameterSpace jarg1_); + public final static native void ParameterSpace_n_experiments_set(long jarg1, ParameterSpace jarg1_, int jarg2); + public final static native int ParameterSpace_n_experiments_get(long jarg1, ParameterSpace jarg1_); + public final static native void ParameterSpace_batchsize_set(long jarg1, ParameterSpace jarg1_, long jarg2); + public final static native long ParameterSpace_batchsize_get(long jarg1, ParameterSpace jarg1_); + public final static native void ParameterSpace_thread_over_batches_set(long jarg1, ParameterSpace jarg1_, boolean jarg2); + public final static native boolean ParameterSpace_thread_over_batches_get(long jarg1, ParameterSpace jarg1_); + public final static native void ParameterSpace_min_test_duration_set(long jarg1, ParameterSpace jarg1_, double jarg2); + public final static native double ParameterSpace_min_test_duration_get(long jarg1, ParameterSpace jarg1_); + public final static native long new_ParameterSpace(); + public final static native long ParameterSpace_n_combinations(long jarg1, ParameterSpace jarg1_); + public final static native boolean ParameterSpace_combination_ge(long jarg1, ParameterSpace jarg1_, long jarg2, long jarg3); + public final static native String ParameterSpace_combination_name(long jarg1, ParameterSpace jarg1_, long jarg2); + public final static native void ParameterSpace_display(long jarg1, ParameterSpace jarg1_); + public final static native long ParameterSpace_add_range(long jarg1, ParameterSpace jarg1_, String jarg2); + public final static native void ParameterSpace_initialize(long jarg1, ParameterSpace jarg1_, long jarg2, Index jarg2_); + public final static native void ParameterSpace_set_index_parameters__SWIG_0(long jarg1, ParameterSpace jarg1_, long jarg2, Index jarg2_, long jarg3); + public final static native void ParameterSpace_set_index_parameters__SWIG_1(long jarg1, ParameterSpace jarg1_, long jarg2, Index jarg2_, String jarg3); + public final static native void ParameterSpace_set_index_parameter(long jarg1, ParameterSpace jarg1_, long jarg2, Index jarg2_, String jarg3, double jarg4); + public final static native void ParameterSpace_update_bounds(long jarg1, ParameterSpace jarg1_, long jarg2, long jarg3, OperatingPoint jarg3_, long jarg4, long jarg5); + public final static native void ParameterSpace_explore(long jarg1, ParameterSpace jarg1_, long jarg2, Index jarg2_, long jarg3, long jarg4, long jarg5, AutoTuneCriterion jarg5_, long jarg6, OperatingPoints jarg6_); + public final static native void delete_ParameterSpace(long jarg1); + public final static native long index_factory__SWIG_0(int jarg1, String jarg2, int jarg3); + public final static native long index_factory__SWIG_1(int jarg1, String jarg2); + public final static native void index_factory_verbose_set(int jarg1); + public final static native int index_factory_verbose_get(); + public final static native long index_binary_factory(int jarg1, String jarg2); + public final static native void simd_histogram_8(long jarg1, int jarg2, long jarg3, int jarg4, long jarg5); + public final static native void simd_histogram_16(long jarg1, int jarg2, long jarg3, int jarg4, long jarg5); + public final static native void PartitionStats_bissect_cycles_set(long jarg1, PartitionStats jarg1_, long jarg2); + public final static native long PartitionStats_bissect_cycles_get(long jarg1, PartitionStats jarg1_); + public final static native void PartitionStats_compress_cycles_set(long jarg1, PartitionStats jarg1_, long jarg2); + public final static native long PartitionStats_compress_cycles_get(long jarg1, PartitionStats jarg1_); + public final static native long new_PartitionStats(); + public final static native void PartitionStats_reset(long jarg1, PartitionStats jarg1_); + public final static native void delete_PartitionStats(long jarg1); + public final static native void partition_stats_set(long jarg1, PartitionStats jarg1_); + public final static native long partition_stats_get(); + public final static native void float_minheap_array_t_nh_set(long jarg1, float_minheap_array_t jarg1_, long jarg2); + public final static native long float_minheap_array_t_nh_get(long jarg1, float_minheap_array_t jarg1_); + public final static native void float_minheap_array_t_k_set(long jarg1, float_minheap_array_t jarg1_, long jarg2); + public final static native long float_minheap_array_t_k_get(long jarg1, float_minheap_array_t jarg1_); + public final static native void float_minheap_array_t_ids_set(long jarg1, float_minheap_array_t jarg1_, long jarg2, LongVector jarg2_); + public final static native long float_minheap_array_t_ids_get(long jarg1, float_minheap_array_t jarg1_); + public final static native void float_minheap_array_t_val_set(long jarg1, float_minheap_array_t jarg1_, long jarg2); + public final static native long float_minheap_array_t_val_get(long jarg1, float_minheap_array_t jarg1_); + public final static native long float_minheap_array_t_get_val(long jarg1, float_minheap_array_t jarg1_, long jarg2); + public final static native long float_minheap_array_t_get_ids(long jarg1, float_minheap_array_t jarg1_, long jarg2); + public final static native void float_minheap_array_t_heapify(long jarg1, float_minheap_array_t jarg1_); + public final static native void float_minheap_array_t_addn__SWIG_0(long jarg1, float_minheap_array_t jarg1_, long jarg2, long jarg3, long jarg4, long jarg5, long jarg6); + public final static native void float_minheap_array_t_addn__SWIG_1(long jarg1, float_minheap_array_t jarg1_, long jarg2, long jarg3, long jarg4, long jarg5); + public final static native void float_minheap_array_t_addn__SWIG_2(long jarg1, float_minheap_array_t jarg1_, long jarg2, long jarg3, long jarg4); + public final static native void float_minheap_array_t_addn__SWIG_3(long jarg1, float_minheap_array_t jarg1_, long jarg2, long jarg3); + public final static native void float_minheap_array_t_addn_with_ids__SWIG_0(long jarg1, float_minheap_array_t jarg1_, long jarg2, long jarg3, long jarg4, LongVector jarg4_, long jarg5, long jarg6, long jarg7); + public final static native void float_minheap_array_t_addn_with_ids__SWIG_1(long jarg1, float_minheap_array_t jarg1_, long jarg2, long jarg3, long jarg4, LongVector jarg4_, long jarg5, long jarg6); + public final static native void float_minheap_array_t_addn_with_ids__SWIG_2(long jarg1, float_minheap_array_t jarg1_, long jarg2, long jarg3, long jarg4, LongVector jarg4_, long jarg5); + public final static native void float_minheap_array_t_addn_with_ids__SWIG_3(long jarg1, float_minheap_array_t jarg1_, long jarg2, long jarg3, long jarg4, LongVector jarg4_); + public final static native void float_minheap_array_t_addn_with_ids__SWIG_4(long jarg1, float_minheap_array_t jarg1_, long jarg2, long jarg3); + public final static native void float_minheap_array_t_reorder(long jarg1, float_minheap_array_t jarg1_); + public final static native void float_minheap_array_t_per_line_extrema(long jarg1, float_minheap_array_t jarg1_, long jarg2, long jarg3, LongVector jarg3_); + public final static native long new_float_minheap_array_t(); + public final static native void delete_float_minheap_array_t(long jarg1); + public final static native void int_minheap_array_t_nh_set(long jarg1, int_minheap_array_t jarg1_, long jarg2); + public final static native long int_minheap_array_t_nh_get(long jarg1, int_minheap_array_t jarg1_); + public final static native void int_minheap_array_t_k_set(long jarg1, int_minheap_array_t jarg1_, long jarg2); + public final static native long int_minheap_array_t_k_get(long jarg1, int_minheap_array_t jarg1_); + public final static native void int_minheap_array_t_ids_set(long jarg1, int_minheap_array_t jarg1_, long jarg2, LongVector jarg2_); + public final static native long int_minheap_array_t_ids_get(long jarg1, int_minheap_array_t jarg1_); + public final static native void int_minheap_array_t_val_set(long jarg1, int_minheap_array_t jarg1_, long jarg2); + public final static native long int_minheap_array_t_val_get(long jarg1, int_minheap_array_t jarg1_); + public final static native long int_minheap_array_t_get_val(long jarg1, int_minheap_array_t jarg1_, long jarg2); + public final static native long int_minheap_array_t_get_ids(long jarg1, int_minheap_array_t jarg1_, long jarg2); + public final static native void int_minheap_array_t_heapify(long jarg1, int_minheap_array_t jarg1_); + public final static native void int_minheap_array_t_addn__SWIG_0(long jarg1, int_minheap_array_t jarg1_, long jarg2, long jarg3, long jarg4, long jarg5, long jarg6); + public final static native void int_minheap_array_t_addn__SWIG_1(long jarg1, int_minheap_array_t jarg1_, long jarg2, long jarg3, long jarg4, long jarg5); + public final static native void int_minheap_array_t_addn__SWIG_2(long jarg1, int_minheap_array_t jarg1_, long jarg2, long jarg3, long jarg4); + public final static native void int_minheap_array_t_addn__SWIG_3(long jarg1, int_minheap_array_t jarg1_, long jarg2, long jarg3); + public final static native void int_minheap_array_t_addn_with_ids__SWIG_0(long jarg1, int_minheap_array_t jarg1_, long jarg2, long jarg3, long jarg4, LongVector jarg4_, long jarg5, long jarg6, long jarg7); + public final static native void int_minheap_array_t_addn_with_ids__SWIG_1(long jarg1, int_minheap_array_t jarg1_, long jarg2, long jarg3, long jarg4, LongVector jarg4_, long jarg5, long jarg6); + public final static native void int_minheap_array_t_addn_with_ids__SWIG_2(long jarg1, int_minheap_array_t jarg1_, long jarg2, long jarg3, long jarg4, LongVector jarg4_, long jarg5); + public final static native void int_minheap_array_t_addn_with_ids__SWIG_3(long jarg1, int_minheap_array_t jarg1_, long jarg2, long jarg3, long jarg4, LongVector jarg4_); + public final static native void int_minheap_array_t_addn_with_ids__SWIG_4(long jarg1, int_minheap_array_t jarg1_, long jarg2, long jarg3); + public final static native void int_minheap_array_t_reorder(long jarg1, int_minheap_array_t jarg1_); + public final static native void int_minheap_array_t_per_line_extrema(long jarg1, int_minheap_array_t jarg1_, long jarg2, long jarg3, LongVector jarg3_); + public final static native long new_int_minheap_array_t(); + public final static native void delete_int_minheap_array_t(long jarg1); + public final static native void float_maxheap_array_t_nh_set(long jarg1, float_maxheap_array_t jarg1_, long jarg2); + public final static native long float_maxheap_array_t_nh_get(long jarg1, float_maxheap_array_t jarg1_); + public final static native void float_maxheap_array_t_k_set(long jarg1, float_maxheap_array_t jarg1_, long jarg2); + public final static native long float_maxheap_array_t_k_get(long jarg1, float_maxheap_array_t jarg1_); + public final static native void float_maxheap_array_t_ids_set(long jarg1, float_maxheap_array_t jarg1_, long jarg2, LongVector jarg2_); + public final static native long float_maxheap_array_t_ids_get(long jarg1, float_maxheap_array_t jarg1_); + public final static native void float_maxheap_array_t_val_set(long jarg1, float_maxheap_array_t jarg1_, long jarg2); + public final static native long float_maxheap_array_t_val_get(long jarg1, float_maxheap_array_t jarg1_); + public final static native long float_maxheap_array_t_get_val(long jarg1, float_maxheap_array_t jarg1_, long jarg2); + public final static native long float_maxheap_array_t_get_ids(long jarg1, float_maxheap_array_t jarg1_, long jarg2); + public final static native void float_maxheap_array_t_heapify(long jarg1, float_maxheap_array_t jarg1_); + public final static native void float_maxheap_array_t_addn__SWIG_0(long jarg1, float_maxheap_array_t jarg1_, long jarg2, long jarg3, long jarg4, long jarg5, long jarg6); + public final static native void float_maxheap_array_t_addn__SWIG_1(long jarg1, float_maxheap_array_t jarg1_, long jarg2, long jarg3, long jarg4, long jarg5); + public final static native void float_maxheap_array_t_addn__SWIG_2(long jarg1, float_maxheap_array_t jarg1_, long jarg2, long jarg3, long jarg4); + public final static native void float_maxheap_array_t_addn__SWIG_3(long jarg1, float_maxheap_array_t jarg1_, long jarg2, long jarg3); + public final static native void float_maxheap_array_t_addn_with_ids__SWIG_0(long jarg1, float_maxheap_array_t jarg1_, long jarg2, long jarg3, long jarg4, LongVector jarg4_, long jarg5, long jarg6, long jarg7); + public final static native void float_maxheap_array_t_addn_with_ids__SWIG_1(long jarg1, float_maxheap_array_t jarg1_, long jarg2, long jarg3, long jarg4, LongVector jarg4_, long jarg5, long jarg6); + public final static native void float_maxheap_array_t_addn_with_ids__SWIG_2(long jarg1, float_maxheap_array_t jarg1_, long jarg2, long jarg3, long jarg4, LongVector jarg4_, long jarg5); + public final static native void float_maxheap_array_t_addn_with_ids__SWIG_3(long jarg1, float_maxheap_array_t jarg1_, long jarg2, long jarg3, long jarg4, LongVector jarg4_); + public final static native void float_maxheap_array_t_addn_with_ids__SWIG_4(long jarg1, float_maxheap_array_t jarg1_, long jarg2, long jarg3); + public final static native void float_maxheap_array_t_reorder(long jarg1, float_maxheap_array_t jarg1_); + public final static native void float_maxheap_array_t_per_line_extrema(long jarg1, float_maxheap_array_t jarg1_, long jarg2, long jarg3, LongVector jarg3_); + public final static native long new_float_maxheap_array_t(); + public final static native void delete_float_maxheap_array_t(long jarg1); + public final static native void int_maxheap_array_t_nh_set(long jarg1, int_maxheap_array_t jarg1_, long jarg2); + public final static native long int_maxheap_array_t_nh_get(long jarg1, int_maxheap_array_t jarg1_); + public final static native void int_maxheap_array_t_k_set(long jarg1, int_maxheap_array_t jarg1_, long jarg2); + public final static native long int_maxheap_array_t_k_get(long jarg1, int_maxheap_array_t jarg1_); + public final static native void int_maxheap_array_t_ids_set(long jarg1, int_maxheap_array_t jarg1_, long jarg2, LongVector jarg2_); + public final static native long int_maxheap_array_t_ids_get(long jarg1, int_maxheap_array_t jarg1_); + public final static native void int_maxheap_array_t_val_set(long jarg1, int_maxheap_array_t jarg1_, long jarg2); + public final static native long int_maxheap_array_t_val_get(long jarg1, int_maxheap_array_t jarg1_); + public final static native long int_maxheap_array_t_get_val(long jarg1, int_maxheap_array_t jarg1_, long jarg2); + public final static native long int_maxheap_array_t_get_ids(long jarg1, int_maxheap_array_t jarg1_, long jarg2); + public final static native void int_maxheap_array_t_heapify(long jarg1, int_maxheap_array_t jarg1_); + public final static native void int_maxheap_array_t_addn__SWIG_0(long jarg1, int_maxheap_array_t jarg1_, long jarg2, long jarg3, long jarg4, long jarg5, long jarg6); + public final static native void int_maxheap_array_t_addn__SWIG_1(long jarg1, int_maxheap_array_t jarg1_, long jarg2, long jarg3, long jarg4, long jarg5); + public final static native void int_maxheap_array_t_addn__SWIG_2(long jarg1, int_maxheap_array_t jarg1_, long jarg2, long jarg3, long jarg4); + public final static native void int_maxheap_array_t_addn__SWIG_3(long jarg1, int_maxheap_array_t jarg1_, long jarg2, long jarg3); + public final static native void int_maxheap_array_t_addn_with_ids__SWIG_0(long jarg1, int_maxheap_array_t jarg1_, long jarg2, long jarg3, long jarg4, LongVector jarg4_, long jarg5, long jarg6, long jarg7); + public final static native void int_maxheap_array_t_addn_with_ids__SWIG_1(long jarg1, int_maxheap_array_t jarg1_, long jarg2, long jarg3, long jarg4, LongVector jarg4_, long jarg5, long jarg6); + public final static native void int_maxheap_array_t_addn_with_ids__SWIG_2(long jarg1, int_maxheap_array_t jarg1_, long jarg2, long jarg3, long jarg4, LongVector jarg4_, long jarg5); + public final static native void int_maxheap_array_t_addn_with_ids__SWIG_3(long jarg1, int_maxheap_array_t jarg1_, long jarg2, long jarg3, long jarg4, LongVector jarg4_); + public final static native void int_maxheap_array_t_addn_with_ids__SWIG_4(long jarg1, int_maxheap_array_t jarg1_, long jarg2, long jarg3); + public final static native void int_maxheap_array_t_reorder(long jarg1, int_maxheap_array_t jarg1_); + public final static native void int_maxheap_array_t_per_line_extrema(long jarg1, int_maxheap_array_t jarg1_, long jarg2, long jarg3, LongVector jarg3_); + public final static native long new_int_maxheap_array_t(); + public final static native void delete_int_maxheap_array_t(long jarg1); + public final static native float CMin_float_partition_fuzzy(long jarg1, long jarg2, LongVector jarg2_, long jarg3, long jarg4, long jarg5, long jarg6); + public final static native float CMax_float_partition_fuzzy(long jarg1, long jarg2, LongVector jarg2_, long jarg3, long jarg4, long jarg5, long jarg6); + public final static native void AlignedTableUint8_tab_set(long jarg1, AlignedTableUint8 jarg1_, long jarg2); + public final static native long AlignedTableUint8_tab_get(long jarg1, AlignedTableUint8 jarg1_); + public final static native void AlignedTableUint8_numel_set(long jarg1, AlignedTableUint8 jarg1_, long jarg2); + public final static native long AlignedTableUint8_numel_get(long jarg1, AlignedTableUint8 jarg1_); + public final static native long AlignedTableUint8_round_capacity(long jarg1); + public final static native long new_AlignedTableUint8__SWIG_0(); + public final static native long new_AlignedTableUint8__SWIG_1(long jarg1); + public final static native long AlignedTableUint8_itemsize(long jarg1, AlignedTableUint8 jarg1_); + public final static native void AlignedTableUint8_resize(long jarg1, AlignedTableUint8 jarg1_, long jarg2); + public final static native void AlignedTableUint8_clear(long jarg1, AlignedTableUint8 jarg1_); + public final static native long AlignedTableUint8_size(long jarg1, AlignedTableUint8 jarg1_); + public final static native long AlignedTableUint8_nbytes(long jarg1, AlignedTableUint8 jarg1_); + public final static native long AlignedTableUint8_get__SWIG_0(long jarg1, AlignedTableUint8 jarg1_); + public final static native long AlignedTableUint8_data__SWIG_0(long jarg1, AlignedTableUint8 jarg1_); + public final static native void delete_AlignedTableUint8(long jarg1); + public final static native void AlignedTableUint16_tab_set(long jarg1, AlignedTableUint16 jarg1_, long jarg2); + public final static native long AlignedTableUint16_tab_get(long jarg1, AlignedTableUint16 jarg1_); + public final static native void AlignedTableUint16_numel_set(long jarg1, AlignedTableUint16 jarg1_, long jarg2); + public final static native long AlignedTableUint16_numel_get(long jarg1, AlignedTableUint16 jarg1_); + public final static native long AlignedTableUint16_round_capacity(long jarg1); + public final static native long new_AlignedTableUint16__SWIG_0(); + public final static native long new_AlignedTableUint16__SWIG_1(long jarg1); + public final static native long AlignedTableUint16_itemsize(long jarg1, AlignedTableUint16 jarg1_); + public final static native void AlignedTableUint16_resize(long jarg1, AlignedTableUint16 jarg1_, long jarg2); + public final static native void AlignedTableUint16_clear(long jarg1, AlignedTableUint16 jarg1_); + public final static native long AlignedTableUint16_size(long jarg1, AlignedTableUint16 jarg1_); + public final static native long AlignedTableUint16_nbytes(long jarg1, AlignedTableUint16 jarg1_); + public final static native long AlignedTableUint16_get__SWIG_0(long jarg1, AlignedTableUint16 jarg1_); + public final static native long AlignedTableUint16_data__SWIG_0(long jarg1, AlignedTableUint16 jarg1_); + public final static native void delete_AlignedTableUint16(long jarg1); + public final static native void AlignedTableFloat32_tab_set(long jarg1, AlignedTableFloat32 jarg1_, long jarg2); + public final static native long AlignedTableFloat32_tab_get(long jarg1, AlignedTableFloat32 jarg1_); + public final static native void AlignedTableFloat32_numel_set(long jarg1, AlignedTableFloat32 jarg1_, long jarg2); + public final static native long AlignedTableFloat32_numel_get(long jarg1, AlignedTableFloat32 jarg1_); + public final static native long AlignedTableFloat32_round_capacity(long jarg1); + public final static native long new_AlignedTableFloat32__SWIG_0(); + public final static native long new_AlignedTableFloat32__SWIG_1(long jarg1); + public final static native long AlignedTableFloat32_itemsize(long jarg1, AlignedTableFloat32 jarg1_); + public final static native void AlignedTableFloat32_resize(long jarg1, AlignedTableFloat32 jarg1_, long jarg2); + public final static native void AlignedTableFloat32_clear(long jarg1, AlignedTableFloat32 jarg1_); + public final static native long AlignedTableFloat32_size(long jarg1, AlignedTableFloat32 jarg1_); + public final static native long AlignedTableFloat32_nbytes(long jarg1, AlignedTableFloat32 jarg1_); + public final static native long AlignedTableFloat32_get__SWIG_0(long jarg1, AlignedTableFloat32 jarg1_); + public final static native long AlignedTableFloat32_data__SWIG_0(long jarg1, AlignedTableFloat32 jarg1_); + public final static native void delete_AlignedTableFloat32(long jarg1); + public final static native long CMax_uint16_partition_fuzzy__SWIG_0(long jarg1, long jarg2, LongVector jarg2_, long jarg3, long jarg4, long jarg5, long jarg6); + public final static native long CMin_uint16_partition_fuzzy__SWIG_0(long jarg1, long jarg2, LongVector jarg2_, long jarg3, long jarg4, long jarg5, long jarg6); + public final static native long CMax_uint16_partition_fuzzy__SWIG_1(long jarg1, long jarg2, long jarg3, long jarg4, long jarg5, long jarg6); + public final static native long CMin_uint16_partition_fuzzy__SWIG_1(long jarg1, long jarg2, long jarg3, long jarg4, long jarg5, long jarg6); + public final static native void omp_set_num_threads(int jarg1); + public final static native int omp_get_max_threads(); + public final static native long memcpy(long jarg1, long jarg2, long jarg3); + public final static native long cast_integer_to_float_ptr(int jarg1); + public final static native long cast_integer_to_long_ptr(int jarg1); + public final static native long cast_integer_to_int_ptr(int jarg1); + public final static native void RangeSearchResult_nq_set(long jarg1, RangeSearchResult jarg1_, long jarg2); + public final static native long RangeSearchResult_nq_get(long jarg1, RangeSearchResult jarg1_); + public final static native void RangeSearchResult_lims_set(long jarg1, RangeSearchResult jarg1_, long jarg2); + public final static native long RangeSearchResult_lims_get(long jarg1, RangeSearchResult jarg1_); + public final static native void RangeSearchResult_labels_set(long jarg1, RangeSearchResult jarg1_, long jarg2, LongVector jarg2_); + public final static native long RangeSearchResult_labels_get(long jarg1, RangeSearchResult jarg1_); + public final static native void RangeSearchResult_distances_set(long jarg1, RangeSearchResult jarg1_, long jarg2); + public final static native long RangeSearchResult_distances_get(long jarg1, RangeSearchResult jarg1_); + public final static native void RangeSearchResult_buffer_size_set(long jarg1, RangeSearchResult jarg1_, long jarg2); + public final static native long RangeSearchResult_buffer_size_get(long jarg1, RangeSearchResult jarg1_); + public final static native void RangeSearchResult_do_allocation(long jarg1, RangeSearchResult jarg1_); + public final static native void delete_RangeSearchResult(long jarg1); + public final static native boolean IDSelector_is_member(long jarg1, IDSelector jarg1_, long jarg2); + public final static native void delete_IDSelector(long jarg1); + public final static native void IDSelectorRange_imin_set(long jarg1, IDSelectorRange jarg1_, long jarg2); + public final static native long IDSelectorRange_imin_get(long jarg1, IDSelectorRange jarg1_); + public final static native void IDSelectorRange_imax_set(long jarg1, IDSelectorRange jarg1_, long jarg2); + public final static native long IDSelectorRange_imax_get(long jarg1, IDSelectorRange jarg1_); + public final static native long new_IDSelectorRange(long jarg1, long jarg2); + public final static native boolean IDSelectorRange_is_member(long jarg1, IDSelectorRange jarg1_, long jarg2); + public final static native void delete_IDSelectorRange(long jarg1); + public final static native void IDSelectorArray_n_set(long jarg1, IDSelectorArray jarg1_, long jarg2); + public final static native long IDSelectorArray_n_get(long jarg1, IDSelectorArray jarg1_); + public final static native void IDSelectorArray_ids_set(long jarg1, IDSelectorArray jarg1_, long jarg2, LongVector jarg2_); + public final static native long IDSelectorArray_ids_get(long jarg1, IDSelectorArray jarg1_); + public final static native long new_IDSelectorArray(long jarg1, long jarg2, LongVector jarg2_); + public final static native boolean IDSelectorArray_is_member(long jarg1, IDSelectorArray jarg1_, long jarg2); + public final static native void delete_IDSelectorArray(long jarg1); + public final static native void IDSelectorBatch_nbits_set(long jarg1, IDSelectorBatch jarg1_, int jarg2); + public final static native int IDSelectorBatch_nbits_get(long jarg1, IDSelectorBatch jarg1_); + public final static native void IDSelectorBatch_mask_set(long jarg1, IDSelectorBatch jarg1_, long jarg2); + public final static native long IDSelectorBatch_mask_get(long jarg1, IDSelectorBatch jarg1_); + public final static native long new_IDSelectorBatch(long jarg1, long jarg2, LongVector jarg2_); + public final static native boolean IDSelectorBatch_is_member(long jarg1, IDSelectorBatch jarg1_, long jarg2); + public final static native void delete_IDSelectorBatch(long jarg1); + public final static native void BufferList_buffer_size_set(long jarg1, BufferList jarg1_, long jarg2); + public final static native long BufferList_buffer_size_get(long jarg1, BufferList jarg1_); + public final static native void BufferList_buffers_set(long jarg1, BufferList jarg1_, long jarg2); + public final static native long BufferList_buffers_get(long jarg1, BufferList jarg1_); + public final static native void BufferList_wp_set(long jarg1, BufferList jarg1_, long jarg2); + public final static native long BufferList_wp_get(long jarg1, BufferList jarg1_); + public final static native long new_BufferList(long jarg1); + public final static native void delete_BufferList(long jarg1); + public final static native void BufferList_append_buffer(long jarg1, BufferList jarg1_); + public final static native void BufferList_add(long jarg1, BufferList jarg1_, long jarg2, float jarg3); + public final static native void BufferList_copy_range(long jarg1, BufferList jarg1_, long jarg2, long jarg3, long jarg4, LongVector jarg4_, long jarg5); + public final static native void RangeQueryResult_qno_set(long jarg1, RangeQueryResult jarg1_, long jarg2); + public final static native long RangeQueryResult_qno_get(long jarg1, RangeQueryResult jarg1_); + public final static native void RangeQueryResult_nres_set(long jarg1, RangeQueryResult jarg1_, long jarg2); + public final static native long RangeQueryResult_nres_get(long jarg1, RangeQueryResult jarg1_); + public final static native void RangeQueryResult_pres_set(long jarg1, RangeQueryResult jarg1_, long jarg2, RangeSearchPartialResult jarg2_); + public final static native long RangeQueryResult_pres_get(long jarg1, RangeQueryResult jarg1_); + public final static native void RangeQueryResult_add(long jarg1, RangeQueryResult jarg1_, float jarg2, long jarg3); + public final static native long new_RangeQueryResult(); + public final static native void delete_RangeQueryResult(long jarg1); + public final static native void RangeSearchPartialResult_res_set(long jarg1, RangeSearchPartialResult jarg1_, long jarg2, RangeSearchResult jarg2_); + public final static native long RangeSearchPartialResult_res_get(long jarg1, RangeSearchPartialResult jarg1_); + public final static native void RangeSearchPartialResult_queries_set(long jarg1, RangeSearchPartialResult jarg1_, long jarg2); + public final static native long RangeSearchPartialResult_queries_get(long jarg1, RangeSearchPartialResult jarg1_); + public final static native long RangeSearchPartialResult_new_result(long jarg1, RangeSearchPartialResult jarg1_, long jarg2); + public final static native void RangeSearchPartialResult_set_lims(long jarg1, RangeSearchPartialResult jarg1_); + public final static native void RangeSearchPartialResult_copy_result__SWIG_0(long jarg1, RangeSearchPartialResult jarg1_, boolean jarg2); + public final static native void RangeSearchPartialResult_copy_result__SWIG_1(long jarg1, RangeSearchPartialResult jarg1_); + public final static native void RangeSearchPartialResult_merge__SWIG_0(long jarg1, boolean jarg2); + public final static native void RangeSearchPartialResult_merge__SWIG_1(long jarg1); + public final static native void delete_RangeSearchPartialResult(long jarg1); + public final static native void DistanceComputer_set_query(long jarg1, DistanceComputer jarg1_, long jarg2); + public final static native float DistanceComputer_symmetric_dis(long jarg1, DistanceComputer jarg1_, long jarg2, long jarg3); + public final static native void delete_DistanceComputer(long jarg1); + public final static native boolean InterruptCallback_want_interrupt(long jarg1, InterruptCallback jarg1_); + public final static native void delete_InterruptCallback(long jarg1); + public final static native void InterruptCallback_clear_instance(); + public final static native void InterruptCallback_check(); + public final static native boolean InterruptCallback_is_interrupted(); + public final static native long InterruptCallback_get_period_hint(long jarg1); + public final static native void VisitedTable_visited_set(long jarg1, VisitedTable jarg1_, long jarg2, ByteVector jarg2_); + public final static native long VisitedTable_visited_get(long jarg1, VisitedTable jarg1_); + public final static native void VisitedTable_visno_set(long jarg1, VisitedTable jarg1_, int jarg2); + public final static native int VisitedTable_visno_get(long jarg1, VisitedTable jarg1_); + public final static native long new_VisitedTable(int jarg1); + public final static native void VisitedTable_set(long jarg1, VisitedTable jarg1_, int jarg2); + public final static native boolean VisitedTable_get(long jarg1, VisitedTable jarg1_, int jarg2); + public final static native void VisitedTable_advance(long jarg1, VisitedTable jarg1_); + public final static native void delete_VisitedTable(long jarg1); + public final static native void ignore_SIGTTIN(); + public final static native void MapLong2Long_map_set(long jarg1, MapLong2Long jarg1_, long jarg2); + public final static native long MapLong2Long_map_get(long jarg1, MapLong2Long jarg1_); + public final static native void MapLong2Long_add(long jarg1, MapLong2Long jarg1_, long jarg2, long jarg3, long jarg4); + public final static native int MapLong2Long_search(long jarg1, MapLong2Long jarg1_, int jarg2); + public final static native void MapLong2Long_search_multiple(long jarg1, MapLong2Long jarg1_, long jarg2, long jarg3, long jarg4); + public final static native long new_MapLong2Long(); + public final static native void delete_MapLong2Long(long jarg1); + public final static native long Clustering_SWIGUpcast(long jarg1); + public final static native long Clustering1D_SWIGUpcast(long jarg1); + public final static native long ProgressiveDimClusteringParameters_SWIGUpcast(long jarg1); + public final static native long ProgressiveDimClustering_SWIGUpcast(long jarg1); + public final static native long LinearTransform_SWIGUpcast(long jarg1); + public final static native long RandomRotationMatrix_SWIGUpcast(long jarg1); + public final static native long PCAMatrix_SWIGUpcast(long jarg1); + public final static native long ITQMatrix_SWIGUpcast(long jarg1); + public final static native long ITQTransform_SWIGUpcast(long jarg1); + public final static native long OPQMatrix_SWIGUpcast(long jarg1); + public final static native long RemapDimensionsTransform_SWIGUpcast(long jarg1); + public final static native long NormalizationTransform_SWIGUpcast(long jarg1); + public final static native long CenteringTransform_SWIGUpcast(long jarg1); + public final static native long IndexFlatCodes_SWIGUpcast(long jarg1); + public final static native long IndexFlat_SWIGUpcast(long jarg1); + public final static native long IndexFlatIP_SWIGUpcast(long jarg1); + public final static native long IndexFlatL2_SWIGUpcast(long jarg1); + public final static native long IndexFlat1D_SWIGUpcast(long jarg1); + public final static native long IndexLSH_SWIGUpcast(long jarg1); + public final static native long ReproduceDistancesObjective_SWIGUpcast(long jarg1); + public final static native long SimulatedAnnealingOptimizer_SWIGUpcast(long jarg1); + public final static native long PolysemousTraining_SWIGUpcast(long jarg1); + public final static native long IndexPQ_SWIGUpcast(long jarg1); + public final static native long MultiIndexQuantizer_SWIGUpcast(long jarg1); + public final static native long MultiIndexQuantizer2_SWIGUpcast(long jarg1); + public final static native long ArrayInvertedLists_SWIGUpcast(long jarg1); + public final static native long ReadOnlyInvertedLists_SWIGUpcast(long jarg1); + public final static native long HStackInvertedLists_SWIGUpcast(long jarg1); + public final static native long SliceInvertedLists_SWIGUpcast(long jarg1); + public final static native long VStackInvertedLists_SWIGUpcast(long jarg1); + public final static native long MaskedInvertedLists_SWIGUpcast(long jarg1); + public final static native long StopWordsInvertedLists_SWIGUpcast(long jarg1); + public final static native long IndexIVF_SWIGUpcast(long jarg1); + public final static native long IndexScalarQuantizer_SWIGUpcast(long jarg1); + public final static native long IndexIVFScalarQuantizer_SWIGUpcast(long jarg1); + public final static native long IndexHNSW_SWIGUpcast(long jarg1); + public final static native long IndexHNSWFlat_SWIGUpcast(long jarg1); + public final static native long IndexHNSWPQ_SWIGUpcast(long jarg1); + public final static native long IndexHNSWSQ_SWIGUpcast(long jarg1); + public final static native long IndexHNSW2Level_SWIGUpcast(long jarg1); + public final static native long IndexIVFFlat_SWIGUpcast(long jarg1); + public final static native long IndexIVFFlatDedup_SWIGUpcast(long jarg1); + public final static native long OnDiskInvertedLists_SWIGUpcast(long jarg1); + public final static native long IVFPQSearchParameters_SWIGUpcast(long jarg1); + public final static native long IndexIVFPQ_SWIGUpcast(long jarg1); + public final static native long Index2Layer_SWIGUpcast(long jarg1); + public final static native long IndexBinaryFlat_SWIGUpcast(long jarg1); + public final static native long IndexBinaryIVF_SWIGUpcast(long jarg1); + public final static native long IndexBinaryFromFloat_SWIGUpcast(long jarg1); + public final static native long IndexBinaryHNSW_SWIGUpcast(long jarg1); + public final static native long IndexRefine_SWIGUpcast(long jarg1); + public final static native long IndexRefineFlat_SWIGUpcast(long jarg1); + public final static native long IndexSplitVectors_SWIGUpcast(long jarg1); + public final static native long IndexIDMap_SWIGUpcast(long jarg1); + public final static native long OneRecallAtRCriterion_SWIGUpcast(long jarg1); + public final static native long IntersectionCriterion_SWIGUpcast(long jarg1); + public final static native long IDSelectorRange_SWIGUpcast(long jarg1); + public final static native long IDSelectorArray_SWIGUpcast(long jarg1); + public final static native long IDSelectorBatch_SWIGUpcast(long jarg1); + public final static native long RangeSearchPartialResult_SWIGUpcast(long jarg1); +} diff --git a/ann/src/main/java/com/twitter/ann/hnsw/BUILD b/ann/src/main/java/com/twitter/ann/hnsw/BUILD new file mode 100644 index 0000000000..b7534c6e70 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/hnsw/BUILD @@ -0,0 +1,18 @@ +java_library( + sources = ["*.java"], + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/com/google/guava", + "3rdparty/jvm/com/google/inject:guice", + "3rdparty/jvm/com/twitter/bijection:core", + "3rdparty/jvm/commons-lang", + "3rdparty/jvm/org/apache/thrift", + "ann/src/main/scala/com/twitter/ann/common", + "ann/src/main/thrift/com/twitter/ann/common:ann-common-java", + "mediaservices/commons/src/main/scala:futuretracker", + "scrooge/scrooge-core", + "src/java/com/twitter/search/common/file", + ], +) diff --git a/ann/src/main/java/com/twitter/ann/hnsw/DistanceFunction.java b/ann/src/main/java/com/twitter/ann/hnsw/DistanceFunction.java new file mode 100644 index 0000000000..a7adf126fc --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/hnsw/DistanceFunction.java @@ -0,0 +1,8 @@ +package com.twitter.ann.hnsw; + +public interface DistanceFunction { + /** + * Distance between two items. + */ + float distance(T t, Q q); +} diff --git a/ann/src/main/java/com/twitter/ann/hnsw/DistancedItem.java b/ann/src/main/java/com/twitter/ann/hnsw/DistancedItem.java new file mode 100644 index 0000000000..cc3fb6a7ac --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/hnsw/DistancedItem.java @@ -0,0 +1,23 @@ +package com.twitter.ann.hnsw; + +/** + * An item associated with a float distance + * @param The type of the item. + */ +public class DistancedItem { + private final T item; + private final float distance; + + public DistancedItem(T item, float distance) { + this.item = item; + this.distance = distance; + } + + public T getItem() { + return item; + } + + public float getDistance() { + return distance; + } +} diff --git a/ann/src/main/java/com/twitter/ann/hnsw/DistancedItemQueue.java b/ann/src/main/java/com/twitter/ann/hnsw/DistancedItemQueue.java new file mode 100644 index 0000000000..f77f9c2b2b --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/hnsw/DistancedItemQueue.java @@ -0,0 +1,196 @@ +package com.twitter.ann.hnsw; + +import java.util.ArrayList; +import java.util.Comparator; +import java.util.Iterator; +import java.util.List; +import java.util.PriorityQueue; + +/** + * Container for items with their distance. + * + * @param Type of origin/reference element. + * @param Type of element that the queue will hold + */ +public class DistancedItemQueue implements Iterable> { + private final U origin; + private final DistanceFunction distFn; + private final PriorityQueue> queue; + private final boolean minQueue; + /** + * Creates ontainer for items with their distances. + * + * @param origin Origin (reference) point + * @param initial Initial list of elements to add in the structure + * @param minQueue True for min queue, False for max queue + * @param distFn Distance function + */ + public DistancedItemQueue( + U origin, + List initial, + boolean minQueue, + DistanceFunction distFn + ) { + this.origin = origin; + this.distFn = distFn; + this.minQueue = minQueue; + final Comparator> cmp; + if (minQueue) { + cmp = (o1, o2) -> Float.compare(o1.getDistance(), o2.getDistance()); + } else { + cmp = (o1, o2) -> Float.compare(o2.getDistance(), o1.getDistance()); + } + this.queue = new PriorityQueue<>(cmp); + enqueueAll(initial); + new DistancedItemQueue<>(origin, distFn, queue, minQueue); + } + + private DistancedItemQueue( + U origin, + DistanceFunction distFn, + PriorityQueue> queue, + boolean minQueue + ) { + this.origin = origin; + this.distFn = distFn; + this.queue = queue; + this.minQueue = minQueue; + } + + /** + * Enqueues all the items into the queue. + */ + public void enqueueAll(List list) { + for (T t : list) { + enqueue(t); + } + } + + /** + * Return if queue is non empty or not + * + * @return true if queue is not empty else false + */ + public boolean nonEmpty() { + return !queue.isEmpty(); + } + + /** + * Return root of the queue + * + * @return root of the queue i.e min/max element depending upon min-max queue + */ + public DistancedItem peek() { + return queue.peek(); + } + + /** + * Dequeue root of the queue. + * + * @return remove and return root of the queue i.e min/max element depending upon min-max queue + */ + public DistancedItem dequeue() { + return queue.poll(); + } + + /** + * Dequeue all the elements from queueu with ordering mantained + * + * @return remove all the elements in the order of the queue i.e min/max queue. + */ + public List> dequeueAll() { + final List> list = new ArrayList<>(queue.size()); + while (!queue.isEmpty()) { + list.add(queue.poll()); + } + + return list; + } + + /** + * Convert queue to list + * + * @return list of elements of queue with distance and without any specific ordering + */ + public List> toList() { + return new ArrayList<>(queue); + } + + /** + * Convert queue to list + * + * @return list of elements of queue without any specific ordering + */ + List toListWithItem() { + List list = new ArrayList<>(queue.size()); + Iterator> itr = iterator(); + while (itr.hasNext()) { + list.add(itr.next().getItem()); + } + return list; + } + + /** + * Enqueue an item into the queue + */ + public void enqueue(T item) { + queue.add(new DistancedItem<>(item, distFn.distance(origin, item))); + } + + /** + * Enqueue an item into the queue with its distance. + */ + public void enqueue(T item, float distance) { + queue.add(new DistancedItem<>(item, distance)); + } + + /** + * Size + * + * @return size of the queue + */ + public int size() { + return queue.size(); + } + + /** + * Is Min queue + * + * @return true if min queue else false + */ + public boolean isMinQueue() { + return minQueue; + } + + /** + * Returns origin (base element) of the queue + * + * @return origin of the queue + */ + public U getOrigin() { + return origin; + } + + /** + * Return a new queue with ordering reversed. + */ + public DistancedItemQueue reverse() { + final PriorityQueue> rqueue = + new PriorityQueue<>(queue.comparator().reversed()); + if (queue.isEmpty()) { + return new DistancedItemQueue<>(origin, distFn, rqueue, !isMinQueue()); + } + + final Iterator> itr = iterator(); + while (itr.hasNext()) { + rqueue.add(itr.next()); + } + + return new DistancedItemQueue<>(origin, distFn, rqueue, !isMinQueue()); + } + + @Override + public Iterator> iterator() { + return queue.iterator(); + } +} diff --git a/ann/src/main/java/com/twitter/ann/hnsw/HnswIndex.java b/ann/src/main/java/com/twitter/ann/hnsw/HnswIndex.java new file mode 100644 index 0000000000..2f9c91409c --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/hnsw/HnswIndex.java @@ -0,0 +1,711 @@ +package com.twitter.ann.hnsw; + +import java.io.IOException; +import java.nio.ByteBuffer; +import java.util.ArrayList; +import java.util.Collections; +import java.util.HashMap; +import java.util.HashSet; +import java.util.List; +import java.util.Map; +import java.util.Objects; +import java.util.Optional; +import java.util.Random; +import java.util.Set; +import java.util.concurrent.ConcurrentHashMap; +import java.util.concurrent.atomic.AtomicReference; +import java.util.concurrent.locks.Lock; +import java.util.concurrent.locks.ReadWriteLock; +import java.util.concurrent.locks.ReentrantLock; +import java.util.concurrent.locks.ReentrantReadWriteLock; +import java.util.function.Function; + +import com.google.common.annotations.VisibleForTesting; +import com.google.common.base.Preconditions; +import com.google.common.collect.ImmutableList; + +import org.apache.thrift.TException; + +import com.twitter.ann.common.IndexOutputFile; +import com.twitter.ann.common.thriftjava.HnswInternalIndexMetadata; +import com.twitter.bijection.Injection; +import com.twitter.logging.Logger; +import com.twitter.mediaservices.commons.codec.ArrayByteBufferCodec; +import com.twitter.search.common.file.AbstractFile; + +/** + * Typed multithreaded HNSW implementation supporting creation/querying of approximate nearest neighbour + * Paper: https://arxiv.org/pdf/1603.09320.pdf + * Multithreading impl based on NMSLIB version : https://github.com/nmslib/hnsw/blob/master/hnswlib/hnswalg.h + * + * @param The type of items inserted / searched in the HNSW index. + * @param The type of KNN query. + */ +public class HnswIndex { + private static final Logger LOG = Logger.get(HnswIndex.class); + private static final String METADATA_FILE_NAME = "hnsw_internal_metadata"; + private static final String GRAPH_FILE_NAME = "hnsw_internal_graph"; + private static final int MAP_SIZE_FACTOR = 5; + + private final DistanceFunction distFnIndex; + private final DistanceFunction distFnQuery; + private final int efConstruction; + private final int maxM; + private final int maxM0; + private final double levelMultiplier; + private final AtomicReference> graphMeta = new AtomicReference<>(); + private final Map, ImmutableList> graph; + // To take lock on vertex level + private final ConcurrentHashMap locks; + // To take lock on whole graph only if vertex addition is on layer above the current maxLevel + private final ReentrantLock globalLock; + private final Function lockProvider; + + private final RandomProvider randomProvider; + + // Probability of reevaluating connections of an element in the neighborhood during an update + // Can be used as a knob to adjust update_speed/search_speed tradeoff. + private final float updateNeighborProbability; + + /** + * Creates instance of hnsw index. + * + * @param distFnIndex Any distance metric/non metric that specifies similarity between two items for indexing. + * @param distFnQuery Any distance metric/non metric that specifies similarity between item for which nearest neighbours queried for and already indexed item. + * @param efConstruction Provide speed vs index quality tradeoff, higher the value better the quality and higher the time to create index. + * Valid range of efConstruction can be anywhere between 1 and tens of thousand. Typically, it should be set so that a search of M + * neighbors with ef=efConstruction should end in recall>0.95. + * @param maxM Maximum connections per layer except 0th level. + * Optimal values between 5-48. + * Smaller M generally produces better result for lower recalls and/ or lower dimensional data, + * while bigger M is better for high recall and/ or high dimensional, data on the expense of more memory/disk usage + * @param expectedElements Approximate number of elements to be indexed + */ + protected HnswIndex( + DistanceFunction distFnIndex, + DistanceFunction distFnQuery, + int efConstruction, + int maxM, + int expectedElements, + RandomProvider randomProvider + ) { + this(distFnIndex, + distFnQuery, + efConstruction, + maxM, + expectedElements, + new HnswMeta<>(-1, Optional.empty()), + new ConcurrentHashMap<>(MAP_SIZE_FACTOR * expectedElements), + randomProvider + ); + } + + private HnswIndex( + DistanceFunction distFnIndex, + DistanceFunction distFnQuery, + int efConstruction, + int maxM, + int expectedElements, + HnswMeta graphMeta, + Map, ImmutableList> graph, + RandomProvider randomProvider + ) { + this.distFnIndex = distFnIndex; + this.distFnQuery = distFnQuery; + this.efConstruction = efConstruction; + this.maxM = maxM; + this.maxM0 = 2 * maxM; + this.levelMultiplier = 1.0 / Math.log(1.0 * maxM); + this.graphMeta.set(graphMeta); + this.graph = graph; + this.locks = new ConcurrentHashMap<>(MAP_SIZE_FACTOR * expectedElements); + this.globalLock = new ReentrantLock(); + this.lockProvider = key -> new ReentrantReadWriteLock(); + this.randomProvider = randomProvider; + this.updateNeighborProbability = 1.0f; + } + + /** + * wireConnectionForAllLayers finds connections for a new element and creates bi-direction links. + * The method assumes using a reentrant lock to link list reads. + * + * @param entryPoint the global entry point + * @param item the item for which the connections are found + * @param itemLevel the level of the added item (maximum layer in which we wire the connections) + * @param maxLayer the level of the entry point + */ + private void wireConnectionForAllLayers(final T entryPoint, final T item, final int itemLevel, + final int maxLayer, final boolean isUpdate) { + T curObj = entryPoint; + if (itemLevel < maxLayer) { + curObj = bestEntryPointUntilLayer(curObj, item, maxLayer, itemLevel, distFnIndex); + } + for (int level = Math.min(itemLevel, maxLayer); level >= 0; level--) { + final DistancedItemQueue candidates = + searchLayerForCandidates(item, curObj, efConstruction, level, distFnIndex, isUpdate); + curObj = mutuallyConnectNewElement(item, candidates, level, isUpdate); + } + } + + /** + * Insert the item into HNSW index. + */ + public void insert(final T item) throws IllegalDuplicateInsertException { + final Lock itemLock = locks.computeIfAbsent(item, lockProvider).writeLock(); + itemLock.lock(); + try { + final HnswMeta metadata = graphMeta.get(); + // If the graph already have the item, should not re-insert it again + // Need to check entry point in case we reinsert first item where is are no graph + // but only a entry point + if (graph.containsKey(HnswNode.from(0, item)) + || (metadata.getEntryPoint().isPresent() + && Objects.equals(metadata.getEntryPoint().get(), item))) { + throw new IllegalDuplicateInsertException( + "Duplicate insertion is not supported: " + item); + } + final int curLevel = getRandomLevel(); + Optional entryPoint = metadata.getEntryPoint(); + // The global lock prevents two threads from making changes to the entry point. This lock + // should get taken very infrequently. Something like log-base-levelMultiplier(num items) + // For a full explanation of locking see this document: http://go/hnsw-locking + int maxLevelCopy = metadata.getMaxLevel(); + if (curLevel > maxLevelCopy) { + globalLock.lock(); + // Re initialize the entryPoint and maxLevel in case these are changed by any other thread + // No need to check the condition again since, + // it is already checked at the end before updating entry point struct + // No need to unlock for optimization and keeping as is if condition fails since threads + // will not be entering this section a lot. + final HnswMeta temp = graphMeta.get(); + entryPoint = temp.getEntryPoint(); + maxLevelCopy = temp.getMaxLevel(); + } + + if (entryPoint.isPresent()) { + wireConnectionForAllLayers(entryPoint.get(), item, curLevel, maxLevelCopy, false); + } + + if (curLevel > maxLevelCopy) { + Preconditions.checkState(globalLock.isHeldByCurrentThread(), + "Global lock not held before updating entry point"); + graphMeta.set(new HnswMeta<>(curLevel, Optional.of(item))); + } + } finally { + if (globalLock.isHeldByCurrentThread()) { + globalLock.unlock(); + } + itemLock.unlock(); + } + } + + /** + * set connections of an element with synchronization + * The only other place that should have the lock for writing is during + * the element insertion + */ + private void setConnectionList(final T item, int layer, List connections) { + final Lock candidateLock = locks.computeIfAbsent(item, lockProvider).writeLock(); + candidateLock.lock(); + try { + graph.put( + HnswNode.from(layer, item), + ImmutableList.copyOf(connections) + ); + } finally { + candidateLock.unlock(); + } + } + + /** + * Reinsert the item into HNSW index. + * This method updates the links of an element assuming + * the element's distance function is changed externally (e.g. by updating the features) + */ + + public void reInsert(final T item) { + final HnswMeta metadata = graphMeta.get(); + + Optional entryPoint = metadata.getEntryPoint(); + + Preconditions.checkState(entryPoint.isPresent(), + "Update cannot be performed if entry point is not present"); + + // This is a check for the single element case + if (entryPoint.get().equals(item) && graph.isEmpty()) { + return; + } + + Preconditions.checkState(graph.containsKey(HnswNode.from(0, item)), + "Graph does not contain the item to be updated at level 0"); + + int curLevel = 0; + + int maxLevelCopy = metadata.getMaxLevel(); + + for (int layer = maxLevelCopy; layer >= 0; layer--) { + if (graph.containsKey(HnswNode.from(layer, item))) { + curLevel = layer; + break; + } + } + + // Updating the links of the elements from the 1-hop radius of the updated element + + for (int layer = 0; layer <= curLevel; layer++) { + + // Filling the element sets for candidates and updated elements + final HashSet setCand = new HashSet(); + final HashSet setNeigh = new HashSet(); + final List listOneHop = getConnectionListForRead(item, layer); + + if (listOneHop.isEmpty()) { + LOG.debug("No links for the updated element. Empty dataset?"); + continue; + } + + setCand.add(item); + + for (T elOneHop : listOneHop) { + setCand.add(elOneHop); + if (randomProvider.get().nextFloat() > updateNeighborProbability) { + continue; + } + setNeigh.add(elOneHop); + final List listTwoHop = getConnectionListForRead(elOneHop, layer); + + if (listTwoHop.isEmpty()) { + LOG.debug("No links for the updated element. Empty dataset?"); + } + + for (T oneHopEl : listTwoHop) { + setCand.add(oneHopEl); + } + } + // No need to update the item itself, so remove it + setNeigh.remove(item); + + // Updating the link lists of elements from setNeigh: + for (T neigh : setNeigh) { + final HashSet setCopy = new HashSet(setCand); + setCopy.remove(neigh); + int keepElementsNum = Math.min(efConstruction, setCopy.size()); + final DistancedItemQueue candidates = new DistancedItemQueue<>( + neigh, + ImmutableList.of(), + false, + distFnIndex + ); + for (T cand : setCopy) { + final float distance = distFnIndex.distance(neigh, cand); + if (candidates.size() < keepElementsNum) { + candidates.enqueue(cand, distance); + } else { + if (distance < candidates.peek().getDistance()) { + candidates.dequeue(); + candidates.enqueue(cand, distance); + } + } + } + final ImmutableList neighbours = selectNearestNeighboursByHeuristic( + candidates, + layer == 0 ? maxM0 : maxM + ); + + final List temp = getConnectionListForRead(neigh, layer); + if (temp.isEmpty()) { + LOG.debug("existing linkslist is empty. Corrupt index"); + } + if (neighbours.isEmpty()) { + LOG.debug("predicted linkslist is empty. Corrupt index"); + } + setConnectionList(neigh, layer, neighbours); + + } + + + } + wireConnectionForAllLayers(metadata.getEntryPoint().get(), item, curLevel, maxLevelCopy, true); + } + + /** + * This method can be used to get the graph statistics, specifically + * it prints the histogram of inbound connections for each element. + */ + private String getStats() { + int histogramMaxBins = 50; + int[] histogram = new int[histogramMaxBins]; + HashMap mmap = new HashMap(); + for (HnswNode key : graph.keySet()) { + if (key.level == 0) { + List linkList = getConnectionListForRead(key.item, key.level); + for (T node : linkList) { + int a = mmap.computeIfAbsent(node, k -> 0); + mmap.put(node, a + 1); + + } + } + } + + for (T key : mmap.keySet()) { + int ind = mmap.get(key) < histogramMaxBins - 1 ? mmap.get(key) : histogramMaxBins - 1; + histogram[ind]++; + } + int minNonZeroIndex; + for (minNonZeroIndex = histogramMaxBins - 1; minNonZeroIndex >= 0; minNonZeroIndex--) { + if (histogram[minNonZeroIndex] > 0) { + break; + } + } + + String output = ""; + for (int i = 0; i <= minNonZeroIndex; i++) { + output += "" + i + "\t" + histogram[i] / (0.01f * mmap.keySet().size()) + "\n"; + } + + return output; + } + + private int getRandomLevel() { + return (int) (-Math.log(randomProvider.get().nextDouble()) * levelMultiplier); + } + + /** + * Note that to avoid deadlocks it is important that this method is called after all the searches + * of the graph have completed. If you take a lock on any items discovered in the graph after + * this, you may get stuck waiting on a thread that is waiting for item to be fully inserted. + *

+ * Note: when using concurrent writers we can miss connections that we would otherwise get. + * This will reduce the recall. + *

+ * For a full explanation of locking see this document: http://go/hnsw-locking + * The method returns the closest nearest neighbor (can be used as an enter point) + */ + private T mutuallyConnectNewElement( + final T item, + final DistancedItemQueue candidates, // Max queue + final int level, + final boolean isUpdate + ) { + + // Using maxM here. Its implementation is ambiguous in HNSW paper, + // so using the way it is getting used in Hnsw lib. + final ImmutableList neighbours = selectNearestNeighboursByHeuristic(candidates, maxM); + setConnectionList(item, level, neighbours); + final int M = level == 0 ? maxM0 : maxM; + for (T nn : neighbours) { + if (nn.equals(item)) { + continue; + } + final Lock curLock = locks.computeIfAbsent(nn, lockProvider).writeLock(); + curLock.lock(); + try { + final HnswNode key = HnswNode.from(level, nn); + final ImmutableList connections = graph.getOrDefault(key, ImmutableList.of()); + final boolean isItemAlreadyPresent = + isUpdate && connections.indexOf(item) != -1 ? true : false; + + // If `item` is already present in the neighboring connections, + // then no need to modify any connections or run the search heuristics. + if (isItemAlreadyPresent) { + continue; + } + + final ImmutableList updatedConnections; + if (connections.size() < M) { + final List temp = new ArrayList<>(connections); + temp.add(item); + updatedConnections = ImmutableList.copyOf(temp.iterator()); + } else { + // Max Queue + final DistancedItemQueue queue = new DistancedItemQueue<>( + nn, + connections, + false, + distFnIndex + ); + queue.enqueue(item); + updatedConnections = selectNearestNeighboursByHeuristic(queue, M); + } + if (updatedConnections.isEmpty()) { + LOG.debug("Internal error: predicted linkslist is empty"); + } + + graph.put(key, updatedConnections); + } finally { + curLock.unlock(); + } + } + return neighbours.get(0); + } + + /* + * bestEntryPointUntilLayer starts the graph search for item from the entry point + * until the searches reaches the selectedLayer layer. + * @return a point from selectedLayer layer, was the closest on the (selectedLayer+1) layer + */ + private T bestEntryPointUntilLayer( + final T entryPoint, + final K item, + int maxLayer, + int selectedLayer, + DistanceFunction distFn + ) { + T curObj = entryPoint; + if (selectedLayer < maxLayer) { + float curDist = distFn.distance(item, curObj); + for (int level = maxLayer; level > selectedLayer; level--) { + boolean changed = true; + while (changed) { + changed = false; + final List list = getConnectionListForRead(curObj, level); + for (T nn : list) { + final float tempDist = distFn.distance(item, nn); + if (tempDist < curDist) { + curDist = tempDist; + curObj = nn; + changed = true; + } + } + } + } + } + + return curObj; + } + + + @VisibleForTesting + protected ImmutableList selectNearestNeighboursByHeuristic( + final DistancedItemQueue candidates, // Max queue + final int maxConnections + ) { + Preconditions.checkState(!candidates.isMinQueue(), + "candidates in selectNearestNeighboursByHeuristic should be a max queue"); + + final T baseElement = candidates.getOrigin(); + if (candidates.size() <= maxConnections) { + List list = candidates.toListWithItem(); + list.remove(baseElement); + return ImmutableList.copyOf(list); + } else { + final List resSet = new ArrayList<>(maxConnections); + // Min queue for closest elements first + final DistancedItemQueue minQueue = candidates.reverse(); + while (minQueue.nonEmpty()) { + if (resSet.size() >= maxConnections) { + break; + } + final DistancedItem candidate = minQueue.dequeue(); + + // We do not want to creates loops: + // While heuristic is used only for creating the links + if (candidate.getItem().equals(baseElement)) { + continue; + } + + boolean toInclude = true; + for (T e : resSet) { + // Do not include candidate if the distance from candidate to any of existing item in + // resSet is closer to the distance from the candidate to the item. By doing this, the + // connection of graph will be more diverse, and in case of highly clustered data set, + // connections will be made between clusters instead of all being in the same cluster. + final float dist = distFnIndex.distance(e, candidate.getItem()); + if (dist < candidate.getDistance()) { + toInclude = false; + break; + } + } + + if (toInclude) { + resSet.add(candidate.getItem()); + } + } + return ImmutableList.copyOf(resSet); + } + } + + /** + * Search the index for the neighbours. + * + * @param query Query + * @param numOfNeighbours Number of neighbours to search for. + * @param ef This param controls the accuracy of the search. + * Bigger the ef better the accuracy on the expense of latency. + * Keep it atleast number of neighbours to find. + * @return Neighbours + */ + public List> searchKnn(final Q query, final int numOfNeighbours, final int ef) { + final HnswMeta metadata = graphMeta.get(); + if (metadata.getEntryPoint().isPresent()) { + T entryPoint = bestEntryPointUntilLayer(metadata.getEntryPoint().get(), + query, metadata.getMaxLevel(), 0, distFnQuery); + // Get the actual neighbours from 0th layer + final List> neighbours = + searchLayerForCandidates(query, entryPoint, Math.max(ef, numOfNeighbours), + 0, distFnQuery, false).dequeueAll(); + Collections.reverse(neighbours); + return neighbours.size() > numOfNeighbours + ? neighbours.subList(0, numOfNeighbours) : neighbours; + } else { + return Collections.emptyList(); + } + } + + // This method is currently not used + // It is needed for debugging purposes only + private void checkIntegrity(String message) { + final HnswMeta metadata = graphMeta.get(); + for (HnswNode node : graph.keySet()) { + List linkList = graph.get(node); + + for (T el : linkList) { + if (el.equals(node.item)) { + LOG.debug(message); + throw new RuntimeException("integrity check failed"); + } + } + } + } + + private DistancedItemQueue searchLayerForCandidates( + final K item, + final T entryPoint, + final int ef, + final int level, + final DistanceFunction distFn, + boolean isUpdate + ) { + // Min queue + final DistancedItemQueue cQueue = new DistancedItemQueue<>( + item, + Collections.singletonList(entryPoint), + true, + distFn + ); + // Max Queue + final DistancedItemQueue wQueue = cQueue.reverse(); + final Set visited = new HashSet<>(); + float lowerBoundDistance = wQueue.peek().getDistance(); + visited.add(entryPoint); + + while (cQueue.nonEmpty()) { + final DistancedItem candidate = cQueue.peek(); + if (candidate.getDistance() > lowerBoundDistance) { + break; + } + + cQueue.dequeue(); + final List list = getConnectionListForRead(candidate.getItem(), level); + for (T nn : list) { + if (!visited.contains(nn)) { + visited.add(nn); + final float distance = distFn.distance(item, nn); + if (wQueue.size() < ef || distance < wQueue.peek().getDistance()) { + cQueue.enqueue(nn, distance); + + if (isUpdate && item.equals(nn)) { + continue; + } + + wQueue.enqueue(nn, distance); + if (wQueue.size() > ef) { + wQueue.dequeue(); + } + + lowerBoundDistance = wQueue.peek().getDistance(); + } + } + } + } + + return wQueue; + } + + /** + * Serialize hnsw index + */ + public void toDirectory(IndexOutputFile indexOutputFile, Injection injection) + throws IOException, TException { + final int totalGraphEntries = HnswIndexIOUtil.saveHnswGraphEntries( + graph, + indexOutputFile.createFile(GRAPH_FILE_NAME).getOutputStream(), + injection); + + HnswIndexIOUtil.saveMetadata( + graphMeta.get(), + efConstruction, + maxM, + totalGraphEntries, + injection, + indexOutputFile.createFile(METADATA_FILE_NAME).getOutputStream()); +} + + /** + * Load hnsw index + */ + public static HnswIndex loadHnswIndex( + DistanceFunction distFnIndex, + DistanceFunction distFnQuery, + AbstractFile directory, + Injection injection, + RandomProvider randomProvider) throws IOException, TException { + final AbstractFile graphFile = directory.getChild(GRAPH_FILE_NAME); + final AbstractFile metadataFile = directory.getChild(METADATA_FILE_NAME); + final HnswInternalIndexMetadata metadata = HnswIndexIOUtil.loadMetadata(metadataFile); + final Map, ImmutableList> graph = + HnswIndexIOUtil.loadHnswGraph(graphFile, injection, metadata.numElements); + final ByteBuffer entryPointBB = metadata.entryPoint; + final HnswMeta graphMeta = new HnswMeta<>( + metadata.maxLevel, + entryPointBB == null ? Optional.empty() + : Optional.of(injection.invert(ArrayByteBufferCodec.decode(entryPointBB)).get()) + ); + return new HnswIndex<>( + distFnIndex, + distFnQuery, + metadata.efConstruction, + metadata.maxM, + metadata.numElements, + graphMeta, + graph, + randomProvider + ); + } + + private List getConnectionListForRead(T node, int level) { + final Lock curLock = locks.computeIfAbsent(node, lockProvider).readLock(); + curLock.lock(); + final List list; + try { + list = graph + .getOrDefault(HnswNode.from(level, node), ImmutableList.of()); + } finally { + curLock.unlock(); + } + + return list; + } + + @VisibleForTesting + AtomicReference> getGraphMeta() { + return graphMeta; + } + + @VisibleForTesting + Map getLocks() { + return locks; + } + + @VisibleForTesting + Map, ImmutableList> getGraph() { + return graph; + } + + public interface RandomProvider { + /** + * RandomProvider interface made public for scala 2.12 compat + */ + Random get(); + } +} diff --git a/ann/src/main/java/com/twitter/ann/hnsw/HnswIndexIOUtil.java b/ann/src/main/java/com/twitter/ann/hnsw/HnswIndexIOUtil.java new file mode 100644 index 0000000000..fba2dc55a9 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/hnsw/HnswIndexIOUtil.java @@ -0,0 +1,133 @@ +package com.twitter.ann.hnsw; + +import java.io.IOException; +import java.io.InputStream; +import java.io.OutputStream; +import java.nio.ByteBuffer; +import java.util.HashMap; +import java.util.List; +import java.util.Map; +import java.util.Set; +import java.util.stream.Collectors; + +import com.google.common.collect.ImmutableList; + +import org.apache.thrift.TDeserializer; +import org.apache.thrift.TException; +import org.apache.thrift.TSerializer; +import org.apache.thrift.protocol.TBinaryProtocol; +import org.apache.thrift.protocol.TProtocol; +import org.apache.thrift.transport.TIOStreamTransport; +import org.apache.thrift.transport.TTransportException; + +import com.twitter.ann.common.thriftjava.HnswGraphEntry; +import com.twitter.ann.common.thriftjava.HnswInternalIndexMetadata; +import com.twitter.bijection.Injection; +import com.twitter.mediaservices.commons.codec.ArrayByteBufferCodec; +import com.twitter.search.common.file.AbstractFile; + +public final class HnswIndexIOUtil { + private HnswIndexIOUtil() { + } + + /** + * Save thrift object in file + */ + public static void saveMetadata( + HnswMeta graphMeta, + int efConstruction, + int maxM, + int numElements, + Injection injection, + OutputStream outputStream + ) throws IOException, TException { + final int maxLevel = graphMeta.getMaxLevel(); + final HnswInternalIndexMetadata metadata = new HnswInternalIndexMetadata( + maxLevel, + efConstruction, + maxM, + numElements + ); + + if (graphMeta.getEntryPoint().isPresent()) { + metadata.setEntryPoint(injection.apply(graphMeta.getEntryPoint().get())); + } + final TSerializer serializer = new TSerializer(new TBinaryProtocol.Factory()); + outputStream.write(serializer.serialize(metadata)); + outputStream.close(); + } + + /** + * Load Hnsw index metadata + */ + public static HnswInternalIndexMetadata loadMetadata(AbstractFile file) + throws IOException, TException { + final HnswInternalIndexMetadata obj = new HnswInternalIndexMetadata(); + final TDeserializer deserializer = new TDeserializer(new TBinaryProtocol.Factory()); + deserializer.deserialize(obj, file.getByteSource().read()); + return obj; + } + + /** + * Load Hnsw graph entries from file + */ + public static Map, ImmutableList> loadHnswGraph( + AbstractFile file, + Injection injection, + int numElements + ) throws IOException, TException { + final InputStream stream = file.getByteSource().openBufferedStream(); + final TProtocol protocol = new TBinaryProtocol(new TIOStreamTransport(stream)); + final Map, ImmutableList> graph = + new HashMap<>(numElements); + while (true) { + try { + final HnswGraphEntry entry = new HnswGraphEntry(); + entry.read(protocol); + final HnswNode node = HnswNode.from(entry.level, + injection.invert(ArrayByteBufferCodec.decode(entry.key)).get()); + final List list = entry.getNeighbours().stream() + .map(bb -> injection.invert(ArrayByteBufferCodec.decode(bb)).get()) + .collect(Collectors.toList()); + graph.put(node, ImmutableList.copyOf(list.iterator())); + } catch (TException e) { + if (e instanceof TTransportException + && TTransportException.class.cast(e).getType() == TTransportException.END_OF_FILE) { + stream.close(); + break; + } + stream.close(); + throw e; + } + } + + return graph; + } + + /** + * Save hnsw graph in file + * + * @return number of keys in the graph + */ + public static int saveHnswGraphEntries( + Map, ImmutableList> graph, + OutputStream outputStream, + Injection injection + ) throws IOException, TException { + final TProtocol protocol = new TBinaryProtocol(new TIOStreamTransport(outputStream)); + final Set> nodes = graph.keySet(); + for (HnswNode node : nodes) { + final HnswGraphEntry entry = new HnswGraphEntry(); + entry.setLevel(node.level); + entry.setKey(injection.apply(node.item)); + final List nn = graph.getOrDefault(node, ImmutableList.of()).stream() + .map(t -> ByteBuffer.wrap(injection.apply(t))) + .collect(Collectors.toList()); + entry.setNeighbours(nn); + entry.write(protocol); + } + + outputStream.close(); + return nodes.size(); + } +} diff --git a/ann/src/main/java/com/twitter/ann/hnsw/HnswMeta.java b/ann/src/main/java/com/twitter/ann/hnsw/HnswMeta.java new file mode 100644 index 0000000000..c990c4bbca --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/hnsw/HnswMeta.java @@ -0,0 +1,45 @@ +package com.twitter.ann.hnsw; + +import java.util.Objects; +import java.util.Optional; + +class HnswMeta { + private final int maxLevel; + private final Optional entryPoint; + + HnswMeta(int maxLevel, Optional entryPoint) { + this.maxLevel = maxLevel; + this.entryPoint = entryPoint; + } + + public int getMaxLevel() { + return maxLevel; + } + + public Optional getEntryPoint() { + return entryPoint; + } + + @Override + public boolean equals(Object o) { + if (this == o) { + return true; + } + if (o == null || getClass() != o.getClass()) { + return false; + } + HnswMeta hnswMeta = (HnswMeta) o; + return maxLevel == hnswMeta.maxLevel + && Objects.equals(entryPoint, hnswMeta.entryPoint); + } + + @Override + public int hashCode() { + return Objects.hash(maxLevel, entryPoint); + } + + @Override + public String toString() { + return "HnswMeta{maxLevel=" + maxLevel + ", entryPoint=" + entryPoint + '}'; + } +} diff --git a/ann/src/main/java/com/twitter/ann/hnsw/HnswNode.java b/ann/src/main/java/com/twitter/ann/hnsw/HnswNode.java new file mode 100644 index 0000000000..95819214cc --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/hnsw/HnswNode.java @@ -0,0 +1,45 @@ +package com.twitter.ann.hnsw; + +import org.apache.commons.lang.builder.EqualsBuilder; +import org.apache.commons.lang.builder.HashCodeBuilder; + +public class HnswNode { + public final int level; + public final T item; + + public HnswNode(int level, T item) { + this.level = level; + this.item = item; + } + + /** + * Create a hnsw node. + */ + public static HnswNode from(int level, T item) { + return new HnswNode<>(level, item); + } + + @Override + public boolean equals(Object o) { + if (o == this) { + return true; + } + if (!(o instanceof HnswNode)) { + return false; + } + + HnswNode that = (HnswNode) o; + return new EqualsBuilder() + .append(this.item, that.item) + .append(this.level, that.level) + .isEquals(); + } + + @Override + public int hashCode() { + return new HashCodeBuilder() + .append(item) + .append(level) + .toHashCode(); + } +} diff --git a/ann/src/main/java/com/twitter/ann/hnsw/IllegalDuplicateInsertException.java b/ann/src/main/java/com/twitter/ann/hnsw/IllegalDuplicateInsertException.java new file mode 100644 index 0000000000..634dc63f64 --- /dev/null +++ b/ann/src/main/java/com/twitter/ann/hnsw/IllegalDuplicateInsertException.java @@ -0,0 +1,7 @@ +package com.twitter.ann.hnsw; + +public class IllegalDuplicateInsertException extends Exception { + public IllegalDuplicateInsertException(String message) { + super(message); + } +} diff --git a/ann/src/main/python/dataflow/BUILD.bazel b/ann/src/main/python/dataflow/BUILD.bazel new file mode 100644 index 0000000000..44eeb72e2a --- /dev/null +++ b/ann/src/main/python/dataflow/BUILD.bazel @@ -0,0 +1,38 @@ +resources( + name = "sql", + sources = ["bq.sql"], +) + +python3_library( + name = "faiss_indexing", + sources = ["**/*.py"], + tags = ["bazel-compatible"], + dependencies = [ + ":sql", + "3rdparty/python/apache-beam:default", + "3rdparty/python/faiss-gpu:default", + "3rdparty/python/gcsfs:default", + "3rdparty/python/google-cloud-bigquery:default", + "3rdparty/python/google-cloud-storage", + "3rdparty/python/numpy:default", + "3rdparty/python/pandas:default", + "3rdparty/python/pandas-gbq:default", + "3rdparty/python/pyarrow:default", + "src/python/twitter/ml/common/apache_beam", + ], +) + +python37_binary( + name = "faiss_indexing_bin", + sources = ["faiss_index_bq_dataset.py"], + platforms = [ + "current", + "linux_x86_64", + ], + tags = ["no-mypy"], + zip_safe = False, + dependencies = [ + ":faiss_indexing", + "3rdparty/python/_closures/ann/src/main/python/dataflow:faiss_indexing_bin", + ], +) diff --git a/ann/src/main/python/dataflow/bq.sql b/ann/src/main/python/dataflow/bq.sql new file mode 100644 index 0000000000..809184bb2d --- /dev/null +++ b/ann/src/main/python/dataflow/bq.sql @@ -0,0 +1,6 @@ +WITH maxts as (SELECT as value MAX(ts) as ts FROM `twttr-recos-ml-prod.ssedhain.twhin_tweet_avg_embedding`) +SELECT entityId, embedding +FROM `twttr-recos-ml-prod.ssedhain.twhin_tweet_avg_embedding` +WHERE ts >= (select max(maxts) from maxts) +AND DATE(TIMESTAMP_MILLIS(createdAt)) <= (select max(maxts) from maxts) +AND DATE(TIMESTAMP_MILLIS(createdAt)) >= DATE_SUB((select max(maxts) from maxts), INTERVAL 1 DAY) \ No newline at end of file diff --git a/ann/src/main/python/dataflow/faiss_index_bq_dataset.py b/ann/src/main/python/dataflow/faiss_index_bq_dataset.py new file mode 100644 index 0000000000..1863cabefa --- /dev/null +++ b/ann/src/main/python/dataflow/faiss_index_bq_dataset.py @@ -0,0 +1,232 @@ +import argparse +import logging +import os +import pkgutil +import sys +from urllib.parse import urlsplit + +import apache_beam as beam +from apache_beam.options.pipeline_options import PipelineOptions +import faiss + + +def parse_d6w_config(argv=None): + """Parse d6w config. + :param argv: d6w config + :return: dictionary containing d6w config + """ + + parser = argparse.ArgumentParser( + description="See https://docbird.twitter.biz/d6w/model.html for any parameters inherited from d6w job config" + ) + parser.add_argument("--job_name", dest="job_name", required=True, help="d6w attribute") + parser.add_argument("--project", dest="project", required=True, help="d6w attribute") + parser.add_argument( + "--staging_location", dest="staging_location", required=True, help="d6w attribute" + ) + parser.add_argument("--temp_location", dest="temp_location", required=True, help="d6w attribute") + parser.add_argument( + "--output_location", + dest="output_location", + required=True, + help="GCS bucket and path where resulting artifacts are uploaded", + ) + parser.add_argument( + "--service_account_email", dest="service_account_email", required=True, help="d6w attribute" + ) + parser.add_argument( + "--factory_string", + dest="factory_string", + required=False, + help="FAISS factory string describing index to build. See https://github.com/facebookresearch/faiss/wiki/The-index-factory", + ) + parser.add_argument( + "--metric", + dest="metric", + required=True, + help="Metric used to compute distance between embeddings. Valid values are 'l2', 'ip', 'l1', 'linf'", + ) + parser.add_argument( + "--use_gpu", + dest="gpu", + required=True, + help="--use_gpu=yes if you want to use GPU during index building", + ) + + known_args, unknown_args = parser.parse_known_args(argv) + d6w_config = vars(known_args) + d6w_config["gpu"] = d6w_config["gpu"].lower() == "yes" + d6w_config["metric"] = parse_metric(d6w_config) + + """ + WARNING: Currently, d6w (a Twitter tool used to deploy Dataflow jobs to GCP) and + PipelineOptions.for_dataflow_runner (a helper method in twitter.ml.common.apache_beam) do not + play nicely together. The helper method will overwrite some of the config specified in the d6w + file using the defaults in https://sourcegraph.twitter.biz/git.twitter.biz/source/-/blob/src/python/twitter/ml/common/apache_beam/__init__.py?L24.' + However, the d6w output message will still report that the config specified in the d6w file was used. + """ + logging.warning( + f"The following d6w config parameters will be overwritten by the defaults in " + f"https://sourcegraph.twitter.biz/git.twitter.biz/source/-/blob/src/python/twitter/ml/common/apache_beam/__init__.py?L24\n" + f"{str(unknown_args)}" + ) + return d6w_config + + +def get_bq_query(): + """ + Query is expected to return rows with unique entityId + """ + return pkgutil.get_data(__name__, "bq.sql").decode("utf-8") + + +def parse_metric(config): + metric_str = config["metric"].lower() + if metric_str == "l2": + return faiss.METRIC_L2 + elif metric_str == "ip": + return faiss.METRIC_INNER_PRODUCT + elif metric_str == "l1": + return faiss.METRIC_L1 + elif metric_str == "linf": + return faiss.METRIC_Linf + else: + raise Exception(f"Uknown metric: {metric_str}") + + +def run_pipeline(argv=[]): + config = parse_d6w_config(argv) + argv_with_extras = argv + if config["gpu"]: + argv_with_extras.extend(["--experiments", "use_runner_v2"]) + argv_with_extras.extend( + ["--experiments", "worker_accelerator=type:nvidia-tesla-t4;count:1;install-nvidia-driver"] + ) + argv_with_extras.extend( + [ + "--worker_harness_container_image", + "gcr.io/twttr-recos-ml-prod/dataflow-gpu/beam2_39_0_py3_7", + ] + ) + + options = PipelineOptions(argv_with_extras) + output_bucket_name = urlsplit(config["output_location"]).netloc + + with beam.Pipeline(options=options) as p: + input_data = p | "Read from BigQuery" >> beam.io.ReadFromBigQuery( + method=beam.io.ReadFromBigQuery.Method.DIRECT_READ, + query=get_bq_query(), + use_standard_sql=True, + ) + + index_built = input_data | "Build and upload index" >> beam.CombineGlobally( + MergeAndBuildIndex( + output_bucket_name, + config["output_location"], + config["factory_string"], + config["metric"], + config["gpu"], + ) + ) + + # Make linter happy + index_built + + +class MergeAndBuildIndex(beam.CombineFn): + def __init__(self, bucket_name, gcs_output_path, factory_string, metric, gpu): + self.bucket_name = bucket_name + self.gcs_output_path = gcs_output_path + self.factory_string = factory_string + self.metric = metric + self.gpu = gpu + + def create_accumulator(self): + return [] + + def add_input(self, accumulator, element): + accumulator.append(element) + return accumulator + + def merge_accumulators(self, accumulators): + merged = [] + for accum in accumulators: + merged.extend(accum) + return merged + + def extract_output(self, rows): + # Reimports are needed on workers + import glob + import subprocess + + import faiss + from google.cloud import storage + import numpy as np + + client = storage.Client() + bucket = client.get_bucket(self.bucket_name) + + logging.info("Building FAISS index") + logging.info(f"There are {len(rows)} rows") + + ids = np.array([x["entityId"] for x in rows]).astype("long") + embeds = np.array([x["embedding"] for x in rows]).astype("float32") + dimensions = len(embeds[0]) + N = ids.shape[0] + logging.info(f"There are {dimensions} dimensions") + + if self.factory_string is None: + M = 48 + + divideable_dimensions = (dimensions // M) * M + if divideable_dimensions != dimensions: + opq_prefix = f"OPQ{M}_{divideable_dimensions}" + else: + opq_prefix = f"OPQ{M}" + + clusters = N // 20 + self.factory_string = f"{opq_prefix},IVF{clusters},PQ{M}" + + logging.info(f"Factory string is {self.factory_string}, metric={self.metric}") + + if self.gpu: + logging.info("Using GPU") + + res = faiss.StandardGpuResources() + cpu_index = faiss.index_factory(dimensions, self.factory_string, self.metric) + cpu_index = faiss.IndexIDMap(cpu_index) + gpu_index = faiss.index_cpu_to_gpu(res, 0, cpu_index) + gpu_index.train(embeds) + gpu_index.add_with_ids(embeds, ids) + cpu_index = faiss.index_gpu_to_cpu(gpu_index) + else: + logging.info("Using CPU") + + cpu_index = faiss.index_factory(dimensions, self.factory_string, self.metric) + cpu_index = faiss.IndexIDMap(cpu_index) + cpu_index.train(embeds) + cpu_index.add_with_ids(embeds, ids) + + logging.info("Built faiss index") + + local_path = "/indices" + logging.info(f"Writing indices to local {local_path}") + subprocess.run(f"mkdir -p {local_path}".strip().split()) + local_index_path = os.path.join(local_path, "result.index") + + faiss.write_index(cpu_index, local_index_path) + logging.info(f"Done writing indices to local {local_path}") + + logging.info(f"Uploading to GCS with path {self.gcs_output_path}") + assert os.path.isdir(local_path) + for local_file in glob.glob(local_path + "/*"): + remote_path = os.path.join( + self.gcs_output_path.split("/")[-1], local_file[1 + len(local_path) :] + ) + blob = bucket.blob(remote_path) + blob.upload_from_filename(local_file) + + +if __name__ == "__main__": + logging.getLogger().setLevel(logging.INFO) + run_pipeline(sys.argv) diff --git a/ann/src/main/python/dataflow/worker_harness/Dockerfile b/ann/src/main/python/dataflow/worker_harness/Dockerfile new file mode 100644 index 0000000000..c5f56e2314 --- /dev/null +++ b/ann/src/main/python/dataflow/worker_harness/Dockerfile @@ -0,0 +1,34 @@ +FROM --platform=linux/amd64 nvidia/cuda:11.2.2-cudnn8-runtime-ubuntu20.04 + +RUN \ + # Add Deadsnakes repository that has a variety of Python packages for Ubuntu. + # See: https://launchpad.net/~deadsnakes/+archive/ubuntu/ppa + apt-key adv --keyserver keyserver.ubuntu.com --recv-keys F23C5A6CF475977595C89F51BA6932366A755776 \ + && echo "deb http://ppa.launchpad.net/deadsnakes/ppa/ubuntu focal main" >> /etc/apt/sources.list.d/custom.list \ + && echo "deb-src http://ppa.launchpad.net/deadsnakes/ppa/ubuntu focal main" >> /etc/apt/sources.list.d/custom.list \ + && apt-get update \ + && apt-get install -y curl \ + python3.7 \ + # With python3.8 package, distutils need to be installed separately. + python3.7-distutils \ + python3-dev \ + python3.7-dev \ + libpython3.7-dev \ + python3-apt \ + gcc \ + g++ \ + && rm -rf /var/lib/apt/lists/* +RUN update-alternatives --install /usr/bin/python python /usr/bin/python3.7 10 +RUN rm -f /usr/bin/python3 && ln -s /usr/bin/python3.7 /usr/bin/python3 +RUN \ + curl https://bootstrap.pypa.io/get-pip.py | python \ + && pip3 install pip==22.0.3 \ + && python3 -m pip install --no-cache-dir apache-beam[gcp]==2.39.0 +# Verify that there are no conflicting dependencies. +RUN pip3 check + +# Copy the Apache Beam worker dependencies from the Beam Python 3.7 SDK image. +COPY --from=apache/beam_python3.7_sdk:2.39.0 /opt/apache/beam /opt/apache/beam + +# Set the entrypoint to Apache Beam SDK worker launcher. +ENTRYPOINT [ "/opt/apache/beam/boot" ] \ No newline at end of file diff --git a/ann/src/main/python/dataflow/worker_harness/cloudbuild.yml b/ann/src/main/python/dataflow/worker_harness/cloudbuild.yml new file mode 100644 index 0000000000..34aab720f5 --- /dev/null +++ b/ann/src/main/python/dataflow/worker_harness/cloudbuild.yml @@ -0,0 +1,6 @@ +steps: +- name: 'gcr.io/cloud-builders/docker' + args: ['build', '-t', 'gcr.io/twttr-recos-ml-prod/dataflow-gpu/beam2_39_0_py3_7', '.'] +- name: 'gcr.io/cloud-builders/docker' + args: ['push', 'gcr.io/twttr-recos-ml-prod/dataflow-gpu/beam2_39_0_py3_7'] +images: ['gcr.io/twttr-recos-ml-prod/dataflow-gpu/beam2_39_0_py3_7'] \ No newline at end of file diff --git a/ann/src/main/scala/com/twitter/ann/annoy/AnnoyCommon.scala b/ann/src/main/scala/com/twitter/ann/annoy/AnnoyCommon.scala new file mode 100644 index 0000000000..ffe035c487 --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/annoy/AnnoyCommon.scala @@ -0,0 +1,44 @@ +package com.twitter.ann.annoy + +import com.twitter.ann.common.RuntimeParams +import com.twitter.ann.common.thriftscala.AnnoyIndexMetadata +import com.twitter.bijection.Injection +import com.twitter.mediaservices.commons.codec.ThriftByteBufferCodec +import com.twitter.ann.common.thriftscala.{AnnoyRuntimeParam, RuntimeParams => ServiceRuntimeParams} +import scala.util.{Failure, Success, Try} + +object AnnoyCommon { + private[annoy] lazy val MetadataCodec = new ThriftByteBufferCodec(AnnoyIndexMetadata) + private[annoy] val IndexFileName = "annoy_index" + private[annoy] val MetaDataFileName = "annoy_index_metadata" + private[annoy] val IndexIdMappingFileName = "annoy_index_id_mapping" + + val RuntimeParamsInjection: Injection[AnnoyRuntimeParams, ServiceRuntimeParams] = + new Injection[AnnoyRuntimeParams, ServiceRuntimeParams] { + override def apply(scalaParams: AnnoyRuntimeParams): ServiceRuntimeParams = { + ServiceRuntimeParams.AnnoyParam( + AnnoyRuntimeParam( + scalaParams.nodesToExplore + ) + ) + } + + override def invert(thriftParams: ServiceRuntimeParams): Try[AnnoyRuntimeParams] = + thriftParams match { + case ServiceRuntimeParams.AnnoyParam(annoyParam) => + Success( + AnnoyRuntimeParams(annoyParam.numOfNodesToExplore) + ) + case p => Failure(new IllegalArgumentException(s"Expected AnnoyRuntimeParams got $p")) + } + } +} + +case class AnnoyRuntimeParams( + /* Number of vectors to evaluate while searching. A larger value will give more accurate results, but will take longer time to return. + * Default value would be numberOfTrees*numberOfNeigboursRequested + */ + nodesToExplore: Option[Int]) + extends RuntimeParams { + override def toString: String = s"AnnoyRuntimeParams( nodesToExplore = $nodesToExplore)" +} diff --git a/ann/src/main/scala/com/twitter/ann/annoy/BUILD b/ann/src/main/scala/com/twitter/ann/annoy/BUILD new file mode 100644 index 0000000000..fb882dac25 --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/annoy/BUILD @@ -0,0 +1,23 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/com/spotify:annoy-java", + "3rdparty/jvm/com/spotify:annoy-snapshot", + "3rdparty/jvm/com/twitter/storehaus:core", + "ann/src/main/scala/com/twitter/ann/common", + "ann/src/main/scala/com/twitter/ann/file_store", + "ann/src/main/thrift/com/twitter/ann/common:ann-common-scala", + "mediaservices/commons", + "src/java/com/twitter/search/common/file", + "src/scala/com/twitter/ml/api/embedding", + ], + exports = [ + "ann/src/main/scala/com/twitter/ann/common", + "src/java/com/twitter/common_internal/hadoop", + "src/java/com/twitter/search/common/file", + "src/scala/com/twitter/ml/api/embedding", + ], +) diff --git a/ann/src/main/scala/com/twitter/ann/annoy/RawAnnoyIndexBuilder.scala b/ann/src/main/scala/com/twitter/ann/annoy/RawAnnoyIndexBuilder.scala new file mode 100644 index 0000000000..d1f889a96c --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/annoy/RawAnnoyIndexBuilder.scala @@ -0,0 +1,123 @@ +package com.twitter.ann.annoy + +import com.spotify.annoy.jni.base.{Annoy => AnnoyLib} +import com.twitter.ann.annoy.AnnoyCommon.IndexFileName +import com.twitter.ann.annoy.AnnoyCommon.MetaDataFileName +import com.twitter.ann.annoy.AnnoyCommon.MetadataCodec +import com.twitter.ann.common.EmbeddingType._ +import com.twitter.ann.common._ +import com.twitter.ann.common.thriftscala.AnnoyIndexMetadata +import com.twitter.concurrent.AsyncSemaphore +import com.twitter.mediaservices.commons.codec.ArrayByteBufferCodec +import com.twitter.search.common.file.AbstractFile +import com.twitter.search.common.file.LocalFile +import com.twitter.util.Future +import com.twitter.util.FuturePool +import java.io.File +import java.nio.file.Files +import org.apache.beam.sdk.io.fs.ResourceId +import scala.collection.JavaConverters._ + +private[annoy] object RawAnnoyIndexBuilder { + private[annoy] def apply[D <: Distance[D]]( + dimension: Int, + numOfTrees: Int, + metric: Metric[D], + futurePool: FuturePool + ): RawAppendable[AnnoyRuntimeParams, D] with Serialization = { + val indexBuilder = AnnoyLib.newIndex(dimension, annoyMetric(metric)) + new RawAnnoyIndexBuilder(dimension, numOfTrees, metric, indexBuilder, futurePool) + } + + private[this] def annoyMetric(metric: Metric[_]): AnnoyLib.Metric = { + metric match { + case L2 => AnnoyLib.Metric.EUCLIDEAN + case Cosine => AnnoyLib.Metric.ANGULAR + case _ => throw new RuntimeException("Not supported: " + metric) + } + } +} + +private[this] class RawAnnoyIndexBuilder[D <: Distance[D]]( + dimension: Int, + numOfTrees: Int, + metric: Metric[D], + indexBuilder: AnnoyLib.Builder, + futurePool: FuturePool) + extends RawAppendable[AnnoyRuntimeParams, D] + with Serialization { + private[this] var counter = 0 + // Note: Only one thread can access the underlying index, multithreaded index building not supported + private[this] val semaphore = new AsyncSemaphore(1) + + override def append(embedding: EmbeddingVector): Future[Long] = + semaphore.acquireAndRun({ + counter += 1 + indexBuilder.addItem( + counter, + embedding.toArray + .map(float => float2Float(float)) + .toList + .asJava + ) + + Future.value(counter) + }) + + override def toQueryable: Queryable[Long, AnnoyRuntimeParams, D] = { + val tempDirParent = Files.createTempDirectory("raw_annoy_index").toFile + tempDirParent.deleteOnExit + val tempDir = new LocalFile(tempDirParent) + this.toDirectory(tempDir) + RawAnnoyQueryIndex( + dimension, + metric, + futurePool, + tempDir + ) + } + + override def toDirectory(directory: ResourceId): Unit = { + toDirectory(new IndexOutputFile(directory)) + } + + /** + * Serialize the annoy index in a directory. + * @param directory: Directory to save to. + */ + override def toDirectory(directory: AbstractFile): Unit = { + toDirectory(new IndexOutputFile(directory)) + } + + private def toDirectory(directory: IndexOutputFile): Unit = { + val indexFile = directory.createFile(IndexFileName) + saveIndex(indexFile) + + val metaDataFile = directory.createFile(MetaDataFileName) + saveMetadata(metaDataFile) + } + + private[this] def saveIndex(indexFile: IndexOutputFile): Unit = { + val index = indexBuilder + .build(numOfTrees) + val temp = new LocalFile(File.createTempFile(IndexFileName, null)) + index.save(temp.getPath) + indexFile.copyFrom(temp.getByteSource.openStream()) + temp.delete() + } + + private[this] def saveMetadata(metadataFile: IndexOutputFile): Unit = { + val numberOfVectorsIndexed = counter + val metadata = AnnoyIndexMetadata( + dimension, + Metric.toThrift(metric), + numOfTrees, + numberOfVectorsIndexed + ) + val bytes = ArrayByteBufferCodec.decode(MetadataCodec.encode(metadata)) + val temp = new LocalFile(File.createTempFile(MetaDataFileName, null)) + temp.getByteSink.write(bytes) + metadataFile.copyFrom(temp.getByteSource.openStream()) + temp.delete() + } +} diff --git a/ann/src/main/scala/com/twitter/ann/annoy/RawAnnoyQueryIndex.scala b/ann/src/main/scala/com/twitter/ann/annoy/RawAnnoyQueryIndex.scala new file mode 100644 index 0000000000..9fc65ddf72 --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/annoy/RawAnnoyQueryIndex.scala @@ -0,0 +1,142 @@ +package com.twitter.ann.annoy + +import com.spotify.annoy.{ANNIndex, IndexType} +import com.twitter.ann.annoy.AnnoyCommon._ +import com.twitter.ann.common._ +import com.twitter.ann.common.EmbeddingType._ +import com.twitter.mediaservices.commons.codec.ArrayByteBufferCodec +import com.twitter.search.common.file.{AbstractFile, LocalFile} +import com.twitter.util.{Future, FuturePool} +import java.io.File +import scala.collection.JavaConverters._ + +private[annoy] object RawAnnoyQueryIndex { + private[annoy] def apply[D <: Distance[D]]( + dimension: Int, + metric: Metric[D], + futurePool: FuturePool, + directory: AbstractFile + ): Queryable[Long, AnnoyRuntimeParams, D] = { + val metadataFile = directory.getChild(MetaDataFileName) + val indexFile = directory.getChild(IndexFileName) + val metadata = MetadataCodec.decode( + ArrayByteBufferCodec.encode(metadataFile.getByteSource.read()) + ) + + val existingDimension = metadata.dimension + assert( + existingDimension == dimension, + s"Dimensions do not match. requested: $dimension existing: $existingDimension" + ) + + val existingMetric = Metric.fromThrift(metadata.distanceMetric) + assert( + existingMetric == metric, + s"DistanceMetric do not match. requested: $metric existing: $existingMetric" + ) + + val index = loadIndex(indexFile, dimension, annoyMetric(metric)) + new RawAnnoyQueryIndex[D]( + dimension, + metric, + metadata.numOfTrees, + index, + futurePool + ) + } + + private[this] def annoyMetric(metric: Metric[_]): IndexType = { + metric match { + case L2 => IndexType.EUCLIDEAN + case Cosine => IndexType.ANGULAR + case _ => throw new RuntimeException("Not supported: " + metric) + } + } + + private[this] def loadIndex( + indexFile: AbstractFile, + dimension: Int, + indexType: IndexType + ): ANNIndex = { + var localIndexFile = indexFile + + // If not a local file copy to local, so that it can be memory mapped. + if (!indexFile.isInstanceOf[LocalFile]) { + val tempFile = File.createTempFile(IndexFileName, null) + tempFile.deleteOnExit() + + val temp = new LocalFile(tempFile) + indexFile.copyTo(temp) + localIndexFile = temp + } + + new ANNIndex( + dimension, + localIndexFile.getPath(), + indexType + ) + } +} + +private[this] class RawAnnoyQueryIndex[D <: Distance[D]]( + dimension: Int, + metric: Metric[D], + numOfTrees: Int, + index: ANNIndex, + futurePool: FuturePool) + extends Queryable[Long, AnnoyRuntimeParams, D] + with AutoCloseable { + override def query( + embedding: EmbeddingVector, + numOfNeighbours: Int, + runtimeParams: AnnoyRuntimeParams + ): Future[List[Long]] = { + queryWithDistance(embedding, numOfNeighbours, runtimeParams) + .map(_.map(_.neighbor)) + } + + override def queryWithDistance( + embedding: EmbeddingVector, + numOfNeighbours: Int, + runtimeParams: AnnoyRuntimeParams + ): Future[List[NeighborWithDistance[Long, D]]] = { + futurePool { + val queryVector = embedding.toArray + val neigboursToRequest = neighboursToRequest(numOfNeighbours, runtimeParams) + val neigbours = index + .getNearestWithDistance(queryVector, neigboursToRequest) + .asScala + .take(numOfNeighbours) + .map { nn => + val id = nn.getFirst.toLong + val distance = metric.fromAbsoluteDistance(nn.getSecond) + NeighborWithDistance(id, distance) + } + .toList + + neigbours + } + } + + // Annoy java lib do not expose param for numOfNodesToExplore. + // Default number is numOfTrees*numOfNeigbours. + // Simple hack is to artificially increase the numOfNeighbours to be requested and then just cap it before returning. + private[this] def neighboursToRequest( + numOfNeighbours: Int, + annoyParams: AnnoyRuntimeParams + ): Int = { + annoyParams.nodesToExplore match { + case Some(nodesToExplore) => { + val neigboursToRequest = nodesToExplore / numOfTrees + if (neigboursToRequest < numOfNeighbours) + numOfNeighbours + else + neigboursToRequest + } + case _ => numOfNeighbours + } + } + + // To close the memory map based file resource. + override def close(): Unit = index.close() +} diff --git a/ann/src/main/scala/com/twitter/ann/annoy/TypedAnnoyIndex.scala b/ann/src/main/scala/com/twitter/ann/annoy/TypedAnnoyIndex.scala new file mode 100644 index 0000000000..e686bbe5e3 --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/annoy/TypedAnnoyIndex.scala @@ -0,0 +1,55 @@ +package com.twitter.ann.annoy + +import com.twitter.ann.common._ +import com.twitter.bijection.Injection +import com.twitter.search.common.file.AbstractFile +import com.twitter.util.FuturePool + +// Class to provide Annoy based ann index. +object TypedAnnoyIndex { + + /** + * Create Annoy based typed index builder that serializes index to a directory (HDFS/Local file system). + * It cannot be used in scalding as it leverage C/C++ jni bindings, whose build conflicts with version of some libs installed on hadoop. + * You can use it on aurora or with IndexBuilding job which triggers scalding job but then streams data to aurora machine for building index. + * @param dimension dimension of embedding + * @param numOfTrees builds a forest of numOfTrees trees. + * More trees gives higher precision when querying at the cost of increased memory and disk storage requirement at the build time. + * At runtime the index will be memory mapped, so memory wont be an issue but disk storage would be needed. + * @param metric distance metric for nearest neighbour search + * @param injection Injection to convert bytes to Id. + * @tparam T Type of Id for embedding + * @tparam D Typed Distance + * @return Serializable AnnoyIndex + */ + def indexBuilder[T, D <: Distance[D]]( + dimension: Int, + numOfTrees: Int, + metric: Metric[D], + injection: Injection[T, Array[Byte]], + futurePool: FuturePool + ): Appendable[T, AnnoyRuntimeParams, D] with Serialization = { + TypedAnnoyIndexBuilderWithFile(dimension, numOfTrees, metric, injection, futurePool) + } + + /** + * Load Annoy based queryable index from a directory + * @param dimension dimension of embedding + * @param metric distance metric for nearest neighbour search + * @param injection Injection to convert bytes to Id. + * @param futurePool FuturePool + * @param directory Directory (HDFS/Local file system) where serialized index is stored. + * @tparam T Type of Id for embedding + * @tparam D Typed Distance + * @return Typed Queryable AnnoyIndex + */ + def loadQueryableIndex[T, D <: Distance[D]]( + dimension: Int, + metric: Metric[D], + injection: Injection[T, Array[Byte]], + futurePool: FuturePool, + directory: AbstractFile + ): Queryable[T, AnnoyRuntimeParams, D] = { + TypedAnnoyQueryIndexWithFile(dimension, metric, injection, futurePool, directory) + } +} diff --git a/ann/src/main/scala/com/twitter/ann/annoy/TypedAnnoyIndexBuilderWithFile.scala b/ann/src/main/scala/com/twitter/ann/annoy/TypedAnnoyIndexBuilderWithFile.scala new file mode 100644 index 0000000000..1dd07ae385 --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/annoy/TypedAnnoyIndexBuilderWithFile.scala @@ -0,0 +1,55 @@ +package com.twitter.ann.annoy + +import com.twitter.ann.annoy.AnnoyCommon.IndexIdMappingFileName +import com.twitter.ann.common._ +import com.twitter.ann.file_store.WritableIndexIdFileStore +import com.twitter.bijection.Injection +import com.twitter.search.common.file.AbstractFile +import com.twitter.util.Future +import com.twitter.util.FuturePool +import org.apache.beam.sdk.io.fs.ResourceId + +private[annoy] object TypedAnnoyIndexBuilderWithFile { + private[annoy] def apply[T, D <: Distance[D]]( + dimension: Int, + numOfTrees: Int, + metric: Metric[D], + injection: Injection[T, Array[Byte]], + futurePool: FuturePool + ): Appendable[T, AnnoyRuntimeParams, D] with Serialization = { + val index = RawAnnoyIndexBuilder(dimension, numOfTrees, metric, futurePool) + val writableFileStore = WritableIndexIdFileStore(injection) + new TypedAnnoyIndexBuilderWithFile[T, D](index, writableFileStore) + } +} + +private[this] class TypedAnnoyIndexBuilderWithFile[T, D <: Distance[D]]( + indexBuilder: RawAppendable[AnnoyRuntimeParams, D] with Serialization, + store: WritableIndexIdFileStore[T]) + extends Appendable[T, AnnoyRuntimeParams, D] + with Serialization { + private[this] val transformedIndex = IndexTransformer.transformAppendable(indexBuilder, store) + + override def append(entity: EntityEmbedding[T]): Future[Unit] = { + transformedIndex.append(entity) + } + + override def toDirectory(directory: ResourceId): Unit = { + indexBuilder.toDirectory(directory) + toDirectory(new IndexOutputFile(directory)) + } + + override def toDirectory(directory: AbstractFile): Unit = { + indexBuilder.toDirectory(directory) + toDirectory(new IndexOutputFile(directory)) + } + + private def toDirectory(directory: IndexOutputFile): Unit = { + val indexIdFile = directory.createFile(IndexIdMappingFileName) + store.save(indexIdFile) + } + + override def toQueryable: Queryable[T, AnnoyRuntimeParams, D] = { + transformedIndex.toQueryable + } +} diff --git a/ann/src/main/scala/com/twitter/ann/annoy/TypedAnnoyQueryIndexWithFile.scala b/ann/src/main/scala/com/twitter/ann/annoy/TypedAnnoyQueryIndexWithFile.scala new file mode 100644 index 0000000000..b728397650 --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/annoy/TypedAnnoyQueryIndexWithFile.scala @@ -0,0 +1,42 @@ +package com.twitter.ann.annoy + +import com.twitter.ann.annoy.AnnoyCommon._ +import com.twitter.ann.common._ +import com.twitter.ann.file_store.ReadableIndexIdFileStore +import com.twitter.bijection.Injection +import com.twitter.search.common.file.AbstractFile +import com.twitter.util.FuturePool + +private[annoy] object TypedAnnoyQueryIndexWithFile { + private[annoy] def apply[T, D <: Distance[D]]( + dimension: Int, + metric: Metric[D], + injection: Injection[T, Array[Byte]], + futurePool: FuturePool, + directory: AbstractFile + ): Queryable[T, AnnoyRuntimeParams, D] = { + val deserializer = + new TypedAnnoyQueryIndexWithFile(dimension, metric, futurePool, injection) + deserializer.fromDirectory(directory) + } +} + +private[this] class TypedAnnoyQueryIndexWithFile[T, D <: Distance[D]]( + dimension: Int, + metric: Metric[D], + futurePool: FuturePool, + injection: Injection[T, Array[Byte]]) + extends QueryableDeserialization[ + T, + AnnoyRuntimeParams, + D, + Queryable[T, AnnoyRuntimeParams, D] + ] { + override def fromDirectory(directory: AbstractFile): Queryable[T, AnnoyRuntimeParams, D] = { + val index = RawAnnoyQueryIndex(dimension, metric, futurePool, directory) + + val indexIdFile = directory.getChild(IndexIdMappingFileName) + val readableFileStore = ReadableIndexIdFileStore(indexIdFile, injection) + IndexTransformer.transformQueryable(index, readableFileStore) + } +} diff --git a/ann/src/main/scala/com/twitter/ann/brute_force/BUILD b/ann/src/main/scala/com/twitter/ann/brute_force/BUILD new file mode 100644 index 0000000000..4f06431c80 --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/brute_force/BUILD @@ -0,0 +1,12 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "ann/src/main/scala/com/twitter/ann/common", + "ann/src/main/scala/com/twitter/ann/serialization", + "ann/src/main/thrift/com/twitter/ann/serialization:serialization-scala", + "src/java/com/twitter/search/common/file", + ], +) diff --git a/ann/src/main/scala/com/twitter/ann/brute_force/BruteForceDeserialization.scala b/ann/src/main/scala/com/twitter/ann/brute_force/BruteForceDeserialization.scala new file mode 100644 index 0000000000..615c39c958 --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/brute_force/BruteForceDeserialization.scala @@ -0,0 +1,64 @@ +package com.twitter.ann.brute_force + +import com.google.common.annotations.VisibleForTesting +import com.twitter.ann.common.{Distance, EntityEmbedding, Metric, QueryableDeserialization} +import com.twitter.ann.serialization.{PersistedEmbeddingInjection, ThriftIteratorIO} +import com.twitter.ann.serialization.thriftscala.PersistedEmbedding +import com.twitter.search.common.file.{AbstractFile, LocalFile} +import com.twitter.util.FuturePool +import java.io.File + +/** + * @param factory creates a BruteForceIndex from the arguments. This is only exposed for testing. + * If for some reason you pass this arg in make sure that it eagerly consumes the + * iterator. If you don't you might close the input stream that the iterator is + * using. + * @tparam T the id of the embeddings + */ +class BruteForceDeserialization[T, D <: Distance[D]] @VisibleForTesting private[brute_force] ( + metric: Metric[D], + embeddingInjection: PersistedEmbeddingInjection[T], + futurePool: FuturePool, + thriftIteratorIO: ThriftIteratorIO[PersistedEmbedding], + factory: (Metric[D], FuturePool, Iterator[EntityEmbedding[T]]) => BruteForceIndex[T, D]) + extends QueryableDeserialization[T, BruteForceRuntimeParams.type, D, BruteForceIndex[T, D]] { + import BruteForceIndex._ + + def this( + metric: Metric[D], + embeddingInjection: PersistedEmbeddingInjection[T], + futurePool: FuturePool, + thriftIteratorIO: ThriftIteratorIO[PersistedEmbedding] + ) = { + this( + metric, + embeddingInjection, + futurePool, + thriftIteratorIO, + factory = BruteForceIndex.apply[T, D] + ) + } + + override def fromDirectory( + serializationDirectory: AbstractFile + ): BruteForceIndex[T, D] = { + val file = File.createTempFile(DataFileName, "tmp") + file.deleteOnExit() + val temp = new LocalFile(file) + val dataFile = serializationDirectory.getChild(DataFileName) + dataFile.copyTo(temp) + val inputStream = temp.getByteSource.openBufferedStream() + try { + val iterator: Iterator[PersistedEmbedding] = thriftIteratorIO.fromInputStream(inputStream) + + val embeddings = iterator.map { thriftEmbedding => + embeddingInjection.invert(thriftEmbedding).get + } + + factory(metric, futurePool, embeddings) + } finally { + inputStream.close() + temp.delete() + } + } +} diff --git a/ann/src/main/scala/com/twitter/ann/brute_force/BruteForceIndex.scala b/ann/src/main/scala/com/twitter/ann/brute_force/BruteForceIndex.scala new file mode 100644 index 0000000000..d737f57b78 --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/brute_force/BruteForceIndex.scala @@ -0,0 +1,162 @@ +package com.twitter.ann.brute_force + +import com.twitter.ann.common.Appendable +import com.twitter.ann.common.Distance +import com.twitter.ann.common.EmbeddingType._ +import com.twitter.ann.common.EntityEmbedding +import com.twitter.ann.common.IndexOutputFile +import com.twitter.ann.common.Metric +import com.twitter.ann.common.NeighborWithDistance +import com.twitter.ann.common.Queryable +import com.twitter.ann.common.RuntimeParams +import com.twitter.ann.common.Serialization +import com.twitter.ann.serialization.PersistedEmbeddingInjection +import com.twitter.ann.serialization.ThriftIteratorIO +import com.twitter.ann.serialization.thriftscala.PersistedEmbedding +import com.twitter.search.common.file.AbstractFile +import com.twitter.util.Future +import com.twitter.util.FuturePool +import java.util.concurrent.ConcurrentLinkedQueue +import org.apache.beam.sdk.io.fs.ResourceId +import scala.collection.JavaConverters._ +import scala.collection.mutable + +object BruteForceRuntimeParams extends RuntimeParams + +object BruteForceIndex { + val DataFileName = "BruteForceFileData" + + def apply[T, D <: Distance[D]]( + metric: Metric[D], + futurePool: FuturePool, + initialEmbeddings: Iterator[EntityEmbedding[T]] = Iterator() + ): BruteForceIndex[T, D] = { + val linkedQueue = new ConcurrentLinkedQueue[EntityEmbedding[T]] + initialEmbeddings.foreach(embedding => linkedQueue.add(embedding)) + new BruteForceIndex(metric, futurePool, linkedQueue) + } +} + +class BruteForceIndex[T, D <: Distance[D]] private ( + metric: Metric[D], + futurePool: FuturePool, + // visible for serialization + private[brute_force] val linkedQueue: ConcurrentLinkedQueue[EntityEmbedding[T]]) + extends Appendable[T, BruteForceRuntimeParams.type, D] + with Queryable[T, BruteForceRuntimeParams.type, D] { + + override def append(embedding: EntityEmbedding[T]): Future[Unit] = { + futurePool { + linkedQueue.add(embedding) + } + } + + override def toQueryable: Queryable[T, BruteForceRuntimeParams.type, D] = this + + override def query( + embedding: EmbeddingVector, + numOfNeighbours: Int, + runtimeParams: BruteForceRuntimeParams.type + ): Future[List[T]] = { + queryWithDistance(embedding, numOfNeighbours, runtimeParams).map { neighborsWithDistance => + neighborsWithDistance.map(_.neighbor) + } + } + + override def queryWithDistance( + embedding: EmbeddingVector, + numOfNeighbours: Int, + runtimeParams: BruteForceRuntimeParams.type + ): Future[List[NeighborWithDistance[T, D]]] = { + futurePool { + // Use the reverse ordering so that we can call dequeue to remove the largest element. + val ordering = Ordering.by[NeighborWithDistance[T, D], D](_.distance) + val priorityQueue = + new mutable.PriorityQueue[NeighborWithDistance[T, D]]()(ordering) + linkedQueue + .iterator() + .asScala + .foreach { entity => + val neighborWithDistance = + NeighborWithDistance(entity.id, metric.distance(entity.embedding, embedding)) + priorityQueue.+=(neighborWithDistance) + if (priorityQueue.size > numOfNeighbours) { + priorityQueue.dequeue() + } + } + val reverseList: List[NeighborWithDistance[T, D]] = + priorityQueue.dequeueAll + reverseList.reverse + } + } +} + +object SerializableBruteForceIndex { + def apply[T, D <: Distance[D]]( + metric: Metric[D], + futurePool: FuturePool, + embeddingInjection: PersistedEmbeddingInjection[T], + thriftIteratorIO: ThriftIteratorIO[PersistedEmbedding] + ): SerializableBruteForceIndex[T, D] = { + val bruteForceIndex = BruteForceIndex[T, D](metric, futurePool) + + new SerializableBruteForceIndex(bruteForceIndex, embeddingInjection, thriftIteratorIO) + } +} + +/** + * This is a class that wrapps a BruteForceIndex and provides a method for serialization. + * + * @param bruteForceIndex all queries and updates are sent to this index. + * @param embeddingInjection injection that can convert embeddings to thrift embeddings. + * @param thriftIteratorIO class that provides a way to write PersistedEmbeddings to disk + */ +class SerializableBruteForceIndex[T, D <: Distance[D]]( + bruteForceIndex: BruteForceIndex[T, D], + embeddingInjection: PersistedEmbeddingInjection[T], + thriftIteratorIO: ThriftIteratorIO[PersistedEmbedding]) + extends Appendable[T, BruteForceRuntimeParams.type, D] + with Queryable[T, BruteForceRuntimeParams.type, D] + with Serialization { + import BruteForceIndex._ + + override def append(entity: EntityEmbedding[T]): Future[Unit] = + bruteForceIndex.append(entity) + + override def toQueryable: Queryable[T, BruteForceRuntimeParams.type, D] = this + + override def query( + embedding: EmbeddingVector, + numOfNeighbours: Int, + runtimeParams: BruteForceRuntimeParams.type + ): Future[List[T]] = + bruteForceIndex.query(embedding, numOfNeighbours, runtimeParams) + + override def queryWithDistance( + embedding: EmbeddingVector, + numOfNeighbours: Int, + runtimeParams: BruteForceRuntimeParams.type + ): Future[List[NeighborWithDistance[T, D]]] = + bruteForceIndex.queryWithDistance(embedding, numOfNeighbours, runtimeParams) + + override def toDirectory(serializationDirectory: ResourceId): Unit = { + toDirectory(new IndexOutputFile(serializationDirectory)) + } + + override def toDirectory(serializationDirectory: AbstractFile): Unit = { + toDirectory(new IndexOutputFile(serializationDirectory)) + } + + private def toDirectory(serializationDirectory: IndexOutputFile): Unit = { + val outputStream = serializationDirectory.createFile(DataFileName).getOutputStream() + val thriftEmbeddings = + bruteForceIndex.linkedQueue.iterator().asScala.map { embedding => + embeddingInjection(embedding) + } + try { + thriftIteratorIO.toOutputStream(thriftEmbeddings, outputStream) + } finally { + outputStream.close() + } + } +} diff --git a/ann/src/main/scala/com/twitter/ann/common/AnnInjections.scala b/ann/src/main/scala/com/twitter/ann/common/AnnInjections.scala new file mode 100644 index 0000000000..44ecbb49e8 --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/common/AnnInjections.scala @@ -0,0 +1,28 @@ +package com.twitter.ann.common + +import com.twitter.bijection.{Bijection, Injection} + +// Class providing commonly used injections that can be used directly with ANN apis. +// Injection prefixed with `J` can be used in java directly with ANN apis. +object AnnInjections { + val LongInjection: Injection[Long, Array[Byte]] = Injection.long2BigEndian + + def StringInjection: Injection[String, Array[Byte]] = Injection.utf8 + + def IntInjection: Injection[Int, Array[Byte]] = Injection.int2BigEndian + + val JLongInjection: Injection[java.lang.Long, Array[Byte]] = + Bijection.long2Boxed + .asInstanceOf[Bijection[Long, java.lang.Long]] + .inverse + .andThen(LongInjection) + + val JStringInjection: Injection[java.lang.String, Array[Byte]] = + StringInjection + + val JIntInjection: Injection[java.lang.Integer, Array[Byte]] = + Bijection.int2Boxed + .asInstanceOf[Bijection[Int, java.lang.Integer]] + .inverse + .andThen(IntInjection) +} diff --git a/ann/src/main/scala/com/twitter/ann/common/Api.scala b/ann/src/main/scala/com/twitter/ann/common/Api.scala new file mode 100644 index 0000000000..5873c1bbbb --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/common/Api.scala @@ -0,0 +1,150 @@ +package com.twitter.ann.common + +import com.twitter.ann.common.EmbeddingType.EmbeddingVector +import com.twitter.ml.api.embedding.Embedding +import com.twitter.ml.api.embedding.EmbeddingMath +import com.twitter.ml.api.embedding.EmbeddingSerDe +import com.twitter.util.Future + +object EmbeddingType { + type EmbeddingVector = Embedding[Float] + val embeddingSerDe = EmbeddingSerDe.apply[Float] + private[common] val math = EmbeddingMath.Float +} + +/** + * Typed entity with an embedding associated with it. + * @param id : Unique Id for an entity. + * @param embedding : Embedding/Vector of an entity. + * @tparam T: Type of id. + */ +case class EntityEmbedding[T](id: T, embedding: EmbeddingVector) + +// Query interface for ANN +trait Queryable[T, P <: RuntimeParams, D <: Distance[D]] { + + /** + * ANN query for ids. + * @param embedding: Embedding/Vector to be queried with. + * @param numOfNeighbors: Number of neighbours to be queried for. + * @param runtimeParams: Runtime params associated with index to control accuracy/latency etc. + * @return List of approximate nearest neighbour ids. + */ + def query( + embedding: EmbeddingVector, + numOfNeighbors: Int, + runtimeParams: P + ): Future[List[T]] + + /** + * ANN query for ids with distance. + * @param embedding: Embedding/Vector to be queried with. + * @param numOfNeighbors: Number of neighbours to be queried for. + * @param runtimeParams: Runtime params associated with index to control accuracy/latency etc. + * @return List of approximate nearest neighbour ids with distance from the query embedding. + */ + def queryWithDistance( + embedding: EmbeddingVector, + numOfNeighbors: Int, + runtimeParams: P + ): Future[List[NeighborWithDistance[T, D]]] +} + +// Query interface for ANN over indexes that are grouped +trait QueryableGrouped[T, P <: RuntimeParams, D <: Distance[D]] extends Queryable[T, P, D] { + + /** + * ANN query for ids. + * @param embedding: Embedding/Vector to be queried with. + * @param numOfNeighbors: Number of neighbours to be queried for. + * @param runtimeParams: Runtime params associated with index to control accuracy/latency etc. + * @param key: Optional key to lookup specific ANN index and perform query there + * @return List of approximate nearest neighbour ids. + */ + def query( + embedding: EmbeddingVector, + numOfNeighbors: Int, + runtimeParams: P, + key: Option[String] + ): Future[List[T]] + + /** + * ANN query for ids with distance. + * @param embedding: Embedding/Vector to be queried with. + * @param numOfNeighbors: Number of neighbours to be queried for. + * @param runtimeParams: Runtime params associated with index to control accuracy/latency etc. + * @param key: Optional key to lookup specific ANN index and perform query there + * @return List of approximate nearest neighbour ids with distance from the query embedding. + */ + def queryWithDistance( + embedding: EmbeddingVector, + numOfNeighbors: Int, + runtimeParams: P, + key: Option[String] + ): Future[List[NeighborWithDistance[T, D]]] +} + +/** + * Runtime params associated with index to control accuracy/latency etc while querying. + */ +trait RuntimeParams {} + +/** + * ANN query result with distance. + * @param neighbor : Id of the neighbours + * @param distance: Distance of neighbour from query ex: D: CosineDistance, L2Distance, InnerProductDistance + */ +case class NeighborWithDistance[T, D <: Distance[D]](neighbor: T, distance: D) + +/** + * ANN query result with seed entity for which this neighbor was provided. + * @param seed: Seed Id for which ann query was called + * @param neighbor : Id of the neighbours + */ +case class NeighborWithSeed[T1, T2](seed: T1, neighbor: T2) + +/** + * ANN query result with distance with seed entity for which this neighbor was provided. + * @param seed: Seed Id for which ann query was called + * @param neighbor : Id of the neighbours + * @param distance: Distance of neighbour from query ex: D: CosineDistance, L2Distance, InnerProductDistance + */ +case class NeighborWithDistanceWithSeed[T1, T2, D <: Distance[D]]( + seed: T1, + neighbor: T2, + distance: D) + +trait RawAppendable[P <: RuntimeParams, D <: Distance[D]] { + + /** + * Append an embedding in an index. + * @param embedding: Embedding/Vector + * @return Future of long id associated with embedding autogenerated. + */ + def append(embedding: EmbeddingVector): Future[Long] + + /** + * Convert an Appendable to Queryable interface to query an index. + */ + def toQueryable: Queryable[Long, P, D] +} + +// Index building interface for ANN. +trait Appendable[T, P <: RuntimeParams, D <: Distance[D]] { + + /** + * Append an entity with embedding in an index. + * @param entity: Entity with its embedding + */ + def append(entity: EntityEmbedding[T]): Future[Unit] + + /** + * Convert an Appendable to Queryable interface to query an index. + */ + def toQueryable: Queryable[T, P, D] +} + +// Updatable index interface for ANN. +trait Updatable[T] { + def update(entity: EntityEmbedding[T]): Future[Unit] +} diff --git a/ann/src/main/scala/com/twitter/ann/common/BUILD b/ann/src/main/scala/com/twitter/ann/common/BUILD new file mode 100644 index 0000000000..49e1a987e2 --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/common/BUILD @@ -0,0 +1,21 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/com/google/guava", + "3rdparty/jvm/com/twitter/bijection:core", + "3rdparty/jvm/com/twitter/storehaus:core", + "3rdparty/jvm/org/apache/beam:beam-sdks-java-io-google-cloud-platform", + "ann/src/main/thrift/com/twitter/ann/common:ann-common-scala", + "finatra/inject/inject-mdc/src/main/scala", + "mediaservices/commons/src/main/scala:futuretracker", + "src/java/com/twitter/search/common/file", + "src/scala/com/twitter/ml/api/embedding", + "stitch/stitch-core", + ], + exports = [ + "3rdparty/jvm/com/twitter/bijection:core", + ], +) diff --git a/ann/src/main/scala/com/twitter/ann/common/EmbeddingProducer.scala b/ann/src/main/scala/com/twitter/ann/common/EmbeddingProducer.scala new file mode 100644 index 0000000000..1b99ed9c47 --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/common/EmbeddingProducer.scala @@ -0,0 +1,13 @@ +package com.twitter.ann.common + +import com.twitter.stitch.Stitch + +trait EmbeddingProducer[T] { + + /** + * Produce an embedding from type T. Implementations of this could do a lookup from an id to an + * embedding. Or they could run a deep model on features that output and embedding. + * @return An embedding Stitch. See go/stitch for details on how to use the Stitch API. + */ + def produceEmbedding(input: T): Stitch[Option[EmbeddingType.EmbeddingVector]] +} diff --git a/ann/src/main/scala/com/twitter/ann/common/IndexOutputFile.scala b/ann/src/main/scala/com/twitter/ann/common/IndexOutputFile.scala new file mode 100644 index 0000000000..0e71a317ae --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/common/IndexOutputFile.scala @@ -0,0 +1,226 @@ +package com.twitter.ann.common + +import com.google.common.io.ByteStreams +import com.twitter.ann.common.thriftscala.AnnIndexMetadata +import com.twitter.mediaservices.commons.codec.ArrayByteBufferCodec +import com.twitter.mediaservices.commons.codec.ThriftByteBufferCodec +import com.twitter.search.common.file.AbstractFile +import java.io.IOException +import java.io.InputStream +import java.io.OutputStream +import java.nio.channels.Channels +import org.apache.beam.sdk.io.FileSystems +import org.apache.beam.sdk.io.fs.MoveOptions +import org.apache.beam.sdk.io.fs.ResolveOptions +import org.apache.beam.sdk.io.fs.ResolveOptions.StandardResolveOptions +import org.apache.beam.sdk.io.fs.ResourceId +import org.apache.beam.sdk.util.MimeTypes +import org.apache.hadoop.io.IOUtils +import scala.collection.JavaConverters._ + +/** + * This class creates a wrapper around GCS filesystem and HDFS filesystem for the index + * generation job. It implements the basic methods required by the index generation job and hides + * the logic around handling HDFS vs GCS. + */ +class IndexOutputFile(val abstractFile: AbstractFile, val resourceId: ResourceId) { + + // Success file name + private val SUCCESS_FILE = "_SUCCESS" + private val INDEX_METADATA_FILE = "ANN_INDEX_METADATA" + private val MetadataCodec = new ThriftByteBufferCodec[AnnIndexMetadata](AnnIndexMetadata) + + /** + * Constructor for ResourceId. This is used for GCS filesystem + * @param resourceId + */ + def this(resourceId: ResourceId) = { + this(null, resourceId) + } + + /** + * Constructor for AbstractFile. This is used for HDFS and local filesystem + * @param abstractFile + */ + def this(abstractFile: AbstractFile) = { + this(abstractFile, null) + } + + /** + * Returns true if this instance is around an AbstractFile. + * @return + */ + def isAbstractFile(): Boolean = { + abstractFile != null + } + + /** + * Creates a _SUCCESS file in the current directory. + */ + def createSuccessFile(): Unit = { + if (isAbstractFile()) { + abstractFile.createSuccessFile() + } else { + val successFile = + resourceId.resolve(SUCCESS_FILE, ResolveOptions.StandardResolveOptions.RESOLVE_FILE) + val successWriterChannel = FileSystems.create(successFile, MimeTypes.BINARY) + successWriterChannel.close() + } + } + + /** + * Returns whether the current instance represents a directory + * @return True if the current instance is a directory + */ + def isDirectory(): Boolean = { + if (isAbstractFile()) { + abstractFile.isDirectory + } else { + resourceId.isDirectory + } + } + + /** + * Return the current path of the file represented by the current instance + * @return The path string of the file/directory + */ + def getPath(): String = { + if (isAbstractFile()) { + abstractFile.getPath.toString + } else { + if (resourceId.isDirectory) { + resourceId.getCurrentDirectory.toString + } else { + resourceId.getCurrentDirectory.toString + resourceId.getFilename + } + } + } + + /** + * Creates a new file @param fileName in the current directory. + * @param fileName + * @return A new file inside the current directory + */ + def createFile(fileName: String): IndexOutputFile = { + if (isAbstractFile()) { + // AbstractFile treats files and directories the same way. Hence, not checking for directory + // here. + new IndexOutputFile(abstractFile.getChild(fileName)) + } else { + if (!resourceId.isDirectory) { + // If this is not a directory, throw exception. + throw new IllegalArgumentException(getPath() + " is not a directory.") + } + new IndexOutputFile( + resourceId.resolve(fileName, ResolveOptions.StandardResolveOptions.RESOLVE_FILE)) + } + } + + /** + * Creates a new directory @param directoryName in the current directory. + * @param directoryName + * @return A new directory inside the current directory + */ + def createDirectory(directoryName: String): IndexOutputFile = { + if (isAbstractFile()) { + // AbstractFile treats files and directories the same way. Hence, not checking for directory + // here. + val dir = abstractFile.getChild(directoryName) + dir.mkdirs() + new IndexOutputFile(dir) + } else { + if (!resourceId.isDirectory) { + // If this is not a directory, throw exception. + throw new IllegalArgumentException(getPath() + " is not a directory.") + } + val newResourceId = + resourceId.resolve(directoryName, ResolveOptions.StandardResolveOptions.RESOLVE_DIRECTORY) + + // Create a tmp file and delete in order to trigger directory creation + val tmpFile = + newResourceId.resolve("tmp", ResolveOptions.StandardResolveOptions.RESOLVE_FILE) + val tmpWriterChannel = FileSystems.create(tmpFile, MimeTypes.BINARY) + tmpWriterChannel.close() + FileSystems.delete(List(tmpFile).asJava, MoveOptions.StandardMoveOptions.IGNORE_MISSING_FILES) + + new IndexOutputFile(newResourceId) + } + } + + def getChild(fileName: String, isDirectory: Boolean = false): IndexOutputFile = { + if (isAbstractFile()) { + new IndexOutputFile(abstractFile.getChild(fileName)) + } else { + val resolveOption = if (isDirectory) { + StandardResolveOptions.RESOLVE_DIRECTORY + } else { + StandardResolveOptions.RESOLVE_FILE + } + new IndexOutputFile(resourceId.resolve(fileName, resolveOption)) + } + } + + /** + * Returns an OutputStream for the underlying file. + * Note: Close the OutputStream after writing + * @return + */ + def getOutputStream(): OutputStream = { + if (isAbstractFile()) { + abstractFile.getByteSink.openStream() + } else { + if (resourceId.isDirectory) { + // If this is a directory, throw exception. + throw new IllegalArgumentException(getPath() + " is a directory.") + } + val writerChannel = FileSystems.create(resourceId, MimeTypes.BINARY) + Channels.newOutputStream(writerChannel) + } + } + + /** + * Returns an InputStream for the underlying file. + * Note: Close the InputStream after reading + * @return + */ + def getInputStream(): InputStream = { + if (isAbstractFile()) { + abstractFile.getByteSource.openStream() + } else { + if (resourceId.isDirectory) { + // If this is a directory, throw exception. + throw new IllegalArgumentException(getPath() + " is a directory.") + } + val readChannel = FileSystems.open(resourceId) + Channels.newInputStream(readChannel) + } + } + + /** + * Copies content from the srcIn into the current file. + * @param srcIn + */ + def copyFrom(srcIn: InputStream): Unit = { + val out = getOutputStream() + try { + IOUtils.copyBytes(srcIn, out, 4096) + out.close() + } catch { + case ex: IOException => + IOUtils.closeStream(out); + throw ex; + } + } + + def writeIndexMetadata(annIndexMetadata: AnnIndexMetadata): Unit = { + val out = createFile(INDEX_METADATA_FILE).getOutputStream() + val bytes = ArrayByteBufferCodec.decode(MetadataCodec.encode(annIndexMetadata)) + out.write(bytes) + out.close() + } + + def loadIndexMetadata(): AnnIndexMetadata = { + val in = ByteStreams.toByteArray(getInputStream()) + MetadataCodec.decode(ArrayByteBufferCodec.encode(in)) + } +} diff --git a/ann/src/main/scala/com/twitter/ann/common/IndexTransformer.scala b/ann/src/main/scala/com/twitter/ann/common/IndexTransformer.scala new file mode 100644 index 0000000000..b096e2a579 --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/common/IndexTransformer.scala @@ -0,0 +1,118 @@ +package com.twitter.ann.common + +import com.twitter.ann.common.EmbeddingType.EmbeddingVector +import com.twitter.storehaus.{ReadableStore, Store} +import com.twitter.util.Future + +// Utility to transform raw index to typed index using Store +object IndexTransformer { + + /** + * Transform a long type queryable index to Typed queryable index + * @param index: Raw Queryable index + * @param store: Readable store to provide mappings between Long and T + * @tparam T: Type to transform to + * @tparam P: Runtime params + * @return Queryable index typed on T + */ + def transformQueryable[T, P <: RuntimeParams, D <: Distance[D]]( + index: Queryable[Long, P, D], + store: ReadableStore[Long, T] + ): Queryable[T, P, D] = { + new Queryable[T, P, D] { + override def query( + embedding: EmbeddingVector, + numOfNeighbors: Int, + runtimeParams: P + ): Future[List[T]] = { + val neighbors = index.query(embedding, numOfNeighbors, runtimeParams) + neighbors + .flatMap(nn => { + val ids = nn.map(id => store.get(id).map(_.get)) + Future + .collect(ids) + .map(_.toList) + }) + } + + override def queryWithDistance( + embedding: EmbeddingVector, + numOfNeighbors: Int, + runtimeParams: P + ): Future[List[NeighborWithDistance[T, D]]] = { + val neighbors = index.queryWithDistance(embedding, numOfNeighbors, runtimeParams) + neighbors + .flatMap(nn => { + val ids = nn.map(obj => + store.get(obj.neighbor).map(id => NeighborWithDistance(id.get, obj.distance))) + Future + .collect(ids) + .map(_.toList) + }) + } + } + } + + /** + * Transform a long type appendable index to Typed appendable index + * @param index: Raw Appendable index + * @param store: Writable store to store mappings between Long and T + * @tparam T: Type to transform to + * @return Appendable index typed on T + */ + def transformAppendable[T, P <: RuntimeParams, D <: Distance[D]]( + index: RawAppendable[P, D], + store: Store[Long, T] + ): Appendable[T, P, D] = { + new Appendable[T, P, D]() { + override def append(entity: EntityEmbedding[T]): Future[Unit] = { + index + .append(entity.embedding) + .flatMap(id => store.put((id, Some(entity.id)))) + } + + override def toQueryable: Queryable[T, P, D] = { + transformQueryable(index.toQueryable, store) + } + } + } + + /** + * Transform a long type appendable and queryable index to Typed appendable and queryable index + * @param index: Raw Appendable and queryable index + * @param store: Store to provide/store mappings between Long and T + * @tparam T: Type to transform to + * @tparam Index: Index + * @return Appendable and queryable index typed on T + */ + def transform1[ + Index <: RawAppendable[P, D] with Queryable[Long, P, D], + T, + P <: RuntimeParams, + D <: Distance[D] + ]( + index: Index, + store: Store[Long, T] + ): Queryable[T, P, D] with Appendable[T, P, D] = { + val queryable = transformQueryable(index, store) + val appendable = transformAppendable(index, store) + + new Queryable[T, P, D] with Appendable[T, P, D] { + override def query( + embedding: EmbeddingVector, + numOfNeighbors: Int, + runtimeParams: P + ) = queryable.query(embedding, numOfNeighbors, runtimeParams) + + override def queryWithDistance( + embedding: EmbeddingVector, + numOfNeighbors: Int, + runtimeParams: P + ) = queryable.queryWithDistance(embedding, numOfNeighbors, runtimeParams) + + override def append(entity: EntityEmbedding[T]) = appendable.append(entity) + + override def toQueryable: Queryable[T, P, D] = appendable.toQueryable + } + } +} diff --git a/ann/src/main/scala/com/twitter/ann/common/MemoizedInEpochs.scala b/ann/src/main/scala/com/twitter/ann/common/MemoizedInEpochs.scala new file mode 100644 index 0000000000..267aee636f --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/common/MemoizedInEpochs.scala @@ -0,0 +1,37 @@ +package com.twitter.ann.common + +import com.twitter.util.Return +import com.twitter.util.Throw +import com.twitter.util.Try +import com.twitter.util.logging.Logging + +// Memoization with a twist +// New epoch reuse K:V pairs from previous and recycle everything else +class MemoizedInEpochs[K, V](f: K => Try[V]) extends Logging { + private var memoizedCalls: Map[K, V] = Map.empty + + def epoch(keys: Seq[K]): Seq[V] = { + val newSet = keys.toSet + val keysToBeComputed = newSet.diff(memoizedCalls.keySet) + val computedKeysAndValues = keysToBeComputed.map { key => + info(s"Memoize ${key}") + (key, f(key)) + } + val keysAndValuesAfterFilteringFailures = computedKeysAndValues + .flatMap { + case (key, Return(value)) => Some((key, value)) + case (key, Throw(e)) => + warn(s"Calling f for ${key} has failed", e) + + None + } + val keysReusedFromLastEpoch = memoizedCalls.filterKeys(newSet.contains) + memoizedCalls = keysReusedFromLastEpoch ++ keysAndValuesAfterFilteringFailures + + debug(s"Final memoization is ${memoizedCalls.keys.mkString(", ")}") + + keys.flatMap(memoizedCalls.get) + } + + def currentEpochKeys: Set[K] = memoizedCalls.keySet +} diff --git a/ann/src/main/scala/com/twitter/ann/common/Metric.scala b/ann/src/main/scala/com/twitter/ann/common/Metric.scala new file mode 100644 index 0000000000..1c2c95aa9c --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/common/Metric.scala @@ -0,0 +1,290 @@ +package com.twitter.ann.common + +import com.google.common.collect.ImmutableBiMap +import com.twitter.ann.common.EmbeddingType._ +import com.twitter.ann.common.thriftscala.DistanceMetric +import com.twitter.ann.common.thriftscala.{CosineDistance => ServiceCosineDistance} +import com.twitter.ann.common.thriftscala.{Distance => ServiceDistance} +import com.twitter.ann.common.thriftscala.{InnerProductDistance => ServiceInnerProductDistance} +import com.twitter.ann.common.thriftscala.{EditDistance => ServiceEditDistance} +import com.twitter.ann.common.thriftscala.{L2Distance => ServiceL2Distance} +import com.twitter.bijection.Injection +import scala.util.Failure +import scala.util.Success +import scala.util.Try + +// Ann distance metrics +trait Distance[D] extends Any with Ordered[D] { + def distance: Float +} + +case class L2Distance(distance: Float) extends AnyVal with Distance[L2Distance] { + override def compare(that: L2Distance): Int = + Ordering.Float.compare(this.distance, that.distance) +} + +case class CosineDistance(distance: Float) extends AnyVal with Distance[CosineDistance] { + override def compare(that: CosineDistance): Int = + Ordering.Float.compare(this.distance, that.distance) +} + +case class InnerProductDistance(distance: Float) + extends AnyVal + with Distance[InnerProductDistance] { + override def compare(that: InnerProductDistance): Int = + Ordering.Float.compare(this.distance, that.distance) +} + +case class EditDistance(distance: Float) extends AnyVal with Distance[EditDistance] { + override def compare(that: EditDistance): Int = + Ordering.Float.compare(this.distance, that.distance) +} + +object Metric { + private[this] val thriftMetricMapping = ImmutableBiMap.of( + L2, + DistanceMetric.L2, + Cosine, + DistanceMetric.Cosine, + InnerProduct, + DistanceMetric.InnerProduct, + Edit, + DistanceMetric.EditDistance + ) + + def fromThrift(metric: DistanceMetric): Metric[_ <: Distance[_]] = { + thriftMetricMapping.inverse().get(metric) + } + + def toThrift(metric: Metric[_ <: Distance[_]]): DistanceMetric = { + thriftMetricMapping.get(metric) + } + + def fromString(metricName: String): Metric[_ <: Distance[_]] + with Injection[_, ServiceDistance] = { + metricName match { + case "Cosine" => Cosine + case "L2" => L2 + case "InnerProduct" => InnerProduct + case "EditDistance" => Edit + case _ => + throw new IllegalArgumentException(s"No Metric with the name $metricName") + } + } +} + +sealed trait Metric[D <: Distance[D]] { + def distance( + embedding1: EmbeddingVector, + embedding2: EmbeddingVector + ): D + def absoluteDistance( + embedding1: EmbeddingVector, + embedding2: EmbeddingVector + ): Float + def fromAbsoluteDistance(distance: Float): D +} + +case object L2 extends Metric[L2Distance] with Injection[L2Distance, ServiceDistance] { + override def distance( + embedding1: EmbeddingVector, + embedding2: EmbeddingVector + ): L2Distance = { + fromAbsoluteDistance(MetricUtil.l2distance(embedding1, embedding2).toFloat) + } + + override def fromAbsoluteDistance(distance: Float): L2Distance = { + L2Distance(distance) + } + + override def absoluteDistance( + embedding1: EmbeddingVector, + embedding2: EmbeddingVector + ): Float = distance(embedding1, embedding2).distance + + override def apply(scalaDistance: L2Distance): ServiceDistance = { + ServiceDistance.L2Distance(ServiceL2Distance(scalaDistance.distance)) + } + + override def invert(serviceDistance: ServiceDistance): Try[L2Distance] = { + serviceDistance match { + case ServiceDistance.L2Distance(l2Distance) => + Success(L2Distance(l2Distance.distance.toFloat)) + case distance => + Failure(new IllegalArgumentException(s"Expected an l2 distance but got $distance")) + } + } +} + +case object Cosine extends Metric[CosineDistance] with Injection[CosineDistance, ServiceDistance] { + override def distance( + embedding1: EmbeddingVector, + embedding2: EmbeddingVector + ): CosineDistance = { + fromAbsoluteDistance(1 - MetricUtil.cosineSimilarity(embedding1, embedding2)) + } + + override def fromAbsoluteDistance(distance: Float): CosineDistance = { + CosineDistance(distance) + } + + override def absoluteDistance( + embedding1: EmbeddingVector, + embedding2: EmbeddingVector + ): Float = distance(embedding1, embedding2).distance + + override def apply(scalaDistance: CosineDistance): ServiceDistance = { + ServiceDistance.CosineDistance(ServiceCosineDistance(scalaDistance.distance)) + } + + override def invert(serviceDistance: ServiceDistance): Try[CosineDistance] = { + serviceDistance match { + case ServiceDistance.CosineDistance(cosineDistance) => + Success(CosineDistance(cosineDistance.distance.toFloat)) + case distance => + Failure(new IllegalArgumentException(s"Expected a cosine distance but got $distance")) + } + } +} + +case object InnerProduct + extends Metric[InnerProductDistance] + with Injection[InnerProductDistance, ServiceDistance] { + override def distance( + embedding1: EmbeddingVector, + embedding2: EmbeddingVector + ): InnerProductDistance = { + fromAbsoluteDistance(1 - MetricUtil.dot(embedding1, embedding2)) + } + + override def fromAbsoluteDistance(distance: Float): InnerProductDistance = { + InnerProductDistance(distance) + } + + override def absoluteDistance( + embedding1: EmbeddingVector, + embedding2: EmbeddingVector + ): Float = distance(embedding1, embedding2).distance + + override def apply(scalaDistance: InnerProductDistance): ServiceDistance = { + ServiceDistance.InnerProductDistance(ServiceInnerProductDistance(scalaDistance.distance)) + } + + override def invert( + serviceDistance: ServiceDistance + ): Try[InnerProductDistance] = { + serviceDistance match { + case ServiceDistance.InnerProductDistance(cosineDistance) => + Success(InnerProductDistance(cosineDistance.distance.toFloat)) + case distance => + Failure( + new IllegalArgumentException(s"Expected a inner product distance but got $distance") + ) + } + } +} + +case object Edit extends Metric[EditDistance] with Injection[EditDistance, ServiceDistance] { + + private def intDistance( + embedding1: EmbeddingVector, + embedding2: EmbeddingVector, + pos1: Int, + pos2: Int, + precomputedDistances: scala.collection.mutable.Map[(Int, Int), Int] + ): Int = { + // return the remaining characters of other String + if (pos1 == 0) return pos2 + if (pos2 == 0) return pos1 + + // To check if the recursive tree + // for given n & m has already been executed + precomputedDistances.getOrElse( + (pos1, pos2), { + // We might want to change this so that capitals are considered the same. + // Also maybe some characters that look similar should also be the same. + val computed = if (embedding1(pos1 - 1) == embedding2(pos2 - 1)) { + intDistance(embedding1, embedding2, pos1 - 1, pos2 - 1, precomputedDistances) + } else { // If characters are nt equal, we need to + // find the minimum cost out of all 3 operations. + val insert = intDistance(embedding1, embedding2, pos1, pos2 - 1, precomputedDistances) + val del = intDistance(embedding1, embedding2, pos1 - 1, pos2, precomputedDistances) + val replace = + intDistance(embedding1, embedding2, pos1 - 1, pos2 - 1, precomputedDistances) + 1 + Math.min(insert, Math.min(del, replace)) + } + precomputedDistances.put((pos1, pos2), computed) + computed + } + ) + } + + override def distance( + embedding1: EmbeddingVector, + embedding2: EmbeddingVector + ): EditDistance = { + val editDistance = intDistance( + embedding1, + embedding2, + embedding1.length, + embedding2.length, + scala.collection.mutable.Map[(Int, Int), Int]() + ) + EditDistance(editDistance) + } + + override def fromAbsoluteDistance(distance: Float): EditDistance = { + EditDistance(distance.toInt) + } + + override def absoluteDistance( + embedding1: EmbeddingVector, + embedding2: EmbeddingVector + ): Float = distance(embedding1, embedding2).distance + + override def apply(scalaDistance: EditDistance): ServiceDistance = { + ServiceDistance.EditDistance(ServiceEditDistance(scalaDistance.distance.toInt)) + } + + override def invert( + serviceDistance: ServiceDistance + ): Try[EditDistance] = { + serviceDistance match { + case ServiceDistance.EditDistance(cosineDistance) => + Success(EditDistance(cosineDistance.distance.toFloat)) + case distance => + Failure( + new IllegalArgumentException(s"Expected a inner product distance but got $distance") + ) + } + } +} + +object MetricUtil { + private[ann] def dot( + embedding1: EmbeddingVector, + embedding2: EmbeddingVector + ): Float = { + math.dotProduct(embedding1, embedding2) + } + + private[ann] def l2distance( + embedding1: EmbeddingVector, + embedding2: EmbeddingVector + ): Double = { + math.l2Distance(embedding1, embedding2) + } + + private[ann] def cosineSimilarity( + embedding1: EmbeddingVector, + embedding2: EmbeddingVector + ): Float = { + math.cosineSimilarity(embedding1, embedding2).toFloat + } + + private[ann] def norm( + embedding: EmbeddingVector + ): EmbeddingVector = { + math.normalize(embedding) + } +} diff --git a/ann/src/main/scala/com/twitter/ann/common/QueryableById.scala b/ann/src/main/scala/com/twitter/ann/common/QueryableById.scala new file mode 100644 index 0000000000..0f1c8cfe21 --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/common/QueryableById.scala @@ -0,0 +1,41 @@ +package com.twitter.ann.common + +import com.twitter.stitch.Stitch + +/** + * This is a trait that allows you to query for nearest neighbors given an arbitrary type T1. This is + * in contrast to a regular com.twitter.ann.common.Appendable, which takes an embedding as the input + * argument. + * + * This interface uses the Stitch API for batching. See go/stitch for details on how to use it. + * + * @tparam T1 type of the query. + * @tparam T2 type of the result. + * @tparam P runtime parameters supported by the index. + * @tparam D distance function used in the index. + */ +trait QueryableById[T1, T2, P <: RuntimeParams, D <: Distance[D]] { + def queryById( + id: T1, + numOfNeighbors: Int, + runtimeParams: P + ): Stitch[List[T2]] + + def queryByIdWithDistance( + id: T1, + numOfNeighbors: Int, + runtimeParams: P + ): Stitch[List[NeighborWithDistance[T2, D]]] + + def batchQueryById( + ids: Seq[T1], + numOfNeighbors: Int, + runtimeParams: P + ): Stitch[List[NeighborWithSeed[T1, T2]]] + + def batchQueryWithDistanceById( + ids: Seq[T1], + numOfNeighbors: Int, + runtimeParams: P + ): Stitch[List[NeighborWithDistanceWithSeed[T1, T2, D]]] +} diff --git a/ann/src/main/scala/com/twitter/ann/common/QueryableByIdImplementation.scala b/ann/src/main/scala/com/twitter/ann/common/QueryableByIdImplementation.scala new file mode 100644 index 0000000000..479414eb96 --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/common/QueryableByIdImplementation.scala @@ -0,0 +1,91 @@ +package com.twitter.ann.common + +import com.twitter.stitch.Stitch + +/** + * Implementation of QueryableById that composes an EmbeddingProducer and a Queryable so that we + * can get nearest neighbors given an id of type T1 + * @param embeddingProducer provides an embedding given an id. + * @param queryable provides a list of neighbors given an embedding. + * @tparam T1 type of the query. + * @tparam T2 type of the result. + * @tparam P runtime parameters supported by the index. + * @tparam D distance function used in the index. + */ +class QueryableByIdImplementation[T1, T2, P <: RuntimeParams, D <: Distance[D]]( + embeddingProducer: EmbeddingProducer[T1], + queryable: Queryable[T2, P, D]) + extends QueryableById[T1, T2, P, D] { + override def queryById( + id: T1, + numOfNeighbors: Int, + runtimeParams: P + ): Stitch[List[T2]] = { + embeddingProducer.produceEmbedding(id).flatMap { embeddingOption => + embeddingOption + .map { embedding => + Stitch.callFuture(queryable.query(embedding, numOfNeighbors, runtimeParams)) + }.getOrElse { + Stitch.value(List.empty) + } + } + } + + override def queryByIdWithDistance( + id: T1, + numOfNeighbors: Int, + runtimeParams: P + ): Stitch[List[NeighborWithDistance[T2, D]]] = { + embeddingProducer.produceEmbedding(id).flatMap { embeddingOption => + embeddingOption + .map { embedding => + Stitch.callFuture(queryable.queryWithDistance(embedding, numOfNeighbors, runtimeParams)) + }.getOrElse { + Stitch.value(List.empty) + } + } + } + + override def batchQueryById( + ids: Seq[T1], + numOfNeighbors: Int, + runtimeParams: P + ): Stitch[List[NeighborWithSeed[T1, T2]]] = { + Stitch + .traverse(ids) { id => + embeddingProducer.produceEmbedding(id).flatMap { embeddingOption => + embeddingOption + .map { embedding => + Stitch + .callFuture(queryable.query(embedding, numOfNeighbors, runtimeParams)).map( + _.map(neighbor => NeighborWithSeed(id, neighbor))) + }.getOrElse { + Stitch.value(List.empty) + }.handle { case _ => List.empty } + } + }.map { _.toList.flatten } + } + + override def batchQueryWithDistanceById( + ids: Seq[T1], + numOfNeighbors: Int, + runtimeParams: P + ): Stitch[List[NeighborWithDistanceWithSeed[T1, T2, D]]] = { + Stitch + .traverse(ids) { id => + embeddingProducer.produceEmbedding(id).flatMap { embeddingOption => + embeddingOption + .map { embedding => + Stitch + .callFuture(queryable.queryWithDistance(embedding, numOfNeighbors, runtimeParams)) + .map(_.map(neighbor => + NeighborWithDistanceWithSeed(id, neighbor.neighbor, neighbor.distance))) + }.getOrElse { + Stitch.value(List.empty) + }.handle { case _ => List.empty } + } + }.map { + _.toList.flatten + } + } +} diff --git a/ann/src/main/scala/com/twitter/ann/common/QueryableOperations.scala b/ann/src/main/scala/com/twitter/ann/common/QueryableOperations.scala new file mode 100644 index 0000000000..9cd72b066b --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/common/QueryableOperations.scala @@ -0,0 +1,26 @@ +package com.twitter.ann.common + +import com.twitter.ann.common.EmbeddingType.EmbeddingVector +import com.twitter.util.Future + +object QueryableOperations { + implicit class Map[T, P <: RuntimeParams, D <: Distance[D]]( + val q: Queryable[T, P, D]) { + def mapRuntimeParameters(f: P => P): Queryable[T, P, D] = { + new Queryable[T, P, D] { + def query( + embedding: EmbeddingVector, + numOfNeighbors: Int, + runtimeParams: P + ): Future[List[T]] = q.query(embedding, numOfNeighbors, f(runtimeParams)) + + def queryWithDistance( + embedding: EmbeddingVector, + numOfNeighbors: Int, + runtimeParams: P + ): Future[List[NeighborWithDistance[T, D]]] = + q.queryWithDistance(embedding, numOfNeighbors, f(runtimeParams)) + } + } + } +} diff --git a/ann/src/main/scala/com/twitter/ann/common/ReadWriteFuturePool.scala b/ann/src/main/scala/com/twitter/ann/common/ReadWriteFuturePool.scala new file mode 100644 index 0000000000..5c6c2ba706 --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/common/ReadWriteFuturePool.scala @@ -0,0 +1,29 @@ +package com.twitter.ann.common +import com.google.common.annotations.VisibleForTesting +import com.twitter.util.{Future, FuturePool} + +trait ReadWriteFuturePool { + def read[T](f: => T): Future[T] + def write[T](f: => T): Future[T] +} + +object ReadWriteFuturePool { + def apply(readPool: FuturePool, writePool: FuturePool): ReadWriteFuturePool = { + new ReadWriteFuturePoolANN(readPool, writePool) + } + + def apply(commonPool: FuturePool): ReadWriteFuturePool = { + new ReadWriteFuturePoolANN(commonPool, commonPool) + } +} + +@VisibleForTesting +private[ann] class ReadWriteFuturePoolANN(readPool: FuturePool, writePool: FuturePool) + extends ReadWriteFuturePool { + def read[T](f: => T): Future[T] = { + readPool.apply(f) + } + def write[T](f: => T): Future[T] = { + writePool.apply(f) + } +} diff --git a/ann/src/main/scala/com/twitter/ann/common/Serialization.scala b/ann/src/main/scala/com/twitter/ann/common/Serialization.scala new file mode 100644 index 0000000000..155042d07b --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/common/Serialization.scala @@ -0,0 +1,28 @@ +package com.twitter.ann.common + +import com.twitter.search.common.file.AbstractFile +import org.apache.beam.sdk.io.fs.ResourceId + +/** + * Interface for writing an Appendable to a directory. + */ +trait Serialization { + def toDirectory( + serializationDirectory: AbstractFile + ): Unit + + def toDirectory( + serializationDirectory: ResourceId + ): Unit +} + +/** + * Interface for reading a Queryable from a directory + * @tparam T the id of the embeddings + * @tparam Q type of the Queryable that is deserialized. + */ +trait QueryableDeserialization[T, P <: RuntimeParams, D <: Distance[D], Q <: Queryable[T, P, D]] { + def fromDirectory( + serializationDirectory: AbstractFile + ): Q +} diff --git a/ann/src/main/scala/com/twitter/ann/common/ServiceClientQueryable.scala b/ann/src/main/scala/com/twitter/ann/common/ServiceClientQueryable.scala new file mode 100644 index 0000000000..6b4893aefa --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/common/ServiceClientQueryable.scala @@ -0,0 +1,64 @@ +package com.twitter.ann.common + +import com.twitter.ann.common.EmbeddingType._ +import com.twitter.ann.common.thriftscala.{ + NearestNeighborQuery, + NearestNeighborResult, + Distance => ServiceDistance, + RuntimeParams => ServiceRuntimeParams +} +import com.twitter.bijection.Injection +import com.twitter.finagle.Service +import com.twitter.mediaservices.commons.codec.ArrayByteBufferCodec +import com.twitter.util.Future + +class ServiceClientQueryable[T, P <: RuntimeParams, D <: Distance[D]]( + service: Service[NearestNeighborQuery, NearestNeighborResult], + runtimeParamInjection: Injection[P, ServiceRuntimeParams], + distanceInjection: Injection[D, ServiceDistance], + idInjection: Injection[T, Array[Byte]]) + extends Queryable[T, P, D] { + override def query( + embedding: EmbeddingVector, + numOfNeighbors: Int, + runtimeParams: P + ): Future[List[T]] = { + service + .apply( + NearestNeighborQuery( + embeddingSerDe.toThrift(embedding), + withDistance = false, + runtimeParamInjection(runtimeParams), + numOfNeighbors + ) + ) + .map { result => + result.nearestNeighbors.map { nearestNeighbor => + idInjection.invert(ArrayByteBufferCodec.decode(nearestNeighbor.id)).get + }.toList + } + } + + override def queryWithDistance( + embedding: EmbeddingVector, + numOfNeighbors: Int, + runtimeParams: P + ): Future[List[NeighborWithDistance[T, D]]] = + service + .apply( + NearestNeighborQuery( + embeddingSerDe.toThrift(embedding), + withDistance = true, + runtimeParamInjection(runtimeParams), + numOfNeighbors + ) + ) + .map { result => + result.nearestNeighbors.map { nearestNeighbor => + NeighborWithDistance( + idInjection.invert(ArrayByteBufferCodec.decode(nearestNeighbor.id)).get, + distanceInjection.invert(nearestNeighbor.distance.get).get + ) + }.toList + } +} diff --git a/ann/src/main/scala/com/twitter/ann/common/ShardApi.scala b/ann/src/main/scala/com/twitter/ann/common/ShardApi.scala new file mode 100644 index 0000000000..9351bd4188 --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/common/ShardApi.scala @@ -0,0 +1,87 @@ +package com.twitter.ann.common + +import com.twitter.ann.common.EmbeddingType.EmbeddingVector +import com.twitter.util.Future +import scala.util.Random + +trait ShardFunction[T] { + + /** + * Shard function to shard embedding based on total shards and embedding data. + * @param shards + * @param entity + * @return Shard index, from 0(Inclusive) to shards(Exclusive)) + */ + def apply(shards: Int, entity: EntityEmbedding[T]): Int +} + +/** + * Randomly shards the embeddings based on number of total shards. + */ +class RandomShardFunction[T] extends ShardFunction[T] { + def apply(shards: Int, entity: EntityEmbedding[T]): Int = { + Random.nextInt(shards) + } +} + +/** + * Sharded appendable to shard the embedding into different appendable indices + * @param indices: Sequence of appendable indices + * @param shardFn: Shard function to shard data into different indices + * @param shards: Total shards + * @tparam T: Type of id. + */ +class ShardedAppendable[T, P <: RuntimeParams, D <: Distance[D]]( + indices: Seq[Appendable[T, P, D]], + shardFn: ShardFunction[T], + shards: Int) + extends Appendable[T, P, D] { + override def append(entity: EntityEmbedding[T]): Future[Unit] = { + val shard = shardFn(shards, entity) + val index = indices(shard) + index.append(entity) + } + + override def toQueryable: Queryable[T, P, D] = { + new ComposedQueryable[T, P, D](indices.map(_.toQueryable)) + } +} + +/** + * Composition of sequence of queryable indices, it queries all the indices, + * and merges the result in memory to return the K nearest neighbours + * @param indices: Sequence of queryable indices + * @tparam T: Type of id + * @tparam P: Type of runtime param + * @tparam D: Type of distance metric + */ +class ComposedQueryable[T, P <: RuntimeParams, D <: Distance[D]]( + indices: Seq[Queryable[T, P, D]]) + extends Queryable[T, P, D] { + private[this] val ordering = + Ordering.by[NeighborWithDistance[T, D], D](_.distance) + override def query( + embedding: EmbeddingVector, + numOfNeighbors: Int, + runtimeParams: P + ): Future[List[T]] = { + val neighbours = queryWithDistance(embedding, numOfNeighbors, runtimeParams) + neighbours.map(list => list.map(nn => nn.neighbor)) + } + + override def queryWithDistance( + embedding: EmbeddingVector, + numOfNeighbors: Int, + runtimeParams: P + ): Future[List[NeighborWithDistance[T, D]]] = { + val futures = Future.collect( + indices.map(index => index.queryWithDistance(embedding, numOfNeighbors, runtimeParams)) + ) + futures.map { list => + list.flatten + .sorted(ordering) + .take(numOfNeighbors) + .toList + } + } +} diff --git a/ann/src/main/scala/com/twitter/ann/common/ShardedSerialization.scala b/ann/src/main/scala/com/twitter/ann/common/ShardedSerialization.scala new file mode 100644 index 0000000000..9b9cda2640 --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/common/ShardedSerialization.scala @@ -0,0 +1,89 @@ +package com.twitter.ann.common + +import com.twitter.search.common.file.AbstractFile +import com.twitter.search.common.file.AbstractFile.Filter +import com.twitter.util.Future +import org.apache.beam.sdk.io.fs.ResourceId +import scala.collection.JavaConverters._ + +object ShardConstants { + val ShardPrefix = "shard_" +} + +/** + * Serialize shards to directory + * @param shards: List of shards to serialize + */ +class ShardedSerialization( + shards: Seq[Serialization]) + extends Serialization { + override def toDirectory(directory: AbstractFile): Unit = { + toDirectory(new IndexOutputFile(directory)) + } + + override def toDirectory(directory: ResourceId): Unit = { + toDirectory(new IndexOutputFile(directory)) + } + + private def toDirectory(directory: IndexOutputFile): Unit = { + shards.indices.foreach { shardId => + val shardDirectory = directory.createDirectory(ShardConstants.ShardPrefix + shardId) + val serialization = shards(shardId) + if (shardDirectory.isAbstractFile) { + serialization.toDirectory(shardDirectory.abstractFile) + } else { + serialization.toDirectory(shardDirectory.resourceId) + } + } + } +} + +/** + * Deserialize directories containing index shards data to a composed queryable + * @param deserializationFn function to deserialize a shard file to Queryable + * @tparam T the id of the embeddings + * @tparam P : Runtime params type + * @tparam D: Distance metric type + */ +class ComposedQueryableDeserialization[T, P <: RuntimeParams, D <: Distance[D]]( + deserializationFn: (AbstractFile) => Queryable[T, P, D]) + extends QueryableDeserialization[T, P, D, Queryable[T, P, D]] { + override def fromDirectory(directory: AbstractFile): Queryable[T, P, D] = { + val shardDirs = directory + .listFiles(new Filter { + override def accept(file: AbstractFile): Boolean = + file.getName.startsWith(ShardConstants.ShardPrefix) + }) + .asScala + .toList + + val indices = shardDirs + .map { shardDir => + deserializationFn(shardDir) + } + + new ComposedQueryable[T, P, D](indices) + } +} + +class ShardedIndexBuilderWithSerialization[T, P <: RuntimeParams, D <: Distance[D]]( + shardedIndex: ShardedAppendable[T, P, D], + shardedSerialization: ShardedSerialization) + extends Appendable[T, P, D] + with Serialization { + override def append(entity: EntityEmbedding[T]): Future[Unit] = { + shardedIndex.append(entity) + } + + override def toDirectory(directory: AbstractFile): Unit = { + shardedSerialization.toDirectory(directory) + } + + override def toDirectory(directory: ResourceId): Unit = { + shardedSerialization.toDirectory(directory) + } + + override def toQueryable: Queryable[T, P, D] = { + shardedIndex.toQueryable + } +} diff --git a/ann/src/main/scala/com/twitter/ann/common/Task.scala b/ann/src/main/scala/com/twitter/ann/common/Task.scala new file mode 100644 index 0000000000..eaad03a88b --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/common/Task.scala @@ -0,0 +1,121 @@ +package com.twitter.ann.common + +import com.twitter.finagle.stats.CategorizingExceptionStatsHandler +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.finagle.tracing.DefaultTracer +import com.twitter.finagle.tracing.Trace +import com.twitter.finagle.util.DefaultTimer +import com.twitter.finagle.util.Rng +import com.twitter.inject.logging.MDCKeys +import com.twitter.util.Closable +import com.twitter.util.Duration +import com.twitter.util.Future +import com.twitter.util.Time +import com.twitter.util.Timer +import com.twitter.util.logging.Logging +import java.util.concurrent.atomic.AtomicInteger +import org.slf4j.MDC + +/** + * A Task that will be scheduled to execute periodically on every interval. If a task takes + * longer than an interval to complete, it will be immediately scheduled to run. + */ +trait Task extends Closable { self: Logging => + + // Exposed if the implementation of `task` need to report failures + val exnStatsHandler = new CategorizingExceptionStatsHandler(categorizer = _ => Some("failures")) + + protected val statsReceiver: StatsReceiver + private val totalTasks = statsReceiver.counter("total") + private val successfulTasks = statsReceiver.counter("success") + private val taskLatency = statsReceiver.stat("latency_ms") + + private val activeTasks = new AtomicInteger(0) + + protected[common] val rng: Rng = Rng.threadLocal + protected[common] val timer: Timer = DefaultTimer + + @volatile private var taskLoop: Future[Unit] = null + + /** Execute the task wih bookkeeping **/ + private def run(): Future[Unit] = { + totalTasks.incr() + activeTasks.getAndIncrement() + + val start = Time.now + val runningTask = + // Setup a new trace root for this task. We also want logs to contain + // the same trace information finatra populates for requests. + // See com.twitter.finatra.thrift.filters.TraceIdMDCFilter + Trace.letTracerAndNextId(DefaultTracer) { + val trace = Trace() + MDC.put(MDCKeys.TraceId, trace.id.traceId.toString) + MDC.put(MDCKeys.TraceSampled, trace.id._sampled.getOrElse(false).toString) + MDC.put(MDCKeys.TraceSpanId, trace.id.spanId.toString) + + info(s"starting task ${getClass.toString}") + task() + .onSuccess({ _ => + info(s"completed task ${getClass.toString}") + successfulTasks.incr() + }) + .onFailure({ e => + warn(s"failed task. ", e) + exnStatsHandler.record(statsReceiver, e) + }) + } + + runningTask.transform { _ => + val elapsed = Time.now - start + activeTasks.getAndDecrement() + taskLatency.add(elapsed.inMilliseconds) + + Future + .sleep(taskInterval)(timer) + .before(run()) + } + } + + // Body of a task to run + protected def task(): Future[Unit] + + // Task interval + protected def taskInterval: Duration + + /** + * Start the task after random jitter + */ + final def jitteredStart(): Unit = synchronized { + if (taskLoop != null) { + throw new RuntimeException(s"task already started") + } else { + val jitterNs = rng.nextLong(taskInterval.inNanoseconds) + val jitter = Duration.fromNanoseconds(jitterNs) + + taskLoop = Future + .sleep(jitter)(timer) + .before(run()) + } + } + + /** + * Start the task without applying any delay + */ + final def startImmediately(): Unit = synchronized { + if (taskLoop != null) { + throw new RuntimeException(s"task already started") + } else { + taskLoop = run() + } + } + + /** + * Close the task. A closed task cannot be restarted. + */ + override def close(deadline: Time): Future[Unit] = { + if (taskLoop != null) { + taskLoop.raise(new InterruptedException("task closed")) + } + Future.Done + } +} diff --git a/ann/src/main/scala/com/twitter/ann/dataflow/offline/ANNIndexBuilderBeamJob.scala b/ann/src/main/scala/com/twitter/ann/dataflow/offline/ANNIndexBuilderBeamJob.scala new file mode 100644 index 0000000000..64ab583ab1 --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/dataflow/offline/ANNIndexBuilderBeamJob.scala @@ -0,0 +1,461 @@ +package com.twitter.ann.dataflow.offline + +import com.spotify.scio.ScioContext +import com.spotify.scio.ScioMetrics +import com.twitter.ann.annoy.TypedAnnoyIndex +import com.twitter.ann.brute_force.SerializableBruteForceIndex +import com.twitter.ann.common.thriftscala.AnnIndexMetadata +import com.twitter.ann.common.Distance +import com.twitter.ann.common.Cosine +import com.twitter.ann.common.EntityEmbedding +import com.twitter.ann.common.IndexOutputFile +import com.twitter.ann.common.Metric +import com.twitter.ann.common.ReadWriteFuturePool +import com.twitter.ann.faiss.FaissIndexer +import com.twitter.ann.hnsw.TypedHnswIndex +import com.twitter.ann.serialization.PersistedEmbeddingInjection +import com.twitter.ann.serialization.ThriftIteratorIO +import com.twitter.ann.serialization.thriftscala.PersistedEmbedding +import com.twitter.ann.util.IndexBuilderUtils +import com.twitter.beam.io.bigquery.BigQueryIO +import com.twitter.beam.io.dal.DalObservedDatasetRegistration +import com.twitter.beam.job.DateRange +import com.twitter.beam.job.DateRangeOptions +import com.twitter.cortex.ml.embeddings.common._ +import com.twitter.ml.api.embedding.Embedding +import com.twitter.ml.api.embedding.EmbeddingMath +import com.twitter.ml.api.embedding.EmbeddingSerDe +import com.twitter.ml.api.thriftscala.{Embedding => TEmbedding} +import com.twitter.ml.featurestore.lib.EntityId +import com.twitter.ml.featurestore.lib.SemanticCoreId +import com.twitter.ml.featurestore.lib.TfwId +import com.twitter.ml.featurestore.lib.TweetId +import com.twitter.ml.featurestore.lib.UserId +import com.twitter.scalding.DateOps +import com.twitter.scalding.RichDate +import com.twitter.scio_internal.job.ScioBeamJob +import com.twitter.statebird.v2.thriftscala.{Environment => StatebirdEnvironment} +import com.twitter.util.Await +import com.twitter.util.FuturePool +import com.twitter.wtf.beam.bq_embedding_export.BQQueryUtils +import java.time.Instant +import java.util.TimeZone +import java.util.concurrent.Executors +import org.apache.beam.sdk.io.FileSystems +import org.apache.beam.sdk.io.fs.ResolveOptions +import org.apache.beam.sdk.io.fs.ResourceId +import org.apache.beam.sdk.io.gcp.bigquery.BigQueryIO.TypedRead +import org.apache.beam.sdk.options.Default +import org.apache.beam.sdk.options.Description +import org.apache.beam.sdk.transforms.DoFn +import org.apache.beam.sdk.transforms.DoFn._ +import org.apache.beam.sdk.transforms.PTransform +import org.apache.beam.sdk.transforms.ParDo +import org.apache.beam.sdk.values.KV +import org.apache.beam.sdk.values.PCollection +import org.apache.beam.sdk.values.PDone +import org.slf4j.Logger +import org.slf4j.LoggerFactory + +trait ANNOptions extends DateRangeOptions { + @Description("Output GCS path for the generated index") + def getOutputPath(): String + def setOutputPath(value: String): Unit + + @Description("If set, the index is grouped") + @Default.Boolean(false) + def getGrouped: Boolean + def setGrouped(value: Boolean): Unit + + @Description( + "If set, a segment will be registered for the provided DAL dataset module which will trigger " + + "DAL registration.") + @Default.Boolean(false) + def getEnableDalRegistration: Boolean + def setEnableDalRegistration(value: Boolean): Unit + + @Description( + "Output GCS path for the generated index. The OutputPath should be of the format " + + "'gs://user.{{user_name}}.dp.gcp.twttr.net/subDir/outputDir' and OutputDALPath will be " + + "'subDir/outputDir' for this to work") + def getOutputDALPath: String + def setOutputDALPath(value: String): Unit + + @Description("Get ANN index dataset name") + def getDatasetModuleName: String + def setDatasetModuleName(value: String): Unit + + @Description("Get ANN index dataset owner role") + def getDatasetOwnerRole: String + def setDatasetOwnerRole(value: String): Unit + + @Description("If set, index is written in /") + @Default.Boolean(false) + def getOutputWithTimestamp: Boolean + def setOutputWithTimestamp(value: Boolean): Unit + + @Description("File which contains a SQL query to retrieve embeddings from BQ") + def getDatasetSqlPath: String + def setDatasetSqlPath(value: String): Unit + + @Description("Dimension of embedding in the input data. See go/ann") + def getDimension: Int + def setDimension(value: Int): Unit + + @Description("The type of entity ID that is used with the embeddings. See go/ann") + def getEntityKind: String + def setEntityKind(value: String): Unit + + @Description("The kind of index you want to generate (HNSW/Annoy/Brute Force/faiss). See go/ann") + def getAlgo: String + def setAlgo(value: String): Unit + + @Description("Distance metric (InnerProduct/Cosine/L2). See go/ann") + def getMetric: String + def setMetric(value: String): Unit + + @Description("Specifies how many parallel inserts happen to the index. See go/ann") + def getConcurrencyLevel: Int + def setConcurrencyLevel(value: Int): Unit + + @Description( + "Used by HNSW algo. Larger value increases build time but will give better recall. See go/ann") + def getEfConstruction: Int + def setEfConstruction(value: Int): Unit + + @Description( + "Used by HNSW algo. Larger value increases the index size but will give better recall. " + + "See go/ann") + def getMaxM: Int + def setMaxM(value: Int): Unit + + @Description("Used by HNSW algo. Approximate number of elements that will be indexed. See go/ann") + def getExpectedElements: Int + def setExpectedElements(value: Int): Unit + + @Description( + "Used by Annoy. num_trees is provided during build time and affects the build time and the " + + "index size. A larger value will give more accurate results, but larger indexes. See go/ann") + def getAnnoyNumTrees: Int + def setAnnoyNumTrees(value: Int): Unit + + @Description( + "FAISS factory string determines the ANN algorithm and compression. " + + "See https://github.com/facebookresearch/faiss/wiki/The-index-factory") + def getFAISSFactoryString: String + def setFAISSFactoryString(value: String): Unit + + @Description("Sample rate for training during creation of FAISS index. Default is 0.05f") + @Default.Float(0.05f) + def getTrainingSampleRate: Float + def setTrainingSampleRate(value: Float): Unit +} + +/** + * Builds ANN index. + * + * The input embeddings are read from BigQuery using the input SQL query. The output from this SQL + * query needs to have two columns, "entityID" [Long] and "embedding" [List[Double]] + * + * Output directory supported is GCS bucket + */ +object ANNIndexBuilderBeamJob extends ScioBeamJob[ANNOptions] { + val counterNameSpace = "ANNIndexBuilderBeamJob" + val LOG: Logger = LoggerFactory.getLogger(this.getClass) + implicit val timeZone: TimeZone = DateOps.UTC + + def configurePipeline(sc: ScioContext, opts: ANNOptions): Unit = { + val startDate: RichDate = RichDate(opts.interval.getStart.toDate) + val endDate: RichDate = RichDate(opts.interval.getEnd.toDate) + val instant = Instant.now() + val out = { + val base = FileSystems.matchNewResource(opts.getOutputPath, /*isDirectory=*/ true) + if (opts.getOutputWithTimestamp) { + base.resolve( + instant.toEpochMilli.toString, + ResolveOptions.StandardResolveOptions.RESOLVE_DIRECTORY) + } else { + base + } + } + + // Define template variables which we would like to be replaced in the corresponding sql file + val templateVariables = + Map( + "START_DATE" -> startDate.toString(DateOps.DATETIME_HMS_WITH_DASH), + "END_DATE" -> endDate.toString(DateOps.DATETIME_HMS_WITH_DASH) + ) + + val embeddingFetchQuery = + BQQueryUtils.getBQQueryFromSqlFile(opts.getDatasetSqlPath, templateVariables) + + val sCollection = if (opts.getGrouped) { + sc.customInput( + "Read grouped data from BQ", + BigQueryIO + .readClass[GroupedEmbeddingData]() + .fromQuery(embeddingFetchQuery).usingStandardSql() + .withMethod(TypedRead.Method.DIRECT_READ) + ) + } else { + sc.customInput( + "Read flat data from BQ", + BigQueryIO + .readClass[FlatEmbeddingData]().fromQuery(embeddingFetchQuery).usingStandardSql() + .withMethod(TypedRead.Method.DIRECT_READ) + ) + } + + val processedCollection = + sCollection + .flatMap(transformTableRowToKeyVal) + .groupBy(_.getKey) + .map { + case (groupName, groupValue) => + Map(groupName -> groupValue.map(_.getValue)) + } + + val annIndexMetadata = + AnnIndexMetadata(timestamp = Some(instant.getEpochSecond), withGroups = Some(opts.getGrouped)) + + // Count the number of groups and output the ANN index metadata + processedCollection.count.map(count => { + val annGroupedIndexMetadata = annIndexMetadata.copy( + numGroups = Some(count.intValue()) + ) + val indexOutDir = new IndexOutputFile(out) + indexOutDir.writeIndexMetadata(annGroupedIndexMetadata) + }) + + // Generate Index + processedCollection.saveAsCustomOutput( + "Serialise to Disk", + OutputSink( + out, + opts.getAlgo.equals("faiss"), + opts.getOutputDALPath, + opts.getEnableDalRegistration, + opts.getDatasetModuleName, + opts.getDatasetOwnerRole, + instant, + opts.getDate(), + counterNameSpace + ) + ) + } + + def transformTableRowToKeyVal( + data: BaseEmbeddingData + ): Option[KV[String, KV[Long, TEmbedding]]] = { + val transformTable = ScioMetrics.counter(counterNameSpace, "transform_table_row_to_kv") + for { + id <- data.entityId + } yield { + transformTable.inc() + val groupName: String = if (data.isInstanceOf[GroupedEmbeddingData]) { + (data.asInstanceOf[GroupedEmbeddingData]).groupId.get + } else { + "" + } + + KV.of[String, KV[Long, TEmbedding]]( + groupName, + KV.of[Long, TEmbedding]( + id, + EmbeddingSerDe.toThrift(Embedding(data.embedding.map(_.toFloat).toArray))) + ) + } + } + + case class OutputSink( + outDir: ResourceId, + isFaiss: Boolean, + outputDALPath: String, + enableDalRegistration: Boolean, + datasetModuleName: String, + datasetOwnerRole: String, + instant: Instant, + date: DateRange, + counterNameSpace: String) + extends PTransform[PCollection[Map[String, Iterable[KV[Long, TEmbedding]]]], PDone] { + override def expand(input: PCollection[Map[String, Iterable[KV[Long, TEmbedding]]]]): PDone = { + PDone.in { + val dummyOutput = { + if (isFaiss) { + input + .apply( + "Build&WriteFaissANNIndex", + ParDo.of(new BuildFaissANNIndex(outDir, counterNameSpace)) + ) + } else { + input + .apply( + "Build&WriteANNIndex", + ParDo.of(new BuildANNIndex(outDir, counterNameSpace)) + ) + } + } + + if (enableDalRegistration) { + input + .apply( + "Register DAL Dataset", + DalObservedDatasetRegistration( + datasetModuleName, + datasetOwnerRole, + outputDALPath, + instant, + Some(StatebirdEnvironment.Prod), + Some("ANN Index Data Files")) + ) + .getPipeline + } else { + dummyOutput.getPipeline + } + } + } + } + + class BuildANNIndex(outDir: ResourceId, counterNameSpace: String) + extends DoFn[Map[String, Iterable[KV[Long, TEmbedding]]], Unit] { + + def transformKeyValToEmbeddingWithEntity[T <: EntityId]( + entityKind: EntityKind[T] + )( + keyVal: KV[Long, TEmbedding] + ): EntityEmbedding[T] = { + val entityId = entityKind match { + case UserKind => UserId(keyVal.getKey).toThrift + case TweetKind => TweetId(keyVal.getKey).toThrift + case TfwKind => TfwId(keyVal.getKey).toThrift + case SemanticCoreKind => SemanticCoreId(keyVal.getKey).toThrift + case _ => throw new IllegalArgumentException(s"Unsupported embedding kind: $entityKind") + } + EntityEmbedding[T]( + EntityId.fromThrift(entityId).asInstanceOf[T], + EmbeddingSerDe.fromThrift(keyVal.getValue)) + } + + @ProcessElement + def processElement[T <: EntityId, D <: Distance[D]]( + @Element dataGrouped: Map[String, Iterable[KV[Long, TEmbedding]]], + context: ProcessContext + ): Unit = { + val opts = context.getPipelineOptions.as(classOf[ANNOptions]) + val uncastEntityKind = EntityKind.getEntityKind(opts.getEntityKind) + val entityKind = uncastEntityKind.asInstanceOf[EntityKind[T]] + val transformKVtoEmbeddings = + ScioMetrics.counter(counterNameSpace, "transform_kv_to_embeddings") + + val _ = dataGrouped.map { + case (groupName, data) => + val annEmbeddings = data.map { kv => + transformKVtoEmbeddings.inc() + transformKeyValToEmbeddingWithEntity(entityKind)(kv) + } + + val out = { + if (opts.getGrouped && groupName != "") { + outDir.resolve(groupName, ResolveOptions.StandardResolveOptions.RESOLVE_DIRECTORY) + } else { + outDir + } + } + LOG.info(s"Writing output to ${out}") + + val metric = Metric.fromString(opts.getMetric).asInstanceOf[Metric[D]] + val concurrencyLevel = opts.getConcurrencyLevel + val dimension = opts.getDimension + val threadPool = Executors.newFixedThreadPool(concurrencyLevel) + + LOG.info(s"Building ANN index of type ${opts.getAlgo}") + val serialization = opts.getAlgo match { + case "brute_force" => + val PersistedEmbeddingIO = + new ThriftIteratorIO[PersistedEmbedding](PersistedEmbedding) + SerializableBruteForceIndex( + metric, + FuturePool.apply(threadPool), + new PersistedEmbeddingInjection(entityKind.byteInjection), + PersistedEmbeddingIO + ) + case "annoy" => + TypedAnnoyIndex.indexBuilder( + dimension, + opts.getAnnoyNumTrees, + metric, + entityKind.byteInjection, + FuturePool.apply(threadPool) + ) + case "hnsw" => + val efConstruction = opts.getEfConstruction + val maxM = opts.getMaxM + val expectedElements = opts.getExpectedElements + TypedHnswIndex.serializableIndex( + dimension, + metric, + efConstruction, + maxM, + expectedElements, + entityKind.byteInjection, + ReadWriteFuturePool(FuturePool.apply(threadPool)) + ) + } + + val future = + IndexBuilderUtils.addToIndex(serialization, annEmbeddings.toSeq, concurrencyLevel) + Await.result(future.map { _ => + serialization.toDirectory(out) + }) + } + } + } + + class BuildFaissANNIndex(outDir: ResourceId, counterNameSpace: String) + extends DoFn[Map[String, Iterable[KV[Long, TEmbedding]]], Unit] { + + @ProcessElement + def processElement[D <: Distance[D]]( + @Element dataGrouped: Map[String, Iterable[KV[Long, TEmbedding]]], + context: ProcessContext + ): Unit = { + val opts = context.getPipelineOptions.as(classOf[ANNOptions]) + val transformKVtoEmbeddings = + ScioMetrics.counter(counterNameSpace, "transform_kv_to_embeddings") + + val _ = dataGrouped.map { + case (groupName, data) => + val out = { + if (opts.getGrouped && groupName != "") { + outDir.resolve(groupName, ResolveOptions.StandardResolveOptions.RESOLVE_DIRECTORY) + } else { + outDir + } + } + LOG.info(s"Writing output to ${out}") + + val metric = Metric.fromString(opts.getMetric).asInstanceOf[Metric[D]] + val maybeNormalizedPipe = data.map { kv => + transformKVtoEmbeddings.inc() + val embedding = EmbeddingSerDe.floatEmbeddingSerDe.fromThrift(kv.getValue) + EntityEmbedding[Long]( + kv.getKey, + if (metric == Cosine) { + EmbeddingMath.Float.normalize(embedding) + } else { + embedding + } + ) + } + + // Generate Index + FaissIndexer.buildAndWriteFaissIndex( + maybeNormalizedPipe, + opts.getTrainingSampleRate, + opts.getFAISSFactoryString, + metric, + new IndexOutputFile(out)) + } + } + } +} diff --git a/ann/src/main/scala/com/twitter/ann/dataflow/offline/BUILD b/ann/src/main/scala/com/twitter/ann/dataflow/offline/BUILD new file mode 100644 index 0000000000..b88ee213e3 --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/dataflow/offline/BUILD @@ -0,0 +1,27 @@ +scala_library( + name = "index_builder_lib", + sources = [ + "*.scala", + ], + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/com/spotify:scio-core", + "3rdparty/jvm/org/apache/beam:beam-sdks-java-core", + "ann/src/main/java/com/twitter/ann/faiss", + "ann/src/main/scala/com/twitter/ann/annoy", + "ann/src/main/scala/com/twitter/ann/brute_force", + "ann/src/main/scala/com/twitter/ann/common", + "ann/src/main/scala/com/twitter/ann/faiss", + "ann/src/main/scala/com/twitter/ann/hnsw", + "ann/src/main/scala/com/twitter/ann/serialization", + "ann/src/main/scala/com/twitter/ann/util", + "ann/src/main/thrift/com/twitter/ann/common:ann-common-scala", + "beam-internal/src/main/scala/com/twitter/beam/io/bigquery", + "beam-internal/src/main/scala/com/twitter/beam/io/dal", + "beam-internal/src/main/scala/com/twitter/beam/job", + "beam-internal/src/main/scala/com/twitter/scio_internal/runner/dataflow", + "src/scala/com/twitter/cortex/ml/embeddings/common:Helpers", + "src/scala/com/twitter/ml/featurestore/lib", + "src/scala/com/twitter/wtf/beam/bq_embedding_export:bq_embedding_export_lib", + ], +) diff --git a/ann/src/main/scala/com/twitter/ann/dataflow/offline/BaseEmbeddingData.scala b/ann/src/main/scala/com/twitter/ann/dataflow/offline/BaseEmbeddingData.scala new file mode 100644 index 0000000000..a9a291d076 --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/dataflow/offline/BaseEmbeddingData.scala @@ -0,0 +1,6 @@ +package com.twitter.ann.dataflow.offline + +trait BaseEmbeddingData { + val entityId: Option[Long] + val embedding: Seq[Double] +} diff --git a/ann/src/main/scala/com/twitter/ann/dataflow/offline/FlatEmbeddingData.scala b/ann/src/main/scala/com/twitter/ann/dataflow/offline/FlatEmbeddingData.scala new file mode 100644 index 0000000000..ad211fb683 --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/dataflow/offline/FlatEmbeddingData.scala @@ -0,0 +1,8 @@ +package com.twitter.ann.dataflow.offline + +import com.twitter.beam.schemas.SchemaFieldName + +case class FlatEmbeddingData( + @SchemaFieldName("entityId") entityId: Option[Long], + @SchemaFieldName("embedding") embedding: Seq[Double]) + extends BaseEmbeddingData diff --git a/ann/src/main/scala/com/twitter/ann/dataflow/offline/GroupedEmbeddingData.scala b/ann/src/main/scala/com/twitter/ann/dataflow/offline/GroupedEmbeddingData.scala new file mode 100644 index 0000000000..63342272fb --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/dataflow/offline/GroupedEmbeddingData.scala @@ -0,0 +1,9 @@ +package com.twitter.ann.dataflow.offline + +import com.twitter.beam.schemas.SchemaFieldName + +case class GroupedEmbeddingData( + @SchemaFieldName("entityId") entityId: Option[Long], + @SchemaFieldName("embedding") embedding: Seq[Double], + @SchemaFieldName("groupId") groupId: Option[String], +) extends BaseEmbeddingData diff --git a/ann/src/main/scala/com/twitter/ann/experimental/BUILD.bazel b/ann/src/main/scala/com/twitter/ann/experimental/BUILD.bazel new file mode 100644 index 0000000000..8335984c19 --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/experimental/BUILD.bazel @@ -0,0 +1,29 @@ +scala_library( + name = "server", + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-only"], + dependencies = [ + "ann/src/main/scala/com/twitter/ann/annoy", + "ann/src/main/scala/com/twitter/ann/brute_force", + "ann/src/main/scala/com/twitter/ann/common", + "ann/src/main/scala/com/twitter/ann/hnsw", + ], +) + +hadoop_binary( + name = "benchmarking", + basename = "benchmarking", + main = "com.twitter.ann.experimental.Runner", + platform = "java8", + runtime_platform = "java8", + tags = [ + "bazel-compatible", + "bazel-compatible:migrated", + "bazel-only", + ], + dependencies = [ + ":server", + ], +) diff --git a/ann/src/main/scala/com/twitter/ann/experimental/Runner.scala b/ann/src/main/scala/com/twitter/ann/experimental/Runner.scala new file mode 100644 index 0000000000..da0d5e3eed --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/experimental/Runner.scala @@ -0,0 +1,171 @@ +package com.twitter.ann.experimental + +import com.twitter.ann.annoy.{AnnoyRuntimeParams, TypedAnnoyIndex} +import com.twitter.ann.brute_force.{BruteForceIndex, BruteForceRuntimeParams} +import com.twitter.ann.common.{Cosine, CosineDistance, EntityEmbedding, ReadWriteFuturePool} +import com.twitter.ann.hnsw.{HnswParams, TypedHnswIndex} +import com.twitter.bijection.Injection +import com.twitter.ml.api.embedding.Embedding +import com.twitter.search.common.file.LocalFile +import com.twitter.util.{Await, Future, FuturePool} +import java.nio.file.Files +import java.util +import java.util.concurrent.Executors +import java.util.{Collections, Random} +import scala.collection.JavaConverters._ +import scala.collection.mutable + +object Runner { + def main(args: Array[String]): Unit = { + val rng = new Random() + val dimen = 300 + val neighbours = 20 + val trainDataSetSize = 2000 + val testDataSetSize = 30 + + // Hnsw (ef -> (time, recall)) + val hnswEfConfig = new mutable.HashMap[Int, (Float, Float)] + val efConstruction = 200 + val maxM = 16 + val threads = 24 + val efSearch = + Seq(20, 30, 50, 70, 100, 120) + efSearch.foreach(hnswEfConfig.put(_, (0.0f, 0.0f))) + + // Annoy (nodes to explore -> (time, recall)) + val numOfTrees = 80 + val annoyConfig = new mutable.HashMap[Int, (Float, Float)] + val nodesToExplore = Seq(0, 2000, 3000, 5000, 7000, 10000, 15000, 20000, + 30000, 35000, 40000, 50000) + nodesToExplore.foreach(annoyConfig.put(_, (0.0f, 0.0f))) + val injection = Injection.int2BigEndian + val distance = Cosine + val exec = Executors.newFixedThreadPool(threads) + val pool = FuturePool.apply(exec) + val hnswMultiThread = + TypedHnswIndex.index[Int, CosineDistance]( + dimen, + distance, + efConstruction = efConstruction, + maxM = maxM, + trainDataSetSize, + ReadWriteFuturePool(pool) + ) + + val bruteforce = BruteForceIndex[Int, CosineDistance](distance, pool) + val annoyBuilder = + TypedAnnoyIndex.indexBuilder(dimen, numOfTrees, distance, injection, FuturePool.immediatePool) + val temp = new LocalFile(Files.createTempDirectory("test").toFile) + + println("Creating bruteforce.........") + val data = + Collections.synchronizedList(new util.ArrayList[EntityEmbedding[Int]]()) + val bruteforceFutures = 1 to trainDataSetSize map { id => + val vec = Array.fill(dimen)(rng.nextFloat() * 50) + val emb = EntityEmbedding[Int](id, Embedding(vec)) + data.add(emb) + bruteforce.append(emb) + } + + Await.result(Future.collect(bruteforceFutures)) + + println("Creating hnsw multithread test.........") + val (_, multiThreadInsertion) = time { + Await.result(Future.collect(data.asScala.toList.map { emb => + hnswMultiThread.append(emb) + })) + } + + println("Creating annoy.........") + val (_, annoyTime) = time { + Await.result(Future.collect(data.asScala.toList.map(emb => + annoyBuilder.append(emb)))) + annoyBuilder.toDirectory(temp) + } + + val annoyQuery = TypedAnnoyIndex.loadQueryableIndex( + dimen, + Cosine, + injection, + FuturePool.immediatePool, + temp + ) + + val hnswQueryable = hnswMultiThread.toQueryable + + println(s"Total train size : $trainDataSetSize") + println(s"Total querySize : $testDataSetSize") + println(s"Dimension : $dimen") + println(s"Distance type : $distance") + println(s"Annoy index creation time trees: $numOfTrees => $annoyTime ms") + println( + s"Hnsw multi thread creation time : $multiThreadInsertion ms efCons: $efConstruction maxM $maxM thread : $threads") + println("Querying.........") + var bruteForceTime = 0.0f + 1 to testDataSetSize foreach { id => + println("Querying id " + id) + val embedding = Embedding(Array.fill(dimen)(rng.nextFloat())) + + val (list, timeTakenB) = + time( + Await + .result( + bruteforce.query(embedding, neighbours, BruteForceRuntimeParams)) + .toSet) + bruteForceTime += timeTakenB + + val annoyConfigCopy = annoyConfig.toMap + val hnswEfConfigCopy = hnswEfConfig.toMap + + hnswEfConfigCopy.keys.foreach { ef => + val (nn, timeTaken) = + time(Await + .result(hnswQueryable.query(embedding, neighbours, HnswParams(ef))) + .toSet) + val recall = (list.intersect(nn).size) * 1.0f / neighbours + val (oldTime, oldRecall) = hnswEfConfig(ef) + hnswEfConfig.put(ef, (oldTime + timeTaken, oldRecall + recall)) + } + + annoyConfigCopy.keys.foreach { nodes => + val (nn, timeTaken) = + time( + Await.result( + annoyQuery + .query(embedding, + neighbours, + AnnoyRuntimeParams(nodesToExplore = Some(nodes))) + .map(_.toSet))) + val recall = (list.intersect(nn).size) * 1.0f / neighbours + val (oldTime, oldRecall) = annoyConfig(nodes) + annoyConfig.put(nodes, (oldTime + timeTaken, oldRecall + recall)) + } + } + + println( + s"Bruteforce avg query time : ${bruteForceTime / testDataSetSize} ms") + + efSearch.foreach { ef => + val data = hnswEfConfig(ef) + println( + s"Hnsw avg recall and time with query ef : $ef => ${data._2 / testDataSetSize} ${data._1 / testDataSetSize} ms" + ) + } + + nodesToExplore.foreach { n => + val data = annoyConfig(n) + println( + s"Annoy avg recall and time with nodes_to_explore : $n => ${data._2 / testDataSetSize} ${data._1 / testDataSetSize} ms" + ) + } + + exec.shutdown() + } + + def time[T](fn: => T): (T, Long) = { + val start = System.currentTimeMillis() + val result = fn + val end = System.currentTimeMillis() + (result, (end - start)) + } +} diff --git a/ann/src/main/scala/com/twitter/ann/faiss/BUILD b/ann/src/main/scala/com/twitter/ann/faiss/BUILD new file mode 100644 index 0000000000..dd09cbbae8 --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/faiss/BUILD @@ -0,0 +1,23 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/org/mapdb", + "ann/src/main/java/com/twitter/ann/faiss", + "ann/src/main/scala/com/twitter/ann/common", + "ann/src/main/scala/com/twitter/ann/serialization", + "ann/src/main/thrift/com/twitter/ann/common:ann-common-scala", + "mediaservices/commons/src/main/scala:futuretracker", + "src/java/com/twitter/common_internal/hadoop", + "src/java/com/twitter/search/common/file", + "src/scala/com/twitter/ml/api/embedding", + ], + exports = [ + "ann/src/main/scala/com/twitter/ann/common", + "src/java/com/twitter/common_internal/hadoop", + "src/java/com/twitter/search/common/file", + "src/scala/com/twitter/ml/api/embedding", + ], +) diff --git a/ann/src/main/scala/com/twitter/ann/faiss/FaissCommon.scala b/ann/src/main/scala/com/twitter/ann/faiss/FaissCommon.scala new file mode 100644 index 0000000000..96f40708ee --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/faiss/FaissCommon.scala @@ -0,0 +1,44 @@ +package com.twitter.ann.faiss + +import com.twitter.ann.common.thriftscala.FaissRuntimeParam +import com.twitter.bijection.Injection +import scala.util.Failure +import scala.util.Success +import scala.util.Try +import com.twitter.ann.common.thriftscala.{RuntimeParams => ServiceRuntimeParams} +import com.twitter.search.common.file.AbstractFile + +object FaissCommon { + val RuntimeParamsInjection: Injection[FaissParams, ServiceRuntimeParams] = + new Injection[FaissParams, ServiceRuntimeParams] { + override def apply(scalaParams: FaissParams): ServiceRuntimeParams = { + ServiceRuntimeParams.FaissParam( + FaissRuntimeParam( + scalaParams.nprobe, + scalaParams.quantizerEf, + scalaParams.quantizerKFactorRF, + scalaParams.quantizerNprobe, + scalaParams.ht) + ) + } + + override def invert(thriftParams: ServiceRuntimeParams): Try[FaissParams] = + thriftParams match { + case ServiceRuntimeParams.FaissParam(faissParam) => + Success( + FaissParams( + faissParam.nprobe, + faissParam.quantizerEf, + faissParam.quantizerKfactorRf, + faissParam.quantizerNprobe, + faissParam.ht)) + case p => Failure(new IllegalArgumentException(s"Expected FaissParams got $p")) + } + } + + def isValidFaissIndex(path: AbstractFile): Boolean = { + path.isDirectory && + path.hasSuccessFile && + path.getChild("faiss.index").exists() + } +} diff --git a/ann/src/main/scala/com/twitter/ann/faiss/FaissIndex.scala b/ann/src/main/scala/com/twitter/ann/faiss/FaissIndex.scala new file mode 100644 index 0000000000..63930247ab --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/faiss/FaissIndex.scala @@ -0,0 +1,43 @@ +package com.twitter.ann.faiss + +import com.twitter.ann.common.Queryable +import com.twitter.ann.common._ +import com.twitter.search.common.file.AbstractFile +import com.twitter.util.logging.Logging + +case class FaissParams( + nprobe: Option[Int], + quantizerEf: Option[Int], + quantizerKFactorRF: Option[Int], + quantizerNprobe: Option[Int], + ht: Option[Int]) + extends RuntimeParams { + override def toString: String = s"FaissParams(${toLibraryString})" + + def toLibraryString: String = + Seq( + nprobe.map { n => s"nprobe=${n}" }, + quantizerEf.map { ef => s"quantizer_efSearch=${ef}" }, + quantizerKFactorRF.map { k => s"quantizer_k_factor_rf=${k}" }, + quantizerNprobe.map { n => s"quantizer_nprobe=${n}" }, + ht.map { ht => s"ht=${ht}" }, + ).flatten.mkString(",") +} + +object FaissIndex { + def loadIndex[T, D <: Distance[D]]( + outerDimension: Int, + outerMetric: Metric[D], + directory: AbstractFile + ): Queryable[T, FaissParams, D] = { + new QueryableIndexAdapter[T, D] with Logging { + protected val metric: Metric[D] = outerMetric + protected val dimension: Int = outerDimension + protected val index: Index = { + info(s"Loading faiss with ${swigfaiss.get_compile_options()}") + + QueryableIndexAdapter.loadJavaIndex(directory) + } + } + } +} diff --git a/ann/src/main/scala/com/twitter/ann/faiss/FaissIndexer.scala b/ann/src/main/scala/com/twitter/ann/faiss/FaissIndexer.scala new file mode 100644 index 0000000000..bb6cae0b57 --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/faiss/FaissIndexer.scala @@ -0,0 +1,154 @@ +package com.twitter.ann.faiss + +import com.google.common.base.Preconditions +import com.twitter.ann.common.Cosine +import com.twitter.ann.common.Distance +import com.twitter.ann.common.EntityEmbedding +import com.twitter.ann.common.IndexOutputFile +import com.twitter.ann.common.InnerProduct +import com.twitter.ann.common.L2 +import com.twitter.ann.common.Metric +import com.twitter.ml.api.embedding.EmbeddingMath +import com.twitter.scalding.Execution +import com.twitter.scalding.TypedPipe +import com.twitter.search.common.file.AbstractFile +import com.twitter.search.common.file.FileUtils +import com.twitter.util.logging.Logging +import java.io.File +import scala.util.Random + +trait FaissIndexer extends Logging { + + /** + * Produce faiss index file specified by factory string + * + * @param pipe Embeddings to be indexed + * @param sampleRate Fraction of embeddings used for training. Regardless of this parameter, all embeddings are present in the output. + * @param factoryString Faiss factory string, see https://github.com/facebookresearch/faiss/wiki/The-index-factory + * @param metric Metric to use + * @param outputDirectory Directory where _SUCCESS and faiss.index will be written. + */ + def build[D <: Distance[D]]( + pipe: TypedPipe[EntityEmbedding[Long]], + sampleRate: Float, + factoryString: String, + metric: Metric[D], + outputDirectory: AbstractFile + ): Execution[Unit] = { + outputDirectory.mkdirs() + Preconditions.checkState( + outputDirectory.canRead, + "Failed to create parent directories for %s", + outputDirectory.toString) + + val maybeNormalizedPipe = if (l2Normalize(metric)) { + pipe.map { idAndEmbedding => + EntityEmbedding(idAndEmbedding.id, EmbeddingMath.Float.normalize(idAndEmbedding.embedding)) + } + } else { + pipe + } + + maybeNormalizedPipe.toIterableExecution.flatMap { annEmbeddings => + logger.info(s"${factoryString}") + val t1 = System.nanoTime + buildAndWriteFaissIndex( + Random.shuffle(annEmbeddings), + sampleRate, + factoryString, + metric, + new IndexOutputFile(outputDirectory)) + val duration = (System.nanoTime - t1) / 1e9d + logger.info(s"It took ${duration}s to build and index") + + Execution.unit + } + } + + def buildAndWriteFaissIndex[D <: Distance[D]]( + entities: Iterable[EntityEmbedding[Long]], + sampleRate: Float, + factoryString: String, + metricType: Metric[D], + outputDirectory: IndexOutputFile + ): Unit = { + val metric = parseMetric(metricType) + val datasetSize = entities.size.toLong + val dimensions = entities.head.embedding.length + logger.info(s"There are $datasetSize embeddings") + logger.info(s"Faiss compile options are ${swigfaiss.get_compile_options()}") + logger.info(s"OMP threads count is ${swigfaiss.omp_get_max_threads()}") + + val index = swigfaiss.index_factory(dimensions, factoryString, metric) + index.setVerbose(true) + val idMap = new IndexIDMap(index) + + val trainingSetSize = Math.min(datasetSize, Math.round(datasetSize * sampleRate)) + val ids = toIndexVector(entities) + val fullDataset = toFloatVector(dimensions, entities) + logger.info("Finished bridging full dataset") + idMap.train(trainingSetSize, fullDataset.data()) + logger.info("Finished training") + idMap.add_with_ids(datasetSize, fullDataset.data(), ids) + logger.info("Added data to the index") + + val tmpFile = File.createTempFile("faiss.index", ".tmp") + swigfaiss.write_index(idMap, tmpFile.toString) + logger.info(s"Wrote to tmp file ${tmpFile.toString}") + copyToOutputAndCreateSuccess(FileUtils.getFileHandle(tmpFile.toString), outputDirectory) + logger.info("Copied file") + } + + private def copyToOutputAndCreateSuccess( + tmpFile: AbstractFile, + outputDirectory: IndexOutputFile + ) = { + val outputFile = outputDirectory.createFile("faiss.index") + logger.info(s"Final output file is ${outputFile.getPath()}") + outputFile.copyFrom(tmpFile.getByteSource.openStream()) + outputDirectory.createSuccessFile() + } + + private def toFloatVector( + dimensions: Int, + entities: Iterable[EntityEmbedding[Long]] + ): FloatVector = { + require(entities.nonEmpty) + + val vector = new FloatVector() + vector.reserve(dimensions.toLong * entities.size.toLong) + for (entity <- entities) { + for (value <- entity.embedding) { + vector.push_back(value) + } + } + + vector + } + + private def toIndexVector(embeddings: Iterable[EntityEmbedding[Long]]): LongVector = { + require(embeddings.nonEmpty) + + val vector = new LongVector() + vector.reserve(embeddings.size) + for (embedding <- embeddings) { + vector.push_back(embedding.id) + } + + vector + } + + private def parseMetric[D <: Distance[D]](metric: Metric[D]): MetricType = metric match { + case L2 => MetricType.METRIC_L2 + case InnerProduct => MetricType.METRIC_INNER_PRODUCT + case Cosine => MetricType.METRIC_INNER_PRODUCT + case _ => throw new AbstractMethodError(s"Not implemented for metric ${metric}") + } + + private def l2Normalize[D <: Distance[D]](metric: Metric[D]): Boolean = metric match { + case Cosine => true + case _ => false + } +} + +object FaissIndexer extends FaissIndexer {} diff --git a/ann/src/main/scala/com/twitter/ann/faiss/HourlyDirectoryWithSuccessFileListing.scala b/ann/src/main/scala/com/twitter/ann/faiss/HourlyDirectoryWithSuccessFileListing.scala new file mode 100644 index 0000000000..938ee80ed9 --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/faiss/HourlyDirectoryWithSuccessFileListing.scala @@ -0,0 +1,64 @@ +package com.twitter.ann.faiss + +import com.twitter.conversions.DurationOps.richDurationFromInt +import com.twitter.search.common.file.AbstractFile +import com.twitter.search.common.file.FileUtils +import com.twitter.util.Return +import com.twitter.util.Throw +import com.twitter.util.Time +import com.twitter.util.Try +import com.twitter.util.logging.Logging +import java.util.Locale + +object HourlyDirectoryWithSuccessFileListing extends Logging { + private val SUCCESS_FILE_NAME = "_SUCCESS" + + def listHourlyIndexDirectories( + root: AbstractFile, + startingFrom: Time, + count: Int, + lookbackInterval: Int + ): Seq[AbstractFile] = listingStep(root, startingFrom, count, lookbackInterval) + + private def listingStep( + root: AbstractFile, + startingFrom: Time, + remainingDirectoriesToFind: Int, + remainingAttempts: Int + ): List[AbstractFile] = { + if (remainingDirectoriesToFind == 0 || remainingAttempts == 0) { + return List.empty + } + + val head = getSuccessfulDirectoryForDate(root, startingFrom) + + val previousHour = startingFrom - 1.hour + + head match { + case Throw(e) => + listingStep(root, previousHour, remainingDirectoriesToFind, remainingAttempts - 1) + case Return(directory) => + directory :: + listingStep(root, previousHour, remainingDirectoriesToFind - 1, remainingAttempts - 1) + } + } + + private def getSuccessfulDirectoryForDate( + root: AbstractFile, + date: Time + ): Try[AbstractFile] = { + val folder = root.getPath + "/" + date.format("yyyy/MM/dd/HH", Locale.ROOT) + val successPath = + folder + "/" + SUCCESS_FILE_NAME + + debug(s"Checking ${successPath}") + + Try(FileUtils.getFileHandle(successPath)).flatMap { file => + if (file.canRead) { + Try(FileUtils.getFileHandle(folder)) + } else { + Throw(new IllegalArgumentException(s"Found ${file.toString} but can't read it")) + } + } + } +} diff --git a/ann/src/main/scala/com/twitter/ann/faiss/HourlyShardedIndex.scala b/ann/src/main/scala/com/twitter/ann/faiss/HourlyShardedIndex.scala new file mode 100644 index 0000000000..9d8d942182 --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/faiss/HourlyShardedIndex.scala @@ -0,0 +1,94 @@ +package com.twitter.ann.faiss + +import com.twitter.ann.common.Distance +import com.twitter.ann.common.MemoizedInEpochs +import com.twitter.ann.common.Metric +import com.twitter.ann.common.Task +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.search.common.file.AbstractFile +import com.twitter.util.Duration +import com.twitter.util.Future +import com.twitter.util.Time +import com.twitter.util.Try +import com.twitter.util.logging.Logging +import java.util.concurrent.atomic.AtomicReference + +object HourlyShardedIndex { + def loadIndex[T, D <: Distance[D]]( + dimension: Int, + metric: Metric[D], + directory: AbstractFile, + shardsToLoad: Int, + shardWatchInterval: Duration, + lookbackInterval: Int, + statsReceiver: StatsReceiver + ): HourlyShardedIndex[T, D] = { + new HourlyShardedIndex[T, D]( + metric, + dimension, + directory, + shardsToLoad, + shardWatchInterval, + lookbackInterval, + statsReceiver) + } +} + +class HourlyShardedIndex[T, D <: Distance[D]]( + outerMetric: Metric[D], + outerDimension: Int, + directory: AbstractFile, + shardsToLoad: Int, + shardWatchInterval: Duration, + lookbackInterval: Int, + override protected val statsReceiver: StatsReceiver) + extends QueryableIndexAdapter[T, D] + with Logging + with Task { + // QueryableIndexAdapter + protected val metric: Metric[D] = outerMetric + protected val dimension: Int = outerDimension + protected def index: Index = { + castedIndex.get() + } + + // Task trait + protected def task(): Future[Unit] = Future.value(reloadShards()) + protected def taskInterval: Duration = shardWatchInterval + + private def loadIndex(directory: AbstractFile): Try[Index] = + Try(QueryableIndexAdapter.loadJavaIndex(directory)) + + private val shardsCache = new MemoizedInEpochs[AbstractFile, Index](loadIndex) + // Destroying original index invalidate casted index. Keep a reference to both. + private val originalIndex = new AtomicReference[IndexShards]() + private val castedIndex = new AtomicReference[Index]() + private def reloadShards(): Unit = { + val freshDirectories = + HourlyDirectoryWithSuccessFileListing.listHourlyIndexDirectories( + directory, + Time.now, + shardsToLoad, + lookbackInterval) + + if (shardsCache.currentEpochKeys == freshDirectories.toSet) { + info("Not reloading shards, as they're exactly same") + } else { + val shards = shardsCache.epoch(freshDirectories) + val indexShards = new IndexShards(dimension, false, false) + for (shard <- shards) { + indexShards.add_shard(shard) + } + + replaceIndex(() => { + castedIndex.set(swigfaiss.upcast_IndexShards(indexShards)) + originalIndex.set(indexShards) + }) + + // Potentially it's time to drop huge native index from memory, ask for GC + System.gc() + } + + require(castedIndex.get() != null, "Failed to find any shards during startup") + } +} diff --git a/ann/src/main/scala/com/twitter/ann/faiss/QueryableIndexAdapter.scala b/ann/src/main/scala/com/twitter/ann/faiss/QueryableIndexAdapter.scala new file mode 100644 index 0000000000..642e7d2089 --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/faiss/QueryableIndexAdapter.scala @@ -0,0 +1,196 @@ +package com.twitter.ann.faiss + +import com.twitter.ann.common.Cosine +import com.twitter.ann.common.Distance +import com.twitter.ann.common.EmbeddingType.EmbeddingVector +import com.twitter.ann.common.Metric +import com.twitter.ann.common.NeighborWithDistance +import com.twitter.ann.common.Queryable +import com.twitter.ml.api.embedding.EmbeddingMath +import com.twitter.search.common.file.AbstractFile +import com.twitter.search.common.file.FileUtils +import com.twitter.util.Future +import com.twitter.util.logging.Logging +import java.io.File +import java.util.concurrent.locks.ReentrantReadWriteLock + +object QueryableIndexAdapter extends Logging { + // swigfaiss.read_index doesn't support hdfs files, hence a copy to temporary directory + def loadJavaIndex(directory: AbstractFile): Index = { + val indexFile = directory.getChild("faiss.index") + val tmpFile = File.createTempFile("faiss.index", ".tmp") + val tmpAbstractFile = FileUtils.getFileHandle(tmpFile.toString) + indexFile.copyTo(tmpAbstractFile) + val index = swigfaiss.read_index(tmpAbstractFile.getPath) + + if (!tmpFile.delete()) { + error(s"Failed to delete ${tmpFile.toString}") + } + + index + } +} + +trait QueryableIndexAdapter[T, D <: Distance[D]] extends Queryable[T, FaissParams, D] { + this: Logging => + + private val MAX_COSINE_DISTANCE = 1f + + protected def index: Index + protected val metric: Metric[D] + protected val dimension: Int + + private def maybeNormalizeEmbedding(embeddingVector: EmbeddingVector): EmbeddingVector = { + // There is no direct support for Cosine, but l2norm + ip == Cosine by definition + if (metric == Cosine) { + EmbeddingMath.Float.normalize(embeddingVector) + } else { + embeddingVector + } + } + + private def maybeTranslateToCosineDistanceInplace(array: floatArray, len: Int): Unit = { + // Faiss reports Cosine similarity while we need Cosine distance. + if (metric == Cosine) { + for (index <- 0 until len) { + val similarity = array.getitem(index) + if (similarity < 0 || similarity > 1) { + warn(s"Expected similarity to be between 0 and 1, got ${similarity} instead") + array.setitem(index, MAX_COSINE_DISTANCE) + } else { + array.setitem(index, 1 - similarity) + } + } + } + } + + private val paramsLock = new ReentrantReadWriteLock() + private var currentParams: Option[String] = None + // Assume that parameters rarely change and try read lock first + private def ensuringParams[R](parameterString: String, f: () => R): R = { + paramsLock.readLock().lock() + try { + if (currentParams.contains(parameterString)) { + return f() + } + } finally { + paramsLock.readLock().unlock() + } + + paramsLock.writeLock().lock() + try { + currentParams = Some(parameterString) + new ParameterSpace().set_index_parameters(index, parameterString) + + f() + } finally { + paramsLock.writeLock().unlock() + } + } + + def replaceIndex(f: () => Unit): Unit = { + paramsLock.writeLock().lock() + try { + currentParams = None + + f() + } finally { + paramsLock.writeLock().unlock() + } + } + + def query( + embedding: EmbeddingVector, + numOfNeighbors: Int, + runtimeParams: FaissParams + ): Future[List[T]] = { + Future.value( + ensuringParams( + runtimeParams.toLibraryString, + () => { + val distances = new floatArray(numOfNeighbors) + val indexes = new LongVector() + indexes.resize(numOfNeighbors) + + val normalizedEmbedding = maybeNormalizeEmbedding(embedding) + index.search( + // Number of query embeddings + 1, + // Array of query embeddings + toFloatArray(normalizedEmbedding).cast(), + // Number of neighbours to return + numOfNeighbors, + // Location to store neighbour distances + distances.cast(), + // Location to store neighbour identifiers + indexes + ) + // This is a shortcoming of current swig bindings + // Nothing prevents JVM from freeing distances while inside index.search + // This might be removed once we start passing FloatVector + // Why java.lang.ref.Reference.reachabilityFence doesn't compile? + debug(distances) + + toSeq(indexes, numOfNeighbors).toList.asInstanceOf[List[T]] + } + )) + } + + def queryWithDistance( + embedding: EmbeddingVector, + numOfNeighbors: Int, + runtimeParams: FaissParams + ): Future[List[NeighborWithDistance[T, D]]] = { + Future.value( + ensuringParams( + runtimeParams.toLibraryString, + () => { + val distances = new floatArray(numOfNeighbors) + val indexes = new LongVector() + indexes.resize(numOfNeighbors) + + val normalizedEmbedding = maybeNormalizeEmbedding(embedding) + index.search( + // Number of query embeddings + 1, + // Array of query embeddings + toFloatArray(normalizedEmbedding).cast(), + // Number of neighbours to return + numOfNeighbors, + // Location to store neighbour distances + distances.cast(), + // Location to store neighbour identifiers + indexes + ) + + val ids = toSeq(indexes, numOfNeighbors).toList.asInstanceOf[List[T]] + + maybeTranslateToCosineDistanceInplace(distances, numOfNeighbors) + + val distancesSeq = toSeq(distances, numOfNeighbors) + + ids.zip(distancesSeq).map { + case (id, distance) => + NeighborWithDistance(id, metric.fromAbsoluteDistance(distance)) + } + } + )) + } + + private def toFloatArray(emb: EmbeddingVector): floatArray = { + val nativeArray = new floatArray(emb.length) + for ((value, aIdx) <- emb.iterator.zipWithIndex) { + nativeArray.setitem(aIdx, value) + } + + nativeArray + } + + private def toSeq(vector: LongVector, len: Long): Seq[Long] = { + (0L until len).map(vector.at) + } + + private def toSeq(array: floatArray, len: Int): Seq[Float] = { + (0 until len).map(array.getitem) + } +} diff --git a/ann/src/main/scala/com/twitter/ann/featurestore/BUILD b/ann/src/main/scala/com/twitter/ann/featurestore/BUILD new file mode 100644 index 0000000000..959745b927 --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/featurestore/BUILD @@ -0,0 +1,10 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + tags = ["bazel-compatible"], + dependencies = [ + "ann/src/main/scala/com/twitter/ann/common", + "src/scala/com/twitter/ml/featurestore/lib", + "src/scala/com/twitter/ml/featurestore/lib/online", + ], +) diff --git a/ann/src/main/scala/com/twitter/ann/featurestore/FeatureStoreEmbeddingProducer.scala b/ann/src/main/scala/com/twitter/ann/featurestore/FeatureStoreEmbeddingProducer.scala new file mode 100644 index 0000000000..d71720ea44 --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/featurestore/FeatureStoreEmbeddingProducer.scala @@ -0,0 +1,66 @@ +package com.twitter.ann.featurestore + +import com.twitter.ann.common.EmbeddingProducer +import com.twitter.finagle.stats.{InMemoryStatsReceiver, StatsReceiver} +import com.twitter.ml.api.embedding.{Embedding, EmbeddingSerDe} +import com.twitter.ml.api.thriftscala +import com.twitter.ml.api.thriftscala.{Embedding => TEmbedding} +import com.twitter.ml.featurestore.lib.dataset.online.VersionedOnlineAccessDataset +import com.twitter.ml.featurestore.lib.{EntityId, RawFloatTensor} +import com.twitter.ml.featurestore.lib.dataset.DatasetParams +import com.twitter.ml.featurestore.lib.entity.EntityWithId +import com.twitter.ml.featurestore.lib.feature.{BoundFeature, BoundFeatureSet} +import com.twitter.ml.featurestore.lib.online.{FeatureStoreClient, FeatureStoreRequest} +import com.twitter.ml.featurestore.lib.params.FeatureStoreParams +import com.twitter.stitch.Stitch +import com.twitter.strato.opcontext.Attribution +import com.twitter.strato.client.Client + +object FeatureStoreEmbeddingProducer { + def apply[T <: EntityId]( + dataset: VersionedOnlineAccessDataset[T, TEmbedding], + version: Long, + boundFeature: BoundFeature[T, RawFloatTensor], + client: Client, + statsReceiver: StatsReceiver = new InMemoryStatsReceiver, + featureStoreAttributions: Seq[Attribution] = Seq.empty + ): EmbeddingProducer[EntityWithId[T]] = { + val featureStoreParams = FeatureStoreParams( + perDataset = Map( + dataset.id -> DatasetParams(datasetVersion = Some(version)) + ), + global = DatasetParams(attributions = featureStoreAttributions) + ) + val featureStoreClient = FeatureStoreClient( + BoundFeatureSet(boundFeature), + client, + statsReceiver, + featureStoreParams + ) + new FeatureStoreEmbeddingProducer(boundFeature, featureStoreClient) + } +} + +private[featurestore] class FeatureStoreEmbeddingProducer[T <: EntityId]( + boundFeature: BoundFeature[T, RawFloatTensor], + featureStoreClient: FeatureStoreClient) + extends EmbeddingProducer[EntityWithId[T]] { + // Looks up embedding from online feature store for an entity. + override def produceEmbedding(input: EntityWithId[T]): Stitch[Option[Embedding[Float]]] = { + val featureStoreRequest = FeatureStoreRequest( + entityIds = Seq(input) + ) + + Stitch.callFuture(featureStoreClient(featureStoreRequest).map { predictionRecord => + predictionRecord.getFeatureValue(boundFeature) match { + case Some(featureValue) => { + val embedding = EmbeddingSerDe.floatEmbeddingSerDe.fromThrift( + thriftscala.Embedding(Some(featureValue.value)) + ) + Some(embedding) + } + case _ => None + } + }) + } +} diff --git a/ann/src/main/scala/com/twitter/ann/file_store/BUILD b/ann/src/main/scala/com/twitter/ann/file_store/BUILD new file mode 100644 index 0000000000..1b7ca6f818 --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/file_store/BUILD @@ -0,0 +1,12 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/com/twitter/storehaus:core", + "ann/src/main/scala/com/twitter/ann/common", + "ann/src/main/thrift/com/twitter/ann/common:ann-common-scala", + "mediaservices/commons/src/main/scala", + ], +) diff --git a/ann/src/main/scala/com/twitter/ann/file_store/ReadableIndexIdFileStore.scala b/ann/src/main/scala/com/twitter/ann/file_store/ReadableIndexIdFileStore.scala new file mode 100644 index 0000000000..8886abaa8e --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/file_store/ReadableIndexIdFileStore.scala @@ -0,0 +1,35 @@ +package com.twitter.ann.file_store + +import com.twitter.ann.common.thriftscala.FileBasedIndexIdStore +import com.twitter.bijection.Injection +import com.twitter.mediaservices.commons.codec.{ArrayByteBufferCodec, ThriftByteBufferCodec} +import com.twitter.search.common.file.AbstractFile +import com.twitter.storehaus.ReadableStore +import java.nio.ByteBuffer + +object ReadableIndexIdFileStore { + + /** + * @param file : File path to read serialized long indexId <-> Id mapping from. + * @param injection: Injection to convert bytes to Id. + * @tparam V: Type of Id + * @return File based Readable Store + */ + def apply[V]( + file: AbstractFile, + injection: Injection[V, Array[Byte]] + ): ReadableStore[Long, V] = { + val codec = new ThriftByteBufferCodec(FileBasedIndexIdStore) + val store: Map[Long, V] = codec + .decode(loadFile(file)) + .indexIdMap + .getOrElse(Map.empty[Long, ByteBuffer]) + .toMap + .mapValues(value => injection.invert(ArrayByteBufferCodec.decode(value)).get) + ReadableStore.fromMap[Long, V](store) + } + + private[this] def loadFile(file: AbstractFile): ByteBuffer = { + ArrayByteBufferCodec.encode(file.getByteSource.read()) + } +} diff --git a/ann/src/main/scala/com/twitter/ann/file_store/WritableIndexIdFileStore.scala b/ann/src/main/scala/com/twitter/ann/file_store/WritableIndexIdFileStore.scala new file mode 100644 index 0000000000..cddf77d2c7 --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/file_store/WritableIndexIdFileStore.scala @@ -0,0 +1,71 @@ +package com.twitter.ann.file_store + +import com.twitter.ann.common.IndexOutputFile +import com.twitter.ann.common.thriftscala.FileBasedIndexIdStore +import com.twitter.bijection.Injection +import com.twitter.mediaservices.commons.codec.ArrayByteBufferCodec +import com.twitter.mediaservices.commons.codec.ThriftByteBufferCodec +import com.twitter.storehaus.Store +import com.twitter.util.Future +import java.util.concurrent.{ConcurrentHashMap => JConcurrentHashMap} +import scala.collection.JavaConverters._ + +object WritableIndexIdFileStore { + + /** + * @param injection: Injection to convert typed Id to bytes. + * @tparam V: Type of Id + * @return File based Writable Store + */ + def apply[V]( + injection: Injection[V, Array[Byte]] + ): WritableIndexIdFileStore[V] = { + new WritableIndexIdFileStore[V]( + new JConcurrentHashMap[Long, Option[V]], + injection + ) + } +} + +class WritableIndexIdFileStore[V] private ( + map: JConcurrentHashMap[Long, Option[V]], + injection: Injection[V, Array[Byte]]) + extends Store[Long, V] { + + private[this] val store = Store.fromJMap(map) + + override def get(k: Long): Future[Option[V]] = { + store.get(k) + } + + override def put(kv: (Long, Option[V])): Future[Unit] = { + store.put(kv) + } + + /** + * Serialize and store the mapping in thrift format + * @param file : File path to store serialized long indexId <-> Id mapping + */ + def save(file: IndexOutputFile): Unit = { + saveThrift(toThrift(), file) + } + + def getInjection: Injection[V, Array[Byte]] = injection + + private[this] def toThrift(): FileBasedIndexIdStore = { + val indexIdMap = map.asScala + .collect { + case (key, Some(value)) => (key, ArrayByteBufferCodec.encode(injection.apply(value))) + } + + FileBasedIndexIdStore(Some(indexIdMap)) + } + + private[this] def saveThrift(thriftObj: FileBasedIndexIdStore, file: IndexOutputFile): Unit = { + val codec = new ThriftByteBufferCodec(FileBasedIndexIdStore) + val bytes = ArrayByteBufferCodec.decode(codec.encode(thriftObj)) + val outputStream = file.getOutputStream() + outputStream.write(bytes) + outputStream.close() + } +} diff --git a/ann/src/main/scala/com/twitter/ann/hnsw/BUILD b/ann/src/main/scala/com/twitter/ann/hnsw/BUILD new file mode 100644 index 0000000000..9783b461df --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/hnsw/BUILD @@ -0,0 +1,23 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/org/mapdb", + "ann/src/main/java/com/twitter/ann/hnsw", + "ann/src/main/scala/com/twitter/ann/common", + "ann/src/main/scala/com/twitter/ann/serialization", + "ann/src/main/thrift/com/twitter/ann/common:ann-common-scala", + "mediaservices/commons/src/main/scala:futuretracker", + "src/java/com/twitter/common_internal/hadoop", + "src/java/com/twitter/search/common/file", + "src/scala/com/twitter/ml/api/embedding", + ], + exports = [ + "ann/src/main/scala/com/twitter/ann/common", + "src/java/com/twitter/common_internal/hadoop", + "src/java/com/twitter/search/common/file", + "src/scala/com/twitter/ml/api/embedding", + ], +) diff --git a/ann/src/main/scala/com/twitter/ann/hnsw/DistanceFunctionGenerator.scala b/ann/src/main/scala/com/twitter/ann/hnsw/DistanceFunctionGenerator.scala new file mode 100644 index 0000000000..a512fad1fb --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/hnsw/DistanceFunctionGenerator.scala @@ -0,0 +1,37 @@ +package com.twitter.ann.hnsw + +import com.twitter.ann.common.EmbeddingType.EmbeddingVector +import com.twitter.ann.common.{Cosine, Distance, InnerProduct, Metric} + +private[hnsw] object DistanceFunctionGenerator { + def apply[T, D <: Distance[D]]( + metric: Metric[D], + idToEmbeddingFn: (T) => EmbeddingVector + ): DistanceFunctionGenerator[T] = { + // Use InnerProduct for cosine and normalize the vectors before appending and querying. + val updatedMetric = metric match { + case Cosine => InnerProduct + case _ => metric + } + + val distFnIndex = new DistanceFunction[T, T] { + override def distance(id1: T, id2: T) = + updatedMetric.absoluteDistance( + idToEmbeddingFn(id1), + idToEmbeddingFn(id2) + ) + } + + val distFnQuery = new DistanceFunction[EmbeddingVector, T] { + override def distance(embedding: EmbeddingVector, id: T) = + updatedMetric.absoluteDistance(embedding, idToEmbeddingFn(id)) + } + + DistanceFunctionGenerator(distFnIndex, distFnQuery, metric == Cosine) + } +} + +private[hnsw] case class DistanceFunctionGenerator[T]( + index: DistanceFunction[T, T], + query: DistanceFunction[EmbeddingVector, T], + shouldNormalize: Boolean) diff --git a/ann/src/main/scala/com/twitter/ann/hnsw/Hnsw.scala b/ann/src/main/scala/com/twitter/ann/hnsw/Hnsw.scala new file mode 100644 index 0000000000..38666eb20a --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/hnsw/Hnsw.scala @@ -0,0 +1,183 @@ +package com.twitter.ann.hnsw + +import com.google.common.annotations.VisibleForTesting +import com.twitter.ann.common.EmbeddingType._ +import com.twitter.ann.common.Metric.toThrift +import com.twitter.ann.common._ +import com.twitter.ann.common.thriftscala.DistanceMetric +import com.twitter.ann.hnsw.HnswIndex.RandomProvider +import com.twitter.util.Future +import java.util.Random +import java.util.concurrent.ConcurrentHashMap +import java.util.concurrent.ThreadLocalRandom +import java.util.concurrent.locks.Lock +import java.util.concurrent.locks.ReentrantLock +import scala.collection.JavaConverters._ + +private[hnsw] object Hnsw { + private[hnsw] def apply[T, D <: Distance[D]]( + dimension: Int, + metric: Metric[D], + efConstruction: Int, + maxM: Int, + expectedElements: Int, + futurePool: ReadWriteFuturePool, + idEmbeddingMap: IdEmbeddingMap[T] + ): Hnsw[T, D] = { + val randomProvider = new RandomProvider { + override def get(): Random = ThreadLocalRandom.current() + } + val distFn = + DistanceFunctionGenerator(metric, (key: T) => idEmbeddingMap.get(key)) + val internalIndex = new HnswIndex[T, EmbeddingVector]( + distFn.index, + distFn.query, + efConstruction, + maxM, + expectedElements, + randomProvider + ) + new Hnsw[T, D]( + dimension, + metric, + internalIndex, + futurePool, + idEmbeddingMap, + distFn.shouldNormalize, + LockedAccess.apply(expectedElements) + ) + } +} + +private[hnsw] object LockedAccess { + protected[hnsw] def apply[T](expectedElements: Int): LockedAccess[T] = + DefaultLockedAccess(new ConcurrentHashMap[T, Lock](expectedElements)) + protected[hnsw] def apply[T](): LockedAccess[T] = + DefaultLockedAccess(new ConcurrentHashMap[T, Lock]()) +} + +private[hnsw] case class DefaultLockedAccess[T](locks: ConcurrentHashMap[T, Lock]) + extends LockedAccess[T] { + override def lockProvider(item: T) = locks.computeIfAbsent(item, (_: T) => new ReentrantLock()) +} + +private[hnsw] trait LockedAccess[T] { + protected def lockProvider(item: T): Lock + def lock[K](item: T)(fn: => K): K = { + val lock = lockProvider(item) + lock.lock() + try { + fn + } finally { + lock.unlock() + } + } +} + +@VisibleForTesting +private[hnsw] class Hnsw[T, D <: Distance[D]]( + dimension: Int, + metric: Metric[D], + hnswIndex: HnswIndex[T, EmbeddingVector], + readWriteFuturePool: ReadWriteFuturePool, + idEmbeddingMap: IdEmbeddingMap[T], + shouldNormalize: Boolean, + lockedAccess: LockedAccess[T] = LockedAccess.apply[T]()) + extends Appendable[T, HnswParams, D] + with Queryable[T, HnswParams, D] + with Updatable[T] { + override def append(entity: EntityEmbedding[T]): Future[Unit] = { + readWriteFuturePool.write { + val indexDimension = entity.embedding.length + assert( + toThrift(metric) == DistanceMetric.EditDistance || indexDimension == dimension, + s"Dimension mismatch for index(${indexDimension}) and embedding($dimension)" + ) + + lockedAccess.lock(entity.id) { + // To make this thread-safe, we are using ConcurrentHashMap#putIfAbsent underneath, + // so if there is a pre-existing item, put() will return something that is not null + val embedding = idEmbeddingMap.putIfAbsent(entity.id, updatedEmbedding(entity.embedding)) + + if (embedding == null) { // New element - insert into the index + hnswIndex.insert(entity.id) + } else { // Existing element - update the embedding and graph structure + throw new IllegalDuplicateInsertException( + "Append method does not permit duplicates (try using update method): " + entity.id) + } + } + } onFailure { e => + Future.exception(e) + } + } + + override def toQueryable: Queryable[T, HnswParams, D] = this + + override def query( + embedding: EmbeddingVector, + numOfNeighbours: Int, + runtimeParams: HnswParams + ): Future[List[T]] = { + queryWithDistance(embedding, numOfNeighbours, runtimeParams) + .map(_.map(_.neighbor)) + } + + override def queryWithDistance( + embedding: EmbeddingVector, + numOfNeighbours: Int, + runtimeParams: HnswParams + ): Future[List[NeighborWithDistance[T, D]]] = { + val indexDimension = embedding.length + assert( + toThrift(metric) == DistanceMetric.EditDistance || indexDimension == dimension, + s"Dimension mismatch for index(${indexDimension}) and embedding($dimension)" + ) + readWriteFuturePool.read { + hnswIndex + .searchKnn(updatedEmbedding(embedding), numOfNeighbours, runtimeParams.ef) + .asScala + .map { nn => + NeighborWithDistance( + nn.getItem, + metric.fromAbsoluteDistance(nn.getDistance) + ) + } + .toList + } + } + + private[this] def updatedEmbedding(embedding: EmbeddingVector): EmbeddingVector = { + if (shouldNormalize) { + MetricUtil.norm(embedding) + } else { + embedding + } + } + + def getIndex: HnswIndex[T, EmbeddingVector] = hnswIndex + def getDimen: Int = dimension + def getMetric: Metric[D] = metric + def getIdEmbeddingMap: IdEmbeddingMap[T] = idEmbeddingMap + override def update( + entity: EntityEmbedding[T] + ): Future[Unit] = { + readWriteFuturePool.write { + val indexDimension = entity.embedding.length + assert( + toThrift(metric) == DistanceMetric.EditDistance || indexDimension == dimension, + s"Dimension mismatch for index(${indexDimension}) and embedding($dimension)" + ) + + lockedAccess.lock(entity.id) { + val embedding = idEmbeddingMap.put(entity.id, updatedEmbedding(entity.embedding)) + if (embedding == null) { // New element - insert into the index + hnswIndex.insert(entity.id) + } else { // Existing element - update the embedding and graph structure + hnswIndex.reInsert(entity.id); + } + } + } onFailure { e => + Future.exception(e) + } + } +} diff --git a/ann/src/main/scala/com/twitter/ann/hnsw/HnswCommon.scala b/ann/src/main/scala/com/twitter/ann/hnsw/HnswCommon.scala new file mode 100644 index 0000000000..1c0a65fd20 --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/hnsw/HnswCommon.scala @@ -0,0 +1,62 @@ +package com.twitter.ann.hnsw + +import com.twitter.ann.common.RuntimeParams +import com.twitter.ann.common.thriftscala.HnswIndexMetadata +import com.twitter.ann.common.thriftscala.HnswRuntimeParam +import com.twitter.ann.common.thriftscala.{RuntimeParams => ServiceRuntimeParams} +import com.twitter.bijection.Injection +import com.twitter.mediaservices.commons.codec.ThriftByteBufferCodec +import com.twitter.search.common.file.AbstractFile +import scala.util.Failure +import scala.util.Success +import scala.util.Try + +object HnswCommon { + private[hnsw] lazy val MetadataCodec = new ThriftByteBufferCodec(HnswIndexMetadata) + private[hnsw] val MetaDataFileName = "hnsw_index_metadata" + private[hnsw] val EmbeddingMappingFileName = "hnsw_embedding_mapping" + private[hnsw] val InternalIndexDir = "hnsw_internal_index" + private[hnsw] val HnswInternalMetadataFileName = "hnsw_internal_metadata" + private[hnsw] val HnswInternalGraphFileName = "hnsw_internal_graph" + + val RuntimeParamsInjection: Injection[HnswParams, ServiceRuntimeParams] = + new Injection[HnswParams, ServiceRuntimeParams] { + override def apply(scalaParams: HnswParams): ServiceRuntimeParams = { + ServiceRuntimeParams.HnswParam( + HnswRuntimeParam( + scalaParams.ef + ) + ) + } + + override def invert(thriftParams: ServiceRuntimeParams): Try[HnswParams] = + thriftParams match { + case ServiceRuntimeParams.HnswParam(hnswParam) => + Success( + HnswParams(hnswParam.ef) + ) + case p => Failure(new IllegalArgumentException(s"Expected HnswRuntimeParam got $p")) + } + } + + def isValidHnswIndex(path: AbstractFile): Boolean = { + path.isDirectory && + path.hasSuccessFile && + path.getChild(MetaDataFileName).exists() && + path.getChild(EmbeddingMappingFileName).exists() && + path.getChild(InternalIndexDir).exists() && + path.getChild(InternalIndexDir).getChild(HnswInternalMetadataFileName).exists() && + path.getChild(InternalIndexDir).getChild(HnswInternalGraphFileName).exists() + } +} + +/** + * Hnsw runtime params + * @param ef: The size of the dynamic list for the nearest neighbors (used during the search). + * Higher ef leads to more accurate but slower search. + * ef cannot be set lower than the number of queried nearest neighbors k. + * The value ef of can be anything between k and the size of the dataset. + */ +case class HnswParams(ef: Int) extends RuntimeParams { + override def toString: String = s"HnswParams(ef = $ef)" +} diff --git a/ann/src/main/scala/com/twitter/ann/hnsw/HnswIOUtil.scala b/ann/src/main/scala/com/twitter/ann/hnsw/HnswIOUtil.scala new file mode 100644 index 0000000000..6d7a1d06ba --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/hnsw/HnswIOUtil.scala @@ -0,0 +1,106 @@ +package com.twitter.ann.hnsw + +import com.google.common.annotations.VisibleForTesting +import com.twitter.ann.common.EmbeddingType.EmbeddingVector +import com.twitter.ann.common.thriftscala.HnswIndexMetadata +import com.twitter.ann.common.Distance +import com.twitter.ann.common.EntityEmbedding +import com.twitter.ann.common.Metric +import com.twitter.ann.hnsw.HnswCommon._ +import com.twitter.ann.serialization.PersistedEmbeddingInjection +import com.twitter.ann.serialization.ThriftIteratorIO +import com.twitter.ann.serialization.thriftscala.PersistedEmbedding +import com.twitter.bijection.Injection +import com.twitter.mediaservices.commons.codec.ArrayByteBufferCodec +import com.twitter.search.common.file.AbstractFile +import java.io.BufferedInputStream +import java.io.BufferedOutputStream +import java.io.OutputStream + +private[hnsw] object HnswIOUtil { + private val BufferSize = 64 * 1024 // Default 64Kb + + @VisibleForTesting + private[hnsw] def loadEmbeddings[T]( + embeddingFile: AbstractFile, + injection: Injection[T, Array[Byte]], + idEmbeddingMap: IdEmbeddingMap[T], + ): IdEmbeddingMap[T] = { + val inputStream = { + val stream = embeddingFile.getByteSource.openStream() + if (stream.isInstanceOf[BufferedInputStream]) { + stream + } else { + new BufferedInputStream(stream, BufferSize) + } + } + + val thriftIteratorIO = + new ThriftIteratorIO[PersistedEmbedding](PersistedEmbedding) + val iterator = thriftIteratorIO.fromInputStream(inputStream) + val embeddingInjection = new PersistedEmbeddingInjection(injection) + try { + iterator.foreach { persistedEmbedding => + val embedding = embeddingInjection.invert(persistedEmbedding).get + idEmbeddingMap.putIfAbsent(embedding.id, embedding.embedding) + Unit + } + } finally { + inputStream.close() + } + idEmbeddingMap + } + + @VisibleForTesting + private[hnsw] def saveEmbeddings[T]( + stream: OutputStream, + injection: Injection[T, Array[Byte]], + iter: Iterator[(T, EmbeddingVector)] + ): Unit = { + val thriftIteratorIO = + new ThriftIteratorIO[PersistedEmbedding](PersistedEmbedding) + val embeddingInjection = new PersistedEmbeddingInjection(injection) + val iterator = iter.map { + case (id, emb) => + embeddingInjection(EntityEmbedding(id, emb)) + } + val outputStream = { + if (stream.isInstanceOf[BufferedOutputStream]) { + stream + } else { + new BufferedOutputStream(stream, BufferSize) + } + } + try { + thriftIteratorIO.toOutputStream(iterator, outputStream) + } finally { + outputStream.close() + } + } + + @VisibleForTesting + private[hnsw] def saveIndexMetadata( + dimension: Int, + metric: Metric[_ <: Distance[_]], + numElements: Int, + metadataStream: OutputStream + ): Unit = { + val metadata = HnswIndexMetadata( + dimension, + Metric.toThrift(metric), + numElements + ) + val bytes = ArrayByteBufferCodec.decode(MetadataCodec.encode(metadata)) + metadataStream.write(bytes) + metadataStream.close() + } + + @VisibleForTesting + private[hnsw] def loadIndexMetadata( + metadataFile: AbstractFile + ): HnswIndexMetadata = { + MetadataCodec.decode( + ArrayByteBufferCodec.encode(metadataFile.getByteSource.read()) + ) + } +} diff --git a/ann/src/main/scala/com/twitter/ann/hnsw/IdEmbeddingMap.scala b/ann/src/main/scala/com/twitter/ann/hnsw/IdEmbeddingMap.scala new file mode 100644 index 0000000000..e60f170883 --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/hnsw/IdEmbeddingMap.scala @@ -0,0 +1,13 @@ +package com.twitter.ann.hnsw + +import com.twitter.ann.common.EmbeddingType._ +import java.io.OutputStream + +trait IdEmbeddingMap[T] { + def putIfAbsent(id: T, embedding: EmbeddingVector): EmbeddingVector + def put(id: T, embedding: EmbeddingVector): EmbeddingVector + def get(id: T): EmbeddingVector + def iter(): Iterator[(T, EmbeddingVector)] + def size(): Int + def toDirectory(embeddingFileOutputStream: OutputStream): Unit +} diff --git a/ann/src/main/scala/com/twitter/ann/hnsw/JMapBasedIdEmbeddingMap.scala b/ann/src/main/scala/com/twitter/ann/hnsw/JMapBasedIdEmbeddingMap.scala new file mode 100644 index 0000000000..bc305dbc48 --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/hnsw/JMapBasedIdEmbeddingMap.scala @@ -0,0 +1,87 @@ +package com.twitter.ann.hnsw + +import com.twitter.ann.common.EmbeddingType.EmbeddingVector +import com.twitter.bijection.Injection +import com.twitter.search.common.file.AbstractFile +import java.io.OutputStream +import java.util.concurrent.ConcurrentHashMap +import scala.collection.JavaConverters._ + +private[hnsw] object JMapBasedIdEmbeddingMap { + + /** + * Creates in-memory concurrent hashmap based container that for storing id embedding mapping. + * @param expectedElements: Expected num of elements for sizing hint, need not be exact. + */ + def applyInMemory[T](expectedElements: Int): IdEmbeddingMap[T] = + new JMapBasedIdEmbeddingMap[T]( + new ConcurrentHashMap[T, EmbeddingVector](expectedElements), + Option.empty + ) + + /** + * Creates in-memory concurrent hashmap based container that can be serialized to disk for storing id embedding mapping. + * @param expectedElements: Expected num of elements for sizing hint, need not be exact. + * @param injection : Injection for typed Id T to Array[Byte] + */ + def applyInMemoryWithSerialization[T]( + expectedElements: Int, + injection: Injection[T, Array[Byte]] + ): IdEmbeddingMap[T] = + new JMapBasedIdEmbeddingMap[T]( + new ConcurrentHashMap[T, EmbeddingVector](expectedElements), + Some(injection) + ) + + /** + * Loads id embedding mapping in in-memory concurrent hashmap. + * @param embeddingFile: Local/Hdfs file path for embeddings + * @param injection : Injection for typed Id T to Array[Byte] + * @param numElements: Expected num of elements for sizing hint, need not be exact + */ + def loadInMemory[T]( + embeddingFile: AbstractFile, + injection: Injection[T, Array[Byte]], + numElements: Option[Int] = Option.empty + ): IdEmbeddingMap[T] = { + val map = numElements match { + case Some(elements) => new ConcurrentHashMap[T, EmbeddingVector](elements) + case None => new ConcurrentHashMap[T, EmbeddingVector]() + } + HnswIOUtil.loadEmbeddings( + embeddingFile, + injection, + new JMapBasedIdEmbeddingMap(map, Some(injection)) + ) + } +} + +private[this] class JMapBasedIdEmbeddingMap[T]( + map: java.util.concurrent.ConcurrentHashMap[T, EmbeddingVector], + injection: Option[Injection[T, Array[Byte]]]) + extends IdEmbeddingMap[T] { + override def putIfAbsent(id: T, embedding: EmbeddingVector): EmbeddingVector = { + map.putIfAbsent(id, embedding) + } + + override def put(id: T, embedding: EmbeddingVector): EmbeddingVector = { + map.put(id, embedding) + } + + override def get(id: T): EmbeddingVector = { + map.get(id) + } + + override def iter(): Iterator[(T, EmbeddingVector)] = + map + .entrySet() + .iterator() + .asScala + .map(e => (e.getKey, e.getValue)) + + override def size(): Int = map.size() + + override def toDirectory(embeddingFile: OutputStream): Unit = { + HnswIOUtil.saveEmbeddings(embeddingFile, injection.get, iter()) + } +} diff --git a/ann/src/main/scala/com/twitter/ann/hnsw/MapDbBasedIdEmbeddingMap.scala b/ann/src/main/scala/com/twitter/ann/hnsw/MapDbBasedIdEmbeddingMap.scala new file mode 100644 index 0000000000..620f0554ee --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/hnsw/MapDbBasedIdEmbeddingMap.scala @@ -0,0 +1,81 @@ +package com.twitter.ann.hnsw + +import com.twitter.ann.common.EmbeddingType.EmbeddingVector +import com.twitter.bijection.Injection +import com.twitter.ml.api.embedding.Embedding +import com.twitter.search.common.file.AbstractFile +import java.io.OutputStream +import org.mapdb.DBMaker +import org.mapdb.HTreeMap +import org.mapdb.Serializer +import scala.collection.JavaConverters._ + +/** + * This class currently only support querying and creates map db on fly from thrift serialized embedding mapping + * Implement index creation with this or altogether replace mapdb with some better performing solution as it takes a lot of time to create/query or precreate while serializing thrift embeddings + */ +private[hnsw] object MapDbBasedIdEmbeddingMap { + + /** + * Loads id embedding mapping in mapDB based container leveraging memory mapped files. + * @param embeddingFile: Local/Hdfs file path for embeddings + * @param injection : Injection for typed Id T to Array[Byte] + */ + def loadAsReadonly[T]( + embeddingFile: AbstractFile, + injection: Injection[T, Array[Byte]] + ): IdEmbeddingMap[T] = { + val diskDb = DBMaker + .tempFileDB() + .concurrencyScale(32) + .fileMmapEnable() + .fileMmapEnableIfSupported() + .fileMmapPreclearDisable() + .cleanerHackEnable() + .closeOnJvmShutdown() + .make() + + val mapDb = diskDb + .hashMap("mapdb", Serializer.BYTE_ARRAY, Serializer.FLOAT_ARRAY) + .createOrOpen() + + HnswIOUtil.loadEmbeddings( + embeddingFile, + injection, + new MapDbBasedIdEmbeddingMap(mapDb, injection) + ) + } +} + +private[this] class MapDbBasedIdEmbeddingMap[T]( + mapDb: HTreeMap[Array[Byte], Array[Float]], + injection: Injection[T, Array[Byte]]) + extends IdEmbeddingMap[T] { + override def putIfAbsent(id: T, embedding: EmbeddingVector): EmbeddingVector = { + val value = mapDb.putIfAbsent(injection.apply(id), embedding.toArray) + if (value == null) null else Embedding(value) + } + + override def put(id: T, embedding: EmbeddingVector): EmbeddingVector = { + val value = mapDb.put(injection.apply(id), embedding.toArray) + if (value == null) null else Embedding(value) + } + + override def get(id: T): EmbeddingVector = { + Embedding(mapDb.get(injection.apply(id))) + } + + override def iter(): Iterator[(T, EmbeddingVector)] = { + mapDb + .entrySet() + .iterator() + .asScala + .map(entry => (injection.invert(entry.getKey).get, Embedding(entry.getValue))) + } + + override def size(): Int = mapDb.size() + + override def toDirectory(embeddingFile: OutputStream): Unit = { + HnswIOUtil.saveEmbeddings(embeddingFile, injection, iter()) + } +} diff --git a/ann/src/main/scala/com/twitter/ann/hnsw/SerializableHnsw.scala b/ann/src/main/scala/com/twitter/ann/hnsw/SerializableHnsw.scala new file mode 100644 index 0000000000..65ffa0e5d7 --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/hnsw/SerializableHnsw.scala @@ -0,0 +1,196 @@ +package com.twitter.ann.hnsw + +import com.google.common.annotations.VisibleForTesting +import com.twitter.ann.common.EmbeddingType.EmbeddingVector +import com.twitter.ann.common._ +import com.twitter.ann.common.thriftscala.HnswIndexMetadata +import com.twitter.ann.hnsw.HnswCommon._ +import com.twitter.ann.hnsw.HnswIndex.RandomProvider +import com.twitter.bijection.Injection +import com.twitter.search.common.file.AbstractFile +import com.twitter.search.common.file.FileUtils +import com.twitter.util.Future +import java.io.IOException +import java.util.concurrent.ThreadLocalRandom +import java.util.Random +import org.apache.beam.sdk.io.fs.ResourceId + +private[hnsw] object SerializableHnsw { + private[hnsw] def apply[T, D <: Distance[D]]( + index: Hnsw[T, D], + injection: Injection[T, Array[Byte]] + ): SerializableHnsw[T, D] = { + new SerializableHnsw[T, D]( + index, + injection + ) + } + + private[hnsw] def loadMapBasedQueryableIndex[T, D <: Distance[D]]( + dimension: Int, + metric: Metric[D], + injection: Injection[T, Array[Byte]], + futurePool: ReadWriteFuturePool, + directory: AbstractFile + ): SerializableHnsw[T, D] = { + val metadata = HnswIOUtil.loadIndexMetadata(directory.getChild(MetaDataFileName)) + validateMetadata(dimension, metric, metadata) + val idEmbeddingMap = JMapBasedIdEmbeddingMap.loadInMemory( + directory.getChild(EmbeddingMappingFileName), + injection, + Some(metadata.numElements) + ) + loadIndex( + dimension, + metric, + injection, + futurePool, + directory, + idEmbeddingMap, + metadata + ) + } + + private[hnsw] def loadMMappedBasedQueryableIndex[T, D <: Distance[D]]( + dimension: Int, + metric: Metric[D], + injection: Injection[T, Array[Byte]], + futurePool: ReadWriteFuturePool, + directory: AbstractFile + ): SerializableHnsw[T, D] = { + val metadata = HnswIOUtil.loadIndexMetadata(directory.getChild(MetaDataFileName)) + validateMetadata(dimension, metric, metadata) + loadIndex( + dimension, + metric, + injection, + futurePool, + directory, + MapDbBasedIdEmbeddingMap + .loadAsReadonly(directory.getChild(EmbeddingMappingFileName), injection), + metadata + ) + } + + private[hnsw] def loadIndex[T, D <: Distance[D]]( + dimension: Int, + metric: Metric[D], + injection: Injection[T, Array[Byte]], + futurePool: ReadWriteFuturePool, + directory: AbstractFile, + idEmbeddingMap: IdEmbeddingMap[T], + metadata: HnswIndexMetadata + ): SerializableHnsw[T, D] = { + val distFn = + DistanceFunctionGenerator(metric, (key: T) => idEmbeddingMap.get(key)) + val randomProvider = new RandomProvider { + override def get(): Random = ThreadLocalRandom.current() + } + val internalIndex = HnswIndex.loadHnswIndex[T, EmbeddingVector]( + distFn.index, + distFn.query, + directory.getChild(InternalIndexDir), + injection, + randomProvider + ) + + val index = new Hnsw[T, D]( + dimension, + metric, + internalIndex, + futurePool, + idEmbeddingMap, + distFn.shouldNormalize, + LockedAccess.apply(metadata.numElements) + ) + + new SerializableHnsw(index, injection) + } + + private[this] def validateMetadata[D <: Distance[D]]( + dimension: Int, + metric: Metric[D], + existingMetadata: HnswIndexMetadata + ): Unit = { + assert( + existingMetadata.dimension == dimension, + s"Dimensions do not match. requested: $dimension existing: ${existingMetadata.dimension}" + ) + + val existingMetric = Metric.fromThrift(existingMetadata.distanceMetric) + assert( + existingMetric == metric, + s"DistanceMetric do not match. requested: $metric existing: $existingMetric" + ) + } +} + +@VisibleForTesting +private[hnsw] class SerializableHnsw[T, D <: Distance[D]]( + index: Hnsw[T, D], + injection: Injection[T, Array[Byte]]) + extends Appendable[T, HnswParams, D] + with Queryable[T, HnswParams, D] + with Serialization + with Updatable[T] { + override def append(entity: EntityEmbedding[T]) = index.append(entity) + + override def toQueryable: Queryable[T, HnswParams, D] = index.toQueryable + + override def query( + embedding: EmbeddingVector, + numOfNeighbours: Int, + runtimeParams: HnswParams + ) = index.query(embedding, numOfNeighbours, runtimeParams) + + override def queryWithDistance( + embedding: EmbeddingVector, + numOfNeighbours: Int, + runtimeParams: HnswParams + ) = index.queryWithDistance(embedding, numOfNeighbours, runtimeParams) + + def toDirectory(directory: ResourceId): Unit = { + toDirectory(new IndexOutputFile(directory)) + } + + def toDirectory(directory: AbstractFile): Unit = { + // Create a temp dir with time prefix, and then do a rename after serialization + val tmpDir = FileUtils.getTmpFileHandle(directory) + if (!tmpDir.exists()) { + tmpDir.mkdirs() + } + + toDirectory(new IndexOutputFile(tmpDir)) + + // Rename tmp dir to original directory supplied + if (!tmpDir.rename(directory)) { + throw new IOException(s"Failed to rename ${tmpDir.getPath} to ${directory.getPath}") + } + } + + private def toDirectory(indexFile: IndexOutputFile): Unit = { + // Save java based hnsw index + index.getIndex.toDirectory(indexFile.createDirectory(InternalIndexDir), injection) + + // Save index metadata + HnswIOUtil.saveIndexMetadata( + index.getDimen, + index.getMetric, + index.getIdEmbeddingMap.size(), + indexFile.createFile(MetaDataFileName).getOutputStream() + ) + + // Save embedding mapping + index.getIdEmbeddingMap.toDirectory( + indexFile.createFile(EmbeddingMappingFileName).getOutputStream()) + + // Create _SUCCESS file + indexFile.createSuccessFile() + } + + override def update( + entity: EntityEmbedding[T] + ): Future[Unit] = { + index.update(entity) + } +} diff --git a/ann/src/main/scala/com/twitter/ann/hnsw/TypedHnswIndex.scala b/ann/src/main/scala/com/twitter/ann/hnsw/TypedHnswIndex.scala new file mode 100644 index 0000000000..6bf99a61bd --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/hnsw/TypedHnswIndex.scala @@ -0,0 +1,173 @@ +package com.twitter.ann.hnsw + +import com.twitter.ann.common._ +import com.twitter.bijection.Injection +import com.twitter.search.common.file.AbstractFile + +// Class to provide HNSW based approximate nearest neighbour index +object TypedHnswIndex { + + /** + * Creates in-memory HNSW based index which supports querying/addition/updates of the entity embeddings. + * See https://docbird.twitter.biz/ann/hnsw.html to check information about arguments. + * + * @param dimension Dimension of the embedding to be indexed + * @param metric Distance metric (InnerProduct/Cosine/L2) + * @param efConstruction The parameter has the same meaning as ef, but controls the + * index_time/index_accuracy ratio. Bigger ef_construction leads to longer + * construction, but better index quality. At some point, increasing + * ef_construction does not improve the quality of the index. One way to + * check if the selection of ef_construction was ok is to measure a recall + * for M nearest neighbor search when ef = ef_constuction: if the recall is + * lower than 0.9, than there is room for improvement. + * @param maxM The number of bi-directional links created for every new element during construction. + * Reasonable range for M is 2-100. Higher M work better on datasets with high + * intrinsic dimensionality and/or high recall, while low M work better for datasets + * with low intrinsic dimensionality and/or low recalls. The parameter also determines + * the algorithm's memory consumption, bigger the param more the memory requirement. + * For high dimensional datasets (word embeddings, good face descriptors), higher M + * are required (e.g. M=48, 64) for optimal performance at high recall. + * The range M=12-48 is ok for the most of the use cases. + * @param expectedElements Approximate number of elements to be indexed + * @param readWriteFuturePool Future pool for performing read (query) and write operation (addition/updates). + * @tparam T Type of item to index + * @tparam D Type of distance + */ + def index[T, D <: Distance[D]]( + dimension: Int, + metric: Metric[D], + efConstruction: Int, + maxM: Int, + expectedElements: Int, + readWriteFuturePool: ReadWriteFuturePool + ): Appendable[T, HnswParams, D] with Queryable[T, HnswParams, D] with Updatable[T] = { + Hnsw[T, D]( + dimension, + metric, + efConstruction, + maxM, + expectedElements, + readWriteFuturePool, + JMapBasedIdEmbeddingMap.applyInMemory[T](expectedElements) + ) + } + + /** + * Creates in-memory HNSW based index which supports querying/addition/updates of the entity embeddings. + * It can be serialized to a directory (HDFS/Local file system) + * See https://docbird.twitter.biz/ann/hnsw.html to check information about arguments. + * + * @param dimension Dimension of the embedding to be indexed + * @param metric Distance metric (InnerProduct/Cosine/L2) + * @param efConstruction The parameter has the same meaning as ef, but controls the + * index_time/index_accuracy ratio. Bigger ef_construction leads to longer + * construction, but better index quality. At some point, increasing + * ef_construction does not improve the quality of the index. One way to + * check if the selection of ef_construction was ok is to measure a recall + * for M nearest neighbor search when ef = ef_constuction: if the recall is + * lower than 0.9, than there is room for improvement. + * @param maxM The number of bi-directional links created for every new element during construction. + * Reasonable range for M is 2-100. Higher M work better on datasets with high + * intrinsic dimensionality and/or high recall, while low M work better for datasets + * with low intrinsic dimensionality and/or low recalls. The parameter also determines + * the algorithm's memory consumption, bigger the param more the memory requirement. + * For high dimensional datasets (word embeddings, good face descriptors), higher M + * are required (e.g. M=48, 64) for optimal performance at high recall. + * The range M=12-48 is ok for the most of the use cases. + * @param expectedElements Approximate number of elements to be indexed + * @param injection Injection for typed Id T to Array[Byte] + * @param readWriteFuturePool Future pool for performing read (query) and write operation (addition/updates). + * @tparam T Type of item to index + * @tparam D Type of distance + */ + def serializableIndex[T, D <: Distance[D]]( + dimension: Int, + metric: Metric[D], + efConstruction: Int, + maxM: Int, + expectedElements: Int, + injection: Injection[T, Array[Byte]], + readWriteFuturePool: ReadWriteFuturePool + ): Appendable[T, HnswParams, D] + with Queryable[T, HnswParams, D] + with Updatable[T] + with Serialization = { + val index = Hnsw[T, D]( + dimension, + metric, + efConstruction, + maxM, + expectedElements, + readWriteFuturePool, + JMapBasedIdEmbeddingMap + .applyInMemoryWithSerialization[T](expectedElements, injection) + ) + + SerializableHnsw[T, D]( + index, + injection + ) + } + + /** + * Loads HNSW index from a directory to in-memory + * @param dimension dimension of the embedding to be indexed + * @param metric Distance metric + * @param readWriteFuturePool Future pool for performing read (query) and write operation (addition/updates). + * @param injection : Injection for typed Id T to Array[Byte] + * @param directory : Directory(HDFS/Local file system) where hnsw index is stored + * @tparam T : Type of item to index + * @tparam D : Type of distance + */ + def loadIndex[T, D <: Distance[D]]( + dimension: Int, + metric: Metric[D], + injection: Injection[T, Array[Byte]], + readWriteFuturePool: ReadWriteFuturePool, + directory: AbstractFile + ): Appendable[T, HnswParams, D] + with Queryable[T, HnswParams, D] + with Updatable[T] + with Serialization = { + SerializableHnsw.loadMapBasedQueryableIndex[T, D]( + dimension, + metric, + injection, + readWriteFuturePool, + directory + ) + } + + /** + * Loads a HNSW index from a directory and memory map it. + * It will take less memory but rely more on disk as it leverages memory mapped file backed by disk. + * Latency will go up considerably (Could be by factor of > 10x) if used on instance with low + * memory since lot of page faults may occur. Best use case to use would with scalding jobs + * where mapper/reducers instance are limited by 8gb memory. + * @param dimension dimension of the embedding to be indexed + * @param metric Distance metric + * @param readWriteFuturePool Future pool for performing read (query) and write operation (addition/updates). + * @param injection Injection for typed Id T to Array[Byte] + * @param directory Directory(HDFS/Local file system) where hnsw index is stored + * @tparam T Type of item to index + * @tparam D Type of distance + */ + def loadMMappedIndex[T, D <: Distance[D]]( + dimension: Int, + metric: Metric[D], + injection: Injection[T, Array[Byte]], + readWriteFuturePool: ReadWriteFuturePool, + directory: AbstractFile + ): Appendable[T, HnswParams, D] + with Queryable[T, HnswParams, D] + with Updatable[T] + with Serialization = { + SerializableHnsw.loadMMappedBasedQueryableIndex[T, D]( + dimension, + metric, + injection, + readWriteFuturePool, + directory + ) + } +} diff --git a/ann/src/main/scala/com/twitter/ann/manhattan/BUILD b/ann/src/main/scala/com/twitter/ann/manhattan/BUILD new file mode 100644 index 0000000000..6b0387d6f3 --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/manhattan/BUILD @@ -0,0 +1,12 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/com/twitter/bijection:core", + "3rdparty/jvm/com/twitter/bijection:scrooge", + "ann/src/main/scala/com/twitter/ann/common", + "src/scala/com/twitter/ml/api/embedding", + "storage/clients/manhattan", + ], +) diff --git a/ann/src/main/scala/com/twitter/ann/manhattan/ManhattanEmbeddingProducer.scala b/ann/src/main/scala/com/twitter/ann/manhattan/ManhattanEmbeddingProducer.scala new file mode 100644 index 0000000000..e8b2f64299 --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/manhattan/ManhattanEmbeddingProducer.scala @@ -0,0 +1,63 @@ +package com.twitter.ann.manhattan + +import com.twitter.ann.common.EmbeddingType.EmbeddingVector +import com.twitter.ann.common.{EmbeddingProducer, EmbeddingType} +import com.twitter.bijection.Injection +import com.twitter.ml.api.embedding.{EmbeddingBijection, EmbeddingSerDe} +import com.twitter.ml.api.{thriftscala => thrift} +import com.twitter.stitch.Stitch +import com.twitter.storage.client.manhattan.bijections.Bijections +import com.twitter.storage.client.manhattan.bijections.Bijections.BinaryScalaInjection +import com.twitter.storage.client.manhattan.kv.ManhattanKVEndpoint +import com.twitter.storage.client.manhattan.kv.impl.{ + DescriptorP1L0, + ReadOnlyKeyDescriptor, + ValueDescriptor +} + +private[manhattan] class ManhattanEmbeddingProducer[T]( + keyDescriptor: DescriptorP1L0.DKey[T], + valueDescriptor: ValueDescriptor.EmptyValue[EmbeddingVector], + manhattanEndpoint: ManhattanKVEndpoint) + extends EmbeddingProducer[T] { + + /** + * Lookup an embedding from manhattan given a key of type T. + * + * @return An embedding stitch. + * An easy way to get a Future from a Stitch is to run Stitch.run(stitch) + */ + override def produceEmbedding(input: T): Stitch[Option[EmbeddingVector]] = { + val fullKey = keyDescriptor.withPkey(input) + val stitchResult = manhattanEndpoint.get(fullKey, valueDescriptor) + stitchResult.map { resultOption => + resultOption.map(_.contents) + } + } +} + +object ManhattanEmbeddingProducer { + private[manhattan] def keyDescriptor[T]( + injection: Injection[T, Array[Byte]], + dataset: String + ): DescriptorP1L0.DKey[T] = + ReadOnlyKeyDescriptor(injection.andThen(Bijections.BytesBijection)) + .withDataset(dataset) + + private[manhattan] val EmbeddingDescriptor: ValueDescriptor.EmptyValue[ + EmbeddingType.EmbeddingVector + ] = { + val embeddingBijection = new EmbeddingBijection(EmbeddingSerDe.floatEmbeddingSerDe) + val thriftInjection = BinaryScalaInjection[thrift.Embedding](thrift.Embedding) + ValueDescriptor(embeddingBijection.andThen(thriftInjection)) + } + + def apply[T]( + dataset: String, + injection: Injection[T, Array[Byte]], + manhattanEndpoint: ManhattanKVEndpoint + ): EmbeddingProducer[T] = { + val descriptor = keyDescriptor(injection, dataset) + new ManhattanEmbeddingProducer(descriptor, EmbeddingDescriptor, manhattanEndpoint) + } +} diff --git a/ann/src/main/scala/com/twitter/ann/manhattan/README b/ann/src/main/scala/com/twitter/ann/manhattan/README new file mode 100644 index 0000000000..a79e6dab9a --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/manhattan/README @@ -0,0 +1,9 @@ +# Description + +The ManhattanEmbeddingProducer is an EmbeddingProducer that is backed by a static manhattan dataset. + +# Setting up Data + +Data needs to be setup correctly in manhattan in order to be able to read the data using the +ManhattanEmbeddingProducer. You can use the EmbeddingSamplingJob to do this. The job can reads +embedding data from HDFS and re-writes it in the manhattan data format on HDFS. \ No newline at end of file diff --git a/ann/src/main/scala/com/twitter/ann/scalding/benchmark/BUILD b/ann/src/main/scala/com/twitter/ann/scalding/benchmark/BUILD new file mode 100644 index 0000000000..4ad971d3dc --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/scalding/benchmark/BUILD @@ -0,0 +1,56 @@ +scala_library( + name = "benchmark", + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = [ + "bazel-compatible", + "bazel-compatible:migrated", + "bazel-only", + ], + dependencies = [ + ":user_item_knn-scala", + "3rdparty/src/jvm/com/twitter/scalding:args", + "3rdparty/src/jvm/com/twitter/scalding:core", + "3rdparty/src/jvm/com/twitter/scalding:date", + "ann/src/main/scala/com/twitter/ann/common", + "ann/src/main/scala/com/twitter/ann/scalding/offline", + "src/scala/com/twitter/scalding_internal/dalv2", + "src/scala/com/twitter/scalding_internal/job", + "src/scala/com/twitter/scalding_internal/job/analytics_batch", + "src/scala/com/twitter/scalding_internal/multiformat/format", + ], +) + +hadoop_binary( + name = "benchmark-adhoc", + main = "com.twitter.scalding.Tool", + platform = "java8", + runtime_platform = "java8", + tags = [ + "bazel-compatible", + "bazel-compatible:migrated", + "bazel-only", + ], + dependencies = [ + ":benchmark", + "3rdparty/jvm/org/slf4j:slf4j-jdk14", + ], +) + +create_datasets( + base_name = "user_item_knn", + description = "List of the top recommendations per search entity (user)", + java_schema = "com.twitter.ann.knn.thriftjava.Knn", + platform = "java8", + role = "cortex-mlx", + scala_schema = "com.twitter.ann.knn.thriftscala.Knn", + segment_type = "partitioned", + tags = ["bazel-compatible"], + java_dependencies = [ + "ann/src/main/thrift/com/twitter/ann/knn:thrift-java", + ], + scala_dependencies = [ + "ann/src/main/thrift/com/twitter/ann/knn:thrift-scala", + ], +) diff --git a/ann/src/main/scala/com/twitter/ann/scalding/benchmark/Knn.scala b/ann/src/main/scala/com/twitter/ann/scalding/benchmark/Knn.scala new file mode 100644 index 0000000000..72daf3fece --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/scalding/benchmark/Knn.scala @@ -0,0 +1,128 @@ +package com.twitter.ann.scalding.offline.com.twitter.ann.scalding.benchmark + +/* +This job will generate KNN ground truth based user and item embeddings. + */ + +import com.twitter.scalding.typed.TypedPipe +import com.twitter.scalding._ +import com.twitter.scalding_internal.dalv2.DALWrite.D +import com.twitter.ann.knn.thriftscala.Knn +import com.twitter.ann.knn.thriftscala.Neighbor +import com.twitter.ann.scalding.offline.IndexingStrategy +import com.twitter.ann.scalding.offline.KnnHelper +import com.twitter.ann.common.Distance +import com.twitter.ml.featurestore.lib.embedding.EmbeddingWithEntity +import com.twitter.cortex.ml.embeddings.common.EmbeddingFormatArgsParser +import com.twitter.cortex.ml.embeddings.common.EntityKind +import java.util.TimeZone +import com.twitter.scalding_internal.dalv2.DALWrite._ +import com.twitter.ann.scalding.benchmark.UserItemKnnScalaDataset +import com.twitter.scalding_internal.job.TwitterExecutionApp +import com.twitter.ml.featurestore.lib.EntityId +import com.twitter.ml.featurestore.lib.UserId + +/** + * This job will take consumer and item embeddings(either url or tweet) and output Knn entities (user id, (distance, item id)). + * + * Example command to run this adhoc job: + * + * scalding remote run \ + * --target ann/src/main/scala/com/twitter/ann/scalding/benchmark:benchmark-adhoc \ + * --hadoop-properties "mapreduce.map.memory.mb=8192 mapreduce.map.java.opts='-Xmx7618M' mapreduce.reduce.memory.mb=8192 mapreduce.reduce.java.opts='-Xmx7618M' mapred.task.timeout=0" \ + * --submitter hadoopnest3.smf1.twitter.com \ + * --user cortex-mlx \ + * --submitter-memory 8000.megabyte \ + * --main-class com.twitter.ann.scalding.offline.com.twitter.ann.scalding.benchmark.KnnJob -- \ + * --dalEnvironment Prod \ + * --search_space_entity_type user \ + * --user.feature_store_embedding ConsumerFollowEmbedding300Dataset \ + * --user.feature_store_major_version 1569196895 \ + * --user.date_range 2019-10-23 \ + * --search_space.feature_store_embedding ConsumerFollowEmbedding300Dataset \ + * --search_space.feature_store_major_version 1569196895 \ + * --search_space.date_range 2019-10-23 \ + * --date 2019-10-25 \ + * --version "consumer_follower_test" \ + * --reducers 10000 \ + * --num_of_random_groups 20 \ + * --num_replicas 1000 \ + * --indexing_strategy.metric InnerProduct \ + * --indexing_strategy.type hnsw \ + * --indexing_strategy.dimension 300 \ + * --indexing_strategy.ef_construction 30 \ + * --indexing_strategy.max_m 10 \ + * --indexing_strategy.ef_query 50 \ + * --search_space_shards 3000 \ + * --query_shards 3000 \ + * --search_space.read_sample_ratio 0.038 + */ +trait KnnJobBase { + val seed: Long = 123 + + def getKnnDataset[B <: EntityId, D <: Distance[D]]( + args: Args + )( + implicit uniqueID: UniqueID + ): TypedPipe[Knn] = { + + val consumerPipe: TypedPipe[EmbeddingWithEntity[UserId]] = EmbeddingFormatArgsParser.User + .getEmbeddingFormat(args, "user") + .getEmbeddings + + val itemPipe = EntityKind + .getEntityKind(args("search_space_entity_type")) + .parser + .getEmbeddingFormat(args, "search_space") + .getEmbeddings + + KnnHelper + // Refer to the documentation of findNearestNeighboursWithIndexingStrategy for more + // information about how to set these settings. + .findNearestNeighboursWithIndexingStrategy[UserId, B, D]( + queryEmbeddings = consumerPipe, + searchSpaceEmbeddings = itemPipe.asInstanceOf[TypedPipe[EmbeddingWithEntity[B]]], + numNeighbors = args.int("candidate_per_user", 20), + reducersOption = args.optional("reducers").map(_.toInt), + numOfSearchGroups = args.int("num_of_random_groups"), + numReplicas = args.int("num_replicas"), + indexingStrategy = IndexingStrategy.parse(args).asInstanceOf[IndexingStrategy[D]], + queryShards = args.optional("query_shards").map(_.toInt), + searchSpaceShards = args.optional("search_space_shards").map(_.toInt) + ) + .map { + case (user, items) => + val neighbors = items.map { + case (item, distance) => + Neighbor( + distance.distance, + item.toThrift + ) + } + Knn(user.toThrift, neighbors) + } + } +} + +object KnnJob extends TwitterExecutionApp with KnnJobBase { + + val KnnPathSuffix: String = "/user/cortex-mlx/qualatative_analysis/knn_ground_truth/" + val partitionKey: String = "version" + + override def job: Execution[Unit] = Execution.withId { implicit uniqueId => + Execution.getArgs.flatMap { args: Args => + implicit val timeZone: TimeZone = TimeZone.getDefault + implicit val dateParser: DateParser = DateParser.default + implicit val dateRange: DateRange = DateRange.parse(args.list("date"))(timeZone, dateParser) + + getKnnDataset(args).writeDALExecution( + UserItemKnnScalaDataset, + D.Daily, + D.Suffix(KnnPathSuffix), + D.Parquet, + Set(D.Partition(partitionKey, args("version"), D.PartitionType.String)) + ) + } + } + +} diff --git a/ann/src/main/scala/com/twitter/ann/scalding/offline/BUILD.bazel b/ann/src/main/scala/com/twitter/ann/scalding/offline/BUILD.bazel new file mode 100644 index 0000000000..eeb124aeba --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/scalding/offline/BUILD.bazel @@ -0,0 +1,44 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = [ + "bazel-compatible", + "bazel-only", + ], + dependencies = [ + "3rdparty/jvm/com/twitter/bijection:scrooge", + "3rdparty/src/jvm/com/twitter/scalding:args", + "3rdparty/src/jvm/com/twitter/scalding:commons", + "3rdparty/src/jvm/com/twitter/scalding:core", + "ann/src/main/scala/com/twitter/ann/brute_force", + "ann/src/main/scala/com/twitter/ann/common", + "ann/src/main/scala/com/twitter/ann/hnsw", + "ann/src/main/scala/com/twitter/ann/util", + "cortex-core/entity-embeddings/src/thrift/com/twitter/entityembeddings/neighbors:embeddings-knn-thrift-scala", + "src/scala/com/twitter/cortex/ml/embeddings/common:Helpers-deploy", + "src/scala/com/twitter/pluck/source/core_workflows/user_model:condensed_user_state-scala", + "src/scala/com/twitter/scalding_internal/dalv2", + "src/scala/com/twitter/scalding_internal/job", + "src/scala/com/twitter/scalding_internal/multiformat/format", + "src/scala/com/twitter/scalding_internal/parquet_thrift", + "usersource/snapshot/src/main/scala/com/twitter/usersource/snapshot/flat:usersource_flat-scala", + "usersource/snapshot/src/main/thrift/com/twitter/usersource/snapshot/flat:flat-scala", + ], +) + +hadoop_binary( + name = "ann-offline-deploy", + main = "com.twitter.scalding.Tool", + platform = "java8", + runtime_platform = "java8", + tags = [ + "bazel-compatible", + "bazel-compatible:migrated", + "bazel-only", + ], + dependencies = [ + ":offline", + "3rdparty/jvm/org/slf4j:slf4j-jdk14", + ], +) diff --git a/ann/src/main/scala/com/twitter/ann/scalding/offline/IndexingStrategy.scala b/ann/src/main/scala/com/twitter/ann/scalding/offline/IndexingStrategy.scala new file mode 100644 index 0000000000..cde35ccfb3 --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/scalding/offline/IndexingStrategy.scala @@ -0,0 +1,116 @@ +package com.twitter.ann.scalding.offline + +import com.twitter.ann.brute_force.{BruteForceIndex, BruteForceRuntimeParams} +import com.twitter.ann.common.{Distance, EntityEmbedding, Metric, ReadWriteFuturePool} +import com.twitter.ann.hnsw.{HnswParams, TypedHnswIndex} +import com.twitter.ann.util.IndexBuilderUtils +import com.twitter.scalding.Args +import com.twitter.util.logging.Logger +import com.twitter.util.{Await, FuturePool} + +/** + * IndexingStrategy is used for determining how we will build the index when doing a KNN in + * scalding. Right now there are 2 strategies a BruteForce and HNSW strategy. + * @tparam D distance that the index uses. + */ +sealed trait IndexingStrategy[D <: Distance[D]] { + private[offline] def buildIndex[T]( + indexItems: TraversableOnce[EntityEmbedding[T]] + ): ParameterlessQueryable[T, _, D] +} + +object IndexingStrategy { + + /** + * Parse an indexing strategy from scalding args. + * ${argumentName}.type Is hsnw or brute_force + * ${argumentName}.type is the metric to use. See Metric.fromString for options. + * + * hsnw has these additional parameters: + * ${argumentName}.dimension the number of dimension for the embeddings. + * ${argumentName}.ef_construction, ${argumentName}.ef_construction and ${argumentName}.ef_query. + * See TypedHnswIndex for more details on these parameters. + * @param args scalding arguments to parse. + * @param argumentName A specifier to use in case you want to parse more than one indexing + * strategy. indexing_strategy by default. + * @return parse indexing strategy + */ + def parse( + args: Args, + argumentName: String = "indexing_strategy" + ): IndexingStrategy[_] = { + def metricArg[D <: Distance[D]] = + Metric.fromString(args(s"$argumentName.metric")).asInstanceOf[Metric[D]] + + args(s"$argumentName.type") match { + case "brute_force" => + BruteForceIndexingStrategy(metricArg) + case "hnsw" => + val dimensionArg = args.int(s"$argumentName.dimension") + val efConstructionArg = args.int(s"$argumentName.ef_construction") + val maxMArg = args.int(s"$argumentName.max_m") + val efQuery = args.int(s"$argumentName.ef_query") + HnswIndexingStrategy( + dimension = dimensionArg, + metric = metricArg, + efConstruction = efConstructionArg, + maxM = maxMArg, + hnswParams = HnswParams(efQuery) + ) + } + } +} + +case class BruteForceIndexingStrategy[D <: Distance[D]](metric: Metric[D]) + extends IndexingStrategy[D] { + private[offline] def buildIndex[T]( + indexItems: TraversableOnce[EntityEmbedding[T]] + ): ParameterlessQueryable[T, _, D] = { + val appendable = BruteForceIndex[T, D](metric, FuturePool.immediatePool) + indexItems.foreach { item => + Await.result(appendable.append(item)) + } + val queryable = appendable.toQueryable + ParameterlessQueryable[T, BruteForceRuntimeParams.type, D]( + queryable, + BruteForceRuntimeParams + ) + } +} + +case class HnswIndexingStrategy[D <: Distance[D]]( + dimension: Int, + metric: Metric[D], + efConstruction: Int, + maxM: Int, + hnswParams: HnswParams, + concurrencyLevel: Int = 1) + extends IndexingStrategy[D] { + private[offline] def buildIndex[T]( + indexItems: TraversableOnce[EntityEmbedding[T]] + ): ParameterlessQueryable[T, _, D] = { + + val log: Logger = Logger(getClass) + val appendable = TypedHnswIndex.index[T, D]( + dimension = dimension, + metric = metric, + efConstruction = efConstruction, + maxM = maxM, + // This is not really that important. + expectedElements = 1000, + readWriteFuturePool = ReadWriteFuturePool(FuturePool.immediatePool) + ) + val future = + IndexBuilderUtils + .addToIndex(appendable, indexItems.toStream, concurrencyLevel) + .map { numberUpdates => + log.info(s"Performed $numberUpdates updates") + } + Await.result(future) + val queryable = appendable.toQueryable + ParameterlessQueryable( + queryable, + hnswParams + ) + } +} diff --git a/ann/src/main/scala/com/twitter/ann/scalding/offline/KnnDebug.scala b/ann/src/main/scala/com/twitter/ann/scalding/offline/KnnDebug.scala new file mode 100644 index 0000000000..c0b4ae5989 --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/scalding/offline/KnnDebug.scala @@ -0,0 +1,117 @@ +package com.twitter.ann.scalding.offline + +import com.twitter.core_workflows.user_model.thriftscala.CondensedUserState +import com.twitter.cortex.ml.embeddings.common.{DataSourceManager, GraphEdge, Helpers, UserKind} +import com.twitter.ml.featurestore.lib.UserId +import com.twitter.entityembeddings.neighbors.thriftscala.{EntityKey, NearestNeighbors} +import com.twitter.pluck.source.core_workflows.user_model.CondensedUserStateScalaDataset +import com.twitter.scalding._ +import com.twitter.scalding.typed.TypedPipe +import com.twitter.scalding_internal.dalv2.DAL +import com.twitter.usersource.snapshot.flat.UsersourceFlatScalaDataset +import com.twitter.usersource.snapshot.flat.thriftscala.FlatUser + +case class ConsumerAssoc(consumerId: UserId, assoc: List[String]) + +object KnnDebug { + + def getConsumerAssociations( + graph: TypedPipe[GraphEdge[UserId, UserId]], + usernames: TypedPipe[(UserId, String)], + reducers: Int + ): TypedPipe[ConsumerAssoc] = { + graph + .groupBy(_.itemId) + .join(usernames).withReducers(reducers) + .values + .map { + case (edge: GraphEdge[UserId, UserId], producerScreenName: String) => + ConsumerAssoc(consumerId = edge.consumerId, assoc = List(producerScreenName)) + } + .groupBy(_.consumerId).withReducers(reducers) + .reduce[ConsumerAssoc] { + case (uFollow1: ConsumerAssoc, uFollow2: ConsumerAssoc) => + ConsumerAssoc(consumerId = uFollow1.consumerId, assoc = uFollow1.assoc ++ uFollow2.assoc) + } + .values + } + + /** + * Write the neighbors and a set of follows to a tsv for easier analysis during debugging + * We take the set of users with between 25-50 follows and grab only those users + * + * This returns 4 strings of the form: + * consumerId, state, followUserNamefollowUserNamefollowUserName, neighborNameneighborNameneighborName + */ + def getDebugTable( + neighborsPipe: TypedPipe[(EntityKey, NearestNeighbors)], + shards: Int, + reducers: Int, + limit: Int = 10000, + userDataset: Option[TypedPipe[FlatUser]] = None, + followDataset: Option[TypedPipe[GraphEdge[UserId, UserId]]] = None, + consumerStatesDataset: Option[TypedPipe[CondensedUserState]] = None, + minFollows: Int = 25, + maxFollows: Int = 50 + )( + implicit dateRange: DateRange + ): TypedPipe[(String, String, String, String)] = { + + val usersourcePipe: TypedPipe[FlatUser] = userDataset + .getOrElse(DAL.readMostRecentSnapshot(UsersourceFlatScalaDataset, dateRange).toTypedPipe) + + val followGraph: TypedPipe[GraphEdge[UserId, UserId]] = followDataset + .getOrElse(new DataSourceManager().getFollowGraph()) + + val consumerStates: TypedPipe[CondensedUserState] = consumerStatesDataset + .getOrElse(DAL.read(CondensedUserStateScalaDataset).toTypedPipe) + + val usernames: TypedPipe[(UserId, String)] = usersourcePipe.flatMap { flatUser => + (flatUser.screenName, flatUser.id) match { + case (Some(name: String), Some(userId: Long)) => Some((UserId(userId), name)) + case _ => None + } + }.fork + + val consumerFollows: TypedPipe[ConsumerAssoc] = + getConsumerAssociations(followGraph, usernames, reducers) + .filter { uFollow => (uFollow.assoc.size > minFollows && uFollow.assoc.size < maxFollows) } + + val neighborGraph: TypedPipe[GraphEdge[UserId, UserId]] = neighborsPipe + .limit(limit) + .flatMap { + case (entityKey: EntityKey, neighbors: NearestNeighbors) => + Helpers.optionalToLong(entityKey.id) match { + case Some(entityId: Long) => + neighbors.neighbors.flatMap { neighbor => + Helpers + .optionalToLong(neighbor.neighbor.id) + .map { neighborId => + GraphEdge[UserId, UserId]( + consumerId = UserId(entityId), + itemId = UserId(neighborId), + weight = 1.0F) + } + } + case None => List() + } + } + val consumerNeighbors: TypedPipe[ConsumerAssoc] = + getConsumerAssociations(neighborGraph, usernames, reducers) + + consumerFollows + .groupBy(_.consumerId) + .join(consumerStates.groupBy { consumer => UserId(consumer.uid) }).withReducers(reducers) + .join(consumerNeighbors.groupBy(_.consumerId)).withReducers(reducers) + .values + .map { + case ((uFollow: ConsumerAssoc, state: CondensedUserState), uNeighbors: ConsumerAssoc) => + ( + UserKind.stringInjection(uFollow.consumerId), + state.state.toString, + uFollow.assoc mkString "", + uNeighbors.assoc mkString "") + } + .shard(shards) + } +} diff --git a/ann/src/main/scala/com/twitter/ann/scalding/offline/KnnEntityRecoDebugJob.scala b/ann/src/main/scala/com/twitter/ann/scalding/offline/KnnEntityRecoDebugJob.scala new file mode 100644 index 0000000000..9a6849d46f --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/scalding/offline/KnnEntityRecoDebugJob.scala @@ -0,0 +1,91 @@ +package com.twitter.ann.scalding.offline +import com.twitter.ann.common.Distance +import com.twitter.ann.common.Metric +import com.twitter.cortex.ml.embeddings.common.EntityKind +import com.twitter.ml.featurestore.lib.EntityId +import com.twitter.scalding.typed.TypedPipe +import com.twitter.scalding._ +import com.twitter.scalding_internal.job.TwitterExecutionApp + +/** + * This job do an exhaustive search for nearest neighbours helpful for debugging recommendations + * for a given list of sample queryIds and entity embeddings for the recos to be made. + * Sample job script: + ./bazel bundle ann/src/main/scala/com/twitter/ann/scalding/offline:ann-offline-deploy + + oscar hdfs \ + --screen --tee log.txt \ + --hadoop-client-memory 6000 \ + --hadoop-properties "yarn.app.mapreduce.am.resource.mb=6000;yarn.app.mapreduce.am.command-opts='-Xmx7500m';mapreduce.map.memory.mb=7500;mapreduce.reduce.java.opts='-Xmx6000m';mapreduce.reduce.memory.mb=7500;mapred.task.timeout=36000000;" \ + --bundle ann-offline-deploy \ + --min-split-size 284217728 \ + --host hadoopnest1.smf1.twitter.com \ + --tool com.twitter.ann.scalding.offline.KnnEntityRecoDebugJob -- \ + --neighbors 10 \ + --metric InnerProduct \ + --query_entity_kind user \ + --search_space_entity_kind user \ + --query.embedding_path /user/apoorvs/sample_embeddings \ + --query.embedding_format tab \ + --search_space.embedding_path /user/apoorvs/sample_embeddings \ + --search_space.embedding_format tab \ + --query_ids 974308319300149248 988871266244464640 2719685122 2489777564 \ + --output_path /user/apoorvs/adhochadoop/test \ + --reducers 100 + */ +object KnnEntityRecoDebugJob extends TwitterExecutionApp { + override def job: Execution[Unit] = Execution.withId { implicit uniqueId => + Execution.getArgs.flatMap { args: Args => + val queryEntityKind = EntityKind.getEntityKind(args("query_entity_kind")) + val searchSpaceEntityKind = EntityKind.getEntityKind(args("search_space_entity_kind")) + val metric = Metric.fromString(args("metric")) + run(queryEntityKind, searchSpaceEntityKind, metric, args) + } + } + + private[this] def run[A <: EntityId, B <: EntityId, D <: Distance[D]]( + uncastQueryEntityKind: EntityKind[_], + uncastSearchSpaceEntityKind: EntityKind[_], + uncastMetric: Metric[_], + args: Args + )( + implicit uniqueID: UniqueID + ): Execution[Unit] = { + import KnnHelper._ + + val numNeighbors = args.int("neighbors") + val reducers = args.getOrElse("reducers", "100").toInt + + val queryEntityKind = uncastQueryEntityKind.asInstanceOf[EntityKind[A]] + val searchSpaceEntityKind = uncastSearchSpaceEntityKind.asInstanceOf[EntityKind[B]] + val metric = uncastMetric.asInstanceOf[Metric[D]] + + // Filter the query entity embeddings with the queryIds + val queryIds = args.list("query_ids") + assert(queryIds.nonEmpty) + val filterQueryIds: TypedPipe[A] = TypedPipe + .from(queryIds) + .map(queryEntityKind.stringInjection.invert(_).get) + val queryEmbeddings = queryEntityKind.parser.getEmbeddingFormat(args, "query").getEmbeddings + + // Get the neighbour embeddings + val searchSpaceEmbeddings = + searchSpaceEntityKind.parser.getEmbeddingFormat(args, "search_space").getEmbeddings + + val nearestNeighborString = findNearestNeighbours( + queryEmbeddings, + searchSpaceEmbeddings, + metric, + numNeighbors, + Some(filterQueryIds), + reducers + )(queryEntityKind.ordering, uniqueID).map( + nearestNeighborsToString(_, queryEntityKind, searchSpaceEntityKind) + ) + + // Write the nearest neighbor string to one part file. + nearestNeighborString + .shard(1) + .writeExecution(TypedTsv(args("output_path"))) + } +} diff --git a/ann/src/main/scala/com/twitter/ann/scalding/offline/KnnHelper.scala b/ann/src/main/scala/com/twitter/ann/scalding/offline/KnnHelper.scala new file mode 100644 index 0000000000..fe12fe1561 --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/scalding/offline/KnnHelper.scala @@ -0,0 +1,438 @@ +package com.twitter.ann.scalding.offline + +import com.twitter.ann.common._ +import com.twitter.ann.hnsw.{HnswParams, TypedHnswIndex} +import com.twitter.bijection.Injection +import com.twitter.cortex.ml.embeddings.common.{EntityKind, Helpers, UserKind} +import com.twitter.entityembeddings.neighbors.thriftscala.{EntityKey, NearestNeighbors, Neighbor} +import com.twitter.ml.api.embedding.Embedding +import com.twitter.ml.api.embedding.EmbeddingMath.{Float => math} +import com.twitter.ml.featurestore.lib.embedding.EmbeddingWithEntity +import com.twitter.ml.featurestore.lib.{EntityId, UserId} +import com.twitter.scalding.typed.{TypedPipe, UnsortedGrouped} +import com.twitter.scalding.{Args, DateRange, Stat, TextLine, UniqueID} +import com.twitter.search.common.file.AbstractFile +import com.twitter.util.{Await, FuturePool} +import scala.util.Random + +case class Index[T, D <: Distance[D]]( + injection: Injection[T, Array[Byte]], + metric: Metric[D], + dimension: Int, + directory: AbstractFile) { + lazy val annIndex = TypedHnswIndex.loadIndex[T, D]( + dimension, + metric, + injection, + ReadWriteFuturePool(FuturePool.immediatePool), + directory + ) +} + +object KnnHelper { + def getFilteredUserEmbeddings( + args: Args, + filterPath: Option[String], + reducers: Int, + useHashJoin: Boolean + )( + implicit dateRange: DateRange + ): TypedPipe[EmbeddingWithEntity[UserId]] = { + val userEmbeddings: TypedPipe[EmbeddingWithEntity[UserId]] = + UserKind.parser.getEmbeddingFormat(args, "consumer").getEmbeddings + filterPath match { + case Some(fileName: String) => + val filterUserIds: TypedPipe[UserId] = TypedPipe + .from(TextLine(fileName)) + .flatMap { idLine => + Helpers.optionalToLong(idLine) + } + .map { id => + UserId(id) + } + Helpers + .adjustableJoin( + left = userEmbeddings.groupBy(_.entityId), + right = filterUserIds.asKeys, + useHashJoin = useHashJoin, + reducers = Some(reducers) + ).map { + case (_, (embedding, _)) => embedding + } + case None => userEmbeddings + } + } + + def getNeighborsPipe[T <: EntityId, D <: Distance[D]]( + args: Args, + uncastEntityKind: EntityKind[_], + uncastMetric: Metric[_], + ef: Int, + consumerEmbeddings: TypedPipe[EmbeddingWithEntity[UserId]], + abstractFile: Option[AbstractFile], + reducers: Int, + numNeighbors: Int, + dimension: Int + )( + implicit dateRange: DateRange + ): TypedPipe[(EntityKey, NearestNeighbors)] = { + val entityKind = uncastEntityKind.asInstanceOf[EntityKind[T]] + val injection = entityKind.byteInjection + val metric = uncastMetric.asInstanceOf[Metric[D]] + abstractFile match { + case Some(directory: AbstractFile) => + val index = Index(injection, metric, dimension, directory) + consumerEmbeddings + .map { embedding => + val knn = Await.result( + index.annIndex.queryWithDistance( + Embedding(embedding.embedding.toArray), + numNeighbors, + HnswParams(ef) + ) + ) + val neighborList = knn + .filter(_.neighbor.toString != embedding.entityId.userId.toString) + .map(nn => + Neighbor( + neighbor = EntityKey(nn.neighbor.toString), + similarity = Some(1 - nn.distance.distance))) + EntityKey(embedding.entityId.toString) -> NearestNeighbors(neighborList) + } + case None => + val producerEmbeddings: TypedPipe[EmbeddingWithEntity[UserId]] = + UserKind.parser.getEmbeddingFormat(args, "producer").getEmbeddings + + bruteForceNearestNeighbors( + consumerEmbeddings, + producerEmbeddings, + numNeighbors, + reducers + ) + } + } + + def bruteForceNearestNeighbors( + consumerEmbeddings: TypedPipe[EmbeddingWithEntity[UserId]], + producerEmbeddings: TypedPipe[EmbeddingWithEntity[UserId]], + numNeighbors: Int, + reducers: Int + ): TypedPipe[(EntityKey, NearestNeighbors)] = { + consumerEmbeddings + .cross(producerEmbeddings) + .map { + case (cEmbed: EmbeddingWithEntity[UserId], pEmbed: EmbeddingWithEntity[UserId]) => + // Cosine similarity + val cEmbedNorm = math.l2Norm(cEmbed.embedding).toFloat + val pEmbedNorm = math.l2Norm(pEmbed.embedding).toFloat + val distance: Float = -math.dotProduct( + (math.scalarProduct(cEmbed.embedding, 1 / cEmbedNorm)), + math.scalarProduct(pEmbed.embedding, 1 / pEmbedNorm)) + ( + UserKind.stringInjection(cEmbed.entityId), + (distance, UserKind.stringInjection(pEmbed.entityId))) + } + .groupBy(_._1).withReducers(reducers) + .sortWithTake(numNeighbors) { + case ((_: String, (sim1: Float, _: String)), (_: String, (sim2: Float, _: String))) => + sim1 < sim2 + } + .map { + case (consumerId: String, (prodSims: Seq[(String, (Float, String))])) => + EntityKey(consumerId) -> NearestNeighbors( + prodSims.map { + case (consumerId: String, (sim: Float, prodId: String)) => + Neighbor(neighbor = EntityKey(prodId), similarity = Some(-sim.toDouble)) + } + ) + } + } + + /** + * Calculate the nearest neighbors exhaustively between two entity embeddings using one as query and other as the search space. + * @param queryEmbeddings entity embeddings for queries + * @param searchSpaceEmbeddings entity embeddings for search space + * @param metric distance metric + * @param numNeighbors number of neighbors + * @param queryIdsFilter optional query ids to filter to query entity embeddings + * @param reducers number of reducers for grouping + * @param isSearchSpaceLarger Used for optimization: Is the search space larger than the query space? Ignored if numOfSearchGroups > 1. + * @param numOfSearchGroups we divide the search space into these groups (randomly). Useful when the search space is too large. Overrides isSearchSpaceLarger. + * @param numReplicas Each search group will be responsible for 1/numReplicas queryEmebeddings. + * This might speed up the search when the size of the index embeddings is + * large. + * @tparam A type of query entity + * @tparam B type of search space entity + * @tparam D type of distance + */ + def findNearestNeighbours[A <: EntityId, B <: EntityId, D <: Distance[D]]( + queryEmbeddings: TypedPipe[EmbeddingWithEntity[A]], + searchSpaceEmbeddings: TypedPipe[EmbeddingWithEntity[B]], + metric: Metric[D], + numNeighbors: Int = 10, + queryIdsFilter: Option[TypedPipe[A]] = Option.empty, + reducers: Int = 100, + mappers: Int = 100, + isSearchSpaceLarger: Boolean = true, + numOfSearchGroups: Int = 1, + numReplicas: Int = 1, + useCounters: Boolean = true + )( + implicit ordering: Ordering[A], + uid: UniqueID + ): TypedPipe[(A, Seq[(B, D)])] = { + val filteredQueryEmbeddings = queryIdsFilter match { + case Some(filter) => { + queryEmbeddings.groupBy(_.entityId).hashJoin(filter.asKeys).map { + case (x, (embedding, _)) => embedding + } + } + case None => queryEmbeddings + } + + if (numOfSearchGroups > 1) { + val indexingStrategy = BruteForceIndexingStrategy(metric) + findNearestNeighboursWithIndexingStrategy( + queryEmbeddings, + searchSpaceEmbeddings, + numNeighbors, + numOfSearchGroups, + indexingStrategy, + numReplicas, + Some(reducers), + useCounters = useCounters + ) + } else { + findNearestNeighboursViaCross( + filteredQueryEmbeddings, + searchSpaceEmbeddings, + metric, + numNeighbors, + reducers, + mappers, + isSearchSpaceLarger) + } + } + + /** + * Calculate the nearest neighbors using the specified indexing strategy between two entity + * embeddings using one as query and other as the search space. + * @param queryEmbeddings entity embeddings for queries + * @param searchSpaceEmbeddings entity embeddings for search space. You should be able to fit + * searchSpaceEmbeddings.size / numOfSearchGroups into memory. + * @param numNeighbors number of neighbors + * @param reducersOption number of reducers for the final sortedTake. + * @param numOfSearchGroups we divide the search space into these groups (randomly). Useful when + * the search space is too large. Search groups are shards. Choose this + * number by ensuring searchSpaceEmbeddings.size / numOfSearchGroups + * embeddings will fit into memory. + * @param numReplicas Each search group will be responsible for 1/numReplicas queryEmebeddings. + * By increasing this number, we can parallelize the work and reduce end to end + * running times. + * @param indexingStrategy How we will search for nearest neighbors within a search group + * @param queryShards one step we have is to fan out the query embeddings. We create one entry + * per search group. If numOfSearchGroups is large, then this fan out can take + * a long time. You can shard the query shard first to parallelize this + * process. One way to estimate what value to use: + * queryEmbeddings.size * numOfSearchGroups / queryShards should be around 1GB. + * @param searchSpaceShards this param is similar to queryShards. Except it shards the search + * space when numReplicas is too large. One way to estimate what value + * to use: searchSpaceEmbeddings.size * numReplicas / searchSpaceShards + * should be around 1GB. + * @tparam A type of query entity + * @tparam B type of search space entity + * @tparam D type of distance + * @return a pipe keyed by the index embedding. The values are the list of numNeighbors nearest + * neighbors along with their distances. + */ + def findNearestNeighboursWithIndexingStrategy[A <: EntityId, B <: EntityId, D <: Distance[D]]( + queryEmbeddings: TypedPipe[EmbeddingWithEntity[A]], + searchSpaceEmbeddings: TypedPipe[EmbeddingWithEntity[B]], + numNeighbors: Int, + numOfSearchGroups: Int, + indexingStrategy: IndexingStrategy[D], + numReplicas: Int = 1, + reducersOption: Option[Int] = None, + queryShards: Option[Int] = None, + searchSpaceShards: Option[Int] = None, + useCounters: Boolean = true + )( + implicit ordering: Ordering[A], + uid: UniqueID + ): UnsortedGrouped[A, Seq[(B, D)]] = { + + implicit val ord: Ordering[NNKey] = Ordering.by(NNKey.unapply) + + val entityEmbeddings = searchSpaceEmbeddings.map { embedding: EmbeddingWithEntity[B] => + val entityEmbedding = + EntityEmbedding(embedding.entityId, Embedding(embedding.embedding.toArray)) + entityEmbedding + } + + val shardedSearchSpace = shard(entityEmbeddings, searchSpaceShards) + + val groupedSearchSpaceEmbeddings = shardedSearchSpace + .flatMap { entityEmbedding => + val searchGroup = Random.nextInt(numOfSearchGroups) + (0 until numReplicas).map { replica => + (NNKey(searchGroup, replica, Some(numReplicas)), entityEmbedding) + } + } + + val shardedQueries = shard(queryEmbeddings, queryShards) + + val groupedQueryEmbeddings = shardedQueries + .flatMap { entity => + val replica = Random.nextInt(numReplicas) + (0 until numOfSearchGroups).map { searchGroup => + (NNKey(searchGroup, replica, Some(numReplicas)), entity) + } + }.group + .withReducers(reducersOption.getOrElse(numOfSearchGroups * numReplicas)) + + val numberAnnIndexQueries = Stat("NumberAnnIndexQueries") + val annIndexQueryTotalMs = Stat("AnnIndexQueryTotalMs") + val numberIndexBuilds = Stat("NumberIndexBuilds") + val annIndexBuildTotalMs = Stat("AnnIndexBuildTotalMs") + val groupedKnn = groupedQueryEmbeddings + .cogroup(groupedSearchSpaceEmbeddings) { + case (_, queryIter, searchSpaceIter) => + // This index build happens numReplicas times. Ideally we could serialize the queryable. + // And only build the index once per search group. + // The issues with that now are: + // - The HNSW queryable is not serializable in scalding + // - The way that map reduce works requires that there is a job that write out the search + // space embeddings numReplicas times. In the current setup, we can do that by sharding + // the embeddings first and then fanning out. But if we had a single queryable, we would + // not be able to shard it easily and writing this out would take a long time. + val indexBuildStartTime = System.currentTimeMillis() + val queryable = indexingStrategy.buildIndex(searchSpaceIter) + if (useCounters) { + numberIndexBuilds.inc() + annIndexBuildTotalMs.incBy(System.currentTimeMillis() - indexBuildStartTime) + } + queryIter.flatMap { query => + val queryStartTime = System.currentTimeMillis() + val embedding = Embedding(query.embedding.toArray) + val result = Await.result( + queryable.queryWithDistance(embedding, numNeighbors) + ) + val queryToTopNeighbors = result + .map { neighbor => + (query.entityId, (neighbor.neighbor, neighbor.distance)) + } + if (useCounters) { + numberAnnIndexQueries.inc() + annIndexQueryTotalMs.incBy(System.currentTimeMillis() - queryStartTime) + } + queryToTopNeighbors + } + } + .values + .group + + val groupedKnnWithReducers = reducersOption + .map { reducers => + groupedKnn + .withReducers(reducers) + }.getOrElse(groupedKnn) + + groupedKnnWithReducers + .sortedTake(numNeighbors) { + Ordering + .by[(B, D), D] { + case (_, distance) => distance + } + } + } + + private[this] def shard[T]( + pipe: TypedPipe[T], + numberOfShards: Option[Int] + ): TypedPipe[T] = { + numberOfShards + .map { shards => + pipe.shard(shards) + }.getOrElse(pipe) + } + + private[this] def findNearestNeighboursViaCross[A <: EntityId, B <: EntityId, D <: Distance[D]]( + queryEmbeddings: TypedPipe[EmbeddingWithEntity[A]], + searchSpaceEmbeddings: TypedPipe[EmbeddingWithEntity[B]], + metric: Metric[D], + numNeighbors: Int, + reducers: Int, + mappers: Int, + isSearchSpaceLarger: Boolean + )( + implicit ordering: Ordering[A] + ): TypedPipe[(A, Seq[(B, D)])] = { + + val crossed: TypedPipe[(A, (B, D))] = if (isSearchSpaceLarger) { + searchSpaceEmbeddings + .shard(mappers) + .cross(queryEmbeddings).map { + case (searchSpaceEmbedding, queryEmbedding) => + val distance = metric.distance(searchSpaceEmbedding.embedding, queryEmbedding.embedding) + (queryEmbedding.entityId, (searchSpaceEmbedding.entityId, distance)) + } + } else { + queryEmbeddings + .shard(mappers) + .cross(searchSpaceEmbeddings).map { + case (queryEmbedding, searchSpaceEmbedding) => + val distance = metric.distance(searchSpaceEmbedding.embedding, queryEmbedding.embedding) + (queryEmbedding.entityId, (searchSpaceEmbedding.entityId, distance)) + } + } + + crossed + .groupBy(_._1) + .withReducers(reducers) + .sortedTake(numNeighbors) { + Ordering + .by[(A, (B, D)), D] { + case (_, (_, distance)) => distance + } // Sort by distance metric in ascending order + }.map { + case (queryId, neighbors) => + (queryId, neighbors.map(_._2)) + } + } + + /** + * Convert nearest neighbors to string format. + * By default format would be (queryId neighbourId:distance neighbourId:distance .....) in ascending order of distance. + * @param nearestNeighbors nearest neighbors tuple in form of (queryId, Seq[(neighborId, distance)] + * @param queryEntityKind entity kind of query + * @param neighborEntityKind entity kind of search space/neighbors + * @param idDistanceSeparator String separator to separate a single neighborId and distance. Default to colon (:) + * @param neighborSeparator String operator to separate neighbors. Default to tab + * @tparam A type of query entity + * @tparam B type of search space entity + * @tparam D type of distance + */ + def nearestNeighborsToString[A <: EntityId, B <: EntityId, D <: Distance[D]]( + nearestNeighbors: (A, Seq[(B, D)]), + queryEntityKind: EntityKind[A], + neighborEntityKind: EntityKind[B], + idDistanceSeparator: String = ":", + neighborSeparator: String = "\t" + ): String = { + val (queryId, neighbors) = nearestNeighbors + val formattedNeighbors = neighbors.map { + case (neighbourId, distance) => + s"${neighborEntityKind.stringInjection.apply(neighbourId)}$idDistanceSeparator${distance.distance}" + } + (queryEntityKind.stringInjection.apply(queryId) +: formattedNeighbors) + .mkString(neighborSeparator) + } + + private[this] case class NNKey( + searchGroup: Int, + replica: Int, + maxReplica: Option[Int] = None) { + override def hashCode(): Int = + maxReplica.map(_ * searchGroup + replica).getOrElse(super.hashCode()) + } +} diff --git a/ann/src/main/scala/com/twitter/ann/scalding/offline/KnnOfflineJob.scala b/ann/src/main/scala/com/twitter/ann/scalding/offline/KnnOfflineJob.scala new file mode 100644 index 0000000000..87b6aacf8a --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/scalding/offline/KnnOfflineJob.scala @@ -0,0 +1,108 @@ +package com.twitter.ann.scalding.offline + +import com.twitter.ann.common.Metric +import com.twitter.bijection.scrooge.BinaryScalaCodec +import com.twitter.ml.featurestore.lib.UserId +import com.twitter.ml.featurestore.lib.embedding.EmbeddingWithEntity +import com.twitter.cortex.ml.embeddings.common.EntityKind +import com.twitter.entityembeddings.neighbors.thriftscala.{EntityKey, NearestNeighbors} +import com.twitter.scalding.commons.source.VersionedKeyValSource +import com.twitter.scalding.typed.TypedPipe +import com.twitter.scalding.{Args, DateOps, DateParser, DateRange, Execution, TypedTsv, UniqueID} +import com.twitter.scalding_internal.job.TwitterExecutionApp +import com.twitter.search.common.file.{AbstractFile, LocalFile} +import java.util.TimeZone + +/** + * Generates the nearest neighbour for users and store them in Manhattan format i.e sequence files. + * See README for oscar usage. + */ +object KnnOfflineJob extends TwitterExecutionApp { + override def job: Execution[Unit] = Execution.withId { implicit uniqueId => + Execution.getArgs.flatMap { args: Args => + val knnDirectoryOpt: Option[String] = args.optional("knn_directory") + knnDirectoryOpt match { + case Some(knnDirectory) => + Execution.withCachedFile(knnDirectory) { directory => + execute(args, Some(new LocalFile(directory.file))) + } + case None => + execute(args, None) + } + } + } + + /** + * Execute KnnOfflineJob + * @param args: The args object for this job + * @param abstractFile: An optional of producer embedding path + */ + def execute( + args: Args, + abstractFile: Option[AbstractFile] + )( + implicit uniqueID: UniqueID + ): Execution[Unit] = { + implicit val tz: TimeZone = TimeZone.getDefault() + implicit val dp: DateParser = DateParser.default + implicit val dateRange = DateRange.parse(args.list("date"))(DateOps.UTC, DateParser.default) + implicit val keyInject = BinaryScalaCodec(EntityKey) + implicit val valueInject = BinaryScalaCodec(NearestNeighbors) + + val entityKind = EntityKind.getEntityKind(args("producer_entity_kind")) + val metric = Metric.fromString(args("metric")) + val outputPath: String = args("output_path") + val numNeighbors: Int = args("neighbors").toInt + val ef = args.getOrElse("ef", numNeighbors.toString).toInt + val reducers: Int = args("reducers").toInt + val knnDimension: Int = args("dimension").toInt + val debugOutputPath: Option[String] = args.optional("debug_output_path") + val filterPath: Option[String] = args.optional("users_filter_path") + val shards: Int = args.getOrElse("shards", "100").toInt + val useHashJoin: Boolean = args.getOrElse("use_hash_join", "false").toBoolean + val mhOutput = VersionedKeyValSource[EntityKey, NearestNeighbors]( + path = outputPath, + sourceVersion = None, + sinkVersion = None, + maxFailures = 0, + versionsToKeep = 1 + ) + + val consumerEmbeddings: TypedPipe[EmbeddingWithEntity[UserId]] = + KnnHelper.getFilteredUserEmbeddings( + args, + filterPath, + reducers, + useHashJoin + ) + + val neighborsPipe: TypedPipe[(EntityKey, NearestNeighbors)] = KnnHelper.getNeighborsPipe( + args, + entityKind, + metric, + ef, + consumerEmbeddings, + abstractFile, + reducers, + numNeighbors, + knnDimension + ) + + val neighborsExecution: Execution[Unit] = neighborsPipe + .writeExecution(mhOutput) + + // Write manual Inspection + debugOutputPath match { + case Some(path: String) => + val debugExecution: Execution[Unit] = KnnDebug + .getDebugTable( + neighborsPipe = neighborsPipe, + shards = shards, + reducers = reducers + ) + .writeExecution(TypedTsv(path)) + Execution.zip(debugExecution, neighborsExecution).unit + case None => neighborsExecution + } + } +} diff --git a/ann/src/main/scala/com/twitter/ann/scalding/offline/KnnTruthSetGenerator.scala b/ann/src/main/scala/com/twitter/ann/scalding/offline/KnnTruthSetGenerator.scala new file mode 100644 index 0000000000..23b064fc3e --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/scalding/offline/KnnTruthSetGenerator.scala @@ -0,0 +1,84 @@ +package com.twitter.ann.scalding.offline + +import com.twitter.ann.common.Distance +import com.twitter.ann.common.Metric +import com.twitter.ann.scalding.offline.KnnHelper.nearestNeighborsToString +import com.twitter.cortex.ml.embeddings.common.EntityKind +import com.twitter.ml.featurestore.lib.EntityId +import com.twitter.scalding.source.TypedText +import com.twitter.scalding.Args +import com.twitter.scalding.Execution +import com.twitter.scalding.UniqueID +import com.twitter.scalding_internal.job.TwitterExecutionApp + +/** + * This job reads index embedding data, query embeddings data, and split into index set, query set and true nearest neigbor set + * from query to index. + */ +object KnnTruthSetGenerator extends TwitterExecutionApp { + override def job: Execution[Unit] = Execution.withId { implicit uniqueId => + Execution.getArgs.flatMap { args: Args => + val queryEntityKind = EntityKind.getEntityKind(args("query_entity_kind")) + val indexEntityKind = EntityKind.getEntityKind(args("index_entity_kind")) + val metric = Metric.fromString(args("metric")) + run(queryEntityKind, indexEntityKind, metric, args) + } + } + + private[this] def run[A <: EntityId, B <: EntityId, D <: Distance[D]]( + uncastQueryEntityKind: EntityKind[_], + uncastIndexSpaceEntityKind: EntityKind[_], + uncastMetric: Metric[_], + args: Args + )( + implicit uniqueID: UniqueID + ): Execution[Unit] = { + val queryEntityKind = uncastQueryEntityKind.asInstanceOf[EntityKind[A]] + val indexEntityKind = uncastIndexSpaceEntityKind.asInstanceOf[EntityKind[B]] + val metric = uncastMetric.asInstanceOf[Metric[D]] + + val reducers = args.int("reducers") + val mappers = args.int("mappers") + val numNeighbors = args.int("neighbors") + val knnOutputPath = args("truth_set_output_path") + val querySamplePercent = args.double("query_sample_percent", 100) / 100 + val indexSamplePercent = args.double("index_sample_percent", 100) / 100 + + val queryEmbeddings = queryEntityKind.parser + .getEmbeddingFormat(args, "query") + .getEmbeddings + .sample(querySamplePercent) + + val indexEmbeddings = indexEntityKind.parser + .getEmbeddingFormat(args, "index") + .getEmbeddings + .sample(indexSamplePercent) + + // calculate and write knn + val knnExecution = KnnHelper + .findNearestNeighbours( + queryEmbeddings, + indexEmbeddings, + metric, + numNeighbors, + reducers = reducers, + mappers = mappers + )(queryEntityKind.ordering, uniqueID).map( + nearestNeighborsToString(_, queryEntityKind, indexEntityKind) + ) + .shard(1) + .writeExecution(TypedText.tsv(knnOutputPath)) + + // write query set embeddings + val querySetExecution = queryEntityKind.parser + .getEmbeddingFormat(args, "query_set_output") + .writeEmbeddings(queryEmbeddings) + + // write index set embeddings + val indexSetExecution = indexEntityKind.parser + .getEmbeddingFormat(args, "index_set_output") + .writeEmbeddings(indexEmbeddings) + + Execution.zip(knnExecution, querySetExecution, indexSetExecution).unit + } +} diff --git a/ann/src/main/scala/com/twitter/ann/scalding/offline/ParameterlessQueryable.scala b/ann/src/main/scala/com/twitter/ann/scalding/offline/ParameterlessQueryable.scala new file mode 100644 index 0000000000..e66a834b53 --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/scalding/offline/ParameterlessQueryable.scala @@ -0,0 +1,24 @@ +package com.twitter.ann.scalding.offline + +import com.twitter.ann.common.EmbeddingType.EmbeddingVector +import com.twitter.ann.common.{Distance, NeighborWithDistance, Queryable, RuntimeParams} +import com.twitter.util.Future + +private[offline] case class ParameterlessQueryable[T, P <: RuntimeParams, D <: Distance[D]]( + queryable: Queryable[T, P, D], + runtimeParamsForAllQueries: P) { + + /** + * ANN query for ids with distance. + * + * @param embedding : Embedding/Vector to be queried with. + * @param numOfNeighbors : Number of neighbours to be queried for. + * + * @return List of approximate nearest neighbour ids with distance from the query embedding. + */ + def queryWithDistance( + embedding: EmbeddingVector, + numOfNeighbors: Int + ): Future[List[NeighborWithDistance[T, D]]] = + queryable.queryWithDistance(embedding, numOfNeighbors, runtimeParamsForAllQueries) +} diff --git a/ann/src/main/scala/com/twitter/ann/scalding/offline/README b/ann/src/main/scala/com/twitter/ann/scalding/offline/README new file mode 100644 index 0000000000..9d4630ca56 --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/scalding/offline/README @@ -0,0 +1,4 @@ +# Description + +This pipeline uses hnsw and scalding to create an hnsw index based on producers embeddings, which +it then uses to construct lists of producer suggestions for each user. diff --git a/ann/src/main/scala/com/twitter/ann/scalding/offline/faissindexbuilder/BUILD.bazel b/ann/src/main/scala/com/twitter/ann/scalding/offline/faissindexbuilder/BUILD.bazel new file mode 100644 index 0000000000..c35a1658e4 --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/scalding/offline/faissindexbuilder/BUILD.bazel @@ -0,0 +1,37 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + platform = "java11", + tags = [ + "bazel-compatible", + "bazel-only", + ], + dependencies = [ + "3rdparty/src/jvm/com/twitter/scalding:args", + "3rdparty/src/jvm/com/twitter/scalding:core", + "ann/src/main/scala/com/twitter/ann/annoy", + "ann/src/main/scala/com/twitter/ann/brute_force", + "ann/src/main/scala/com/twitter/ann/common", + "ann/src/main/scala/com/twitter/ann/faiss", + "ann/src/main/scala/com/twitter/ann/serialization", + "ann/src/main/scala/com/twitter/ann/util", + "src/scala/com/twitter/cortex/ml/embeddings/common:Helpers", + "src/scala/com/twitter/scalding_internal/job", + ], +) + +hadoop_binary( + name = "faissindexbuilder-deploy", + main = "com.twitter.ann.scalding.offline.faissindexbuilder.IndexBuilderApp", + platform = "java11", + runtime_platform = "java11", + tags = [ + "bazel-compatible", + "bazel-compatible:migrated", + "bazel-only", + ], + dependencies = [ + ":faissindexbuilder", + "3rdparty/jvm/org/slf4j:slf4j-jdk14", + ], +) diff --git a/ann/src/main/scala/com/twitter/ann/scalding/offline/faissindexbuilder/IndexBuilder.scala b/ann/src/main/scala/com/twitter/ann/scalding/offline/faissindexbuilder/IndexBuilder.scala new file mode 100644 index 0000000000..879736b136 --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/scalding/offline/faissindexbuilder/IndexBuilder.scala @@ -0,0 +1,42 @@ +package com.twitter.ann.scalding.offline.faissindexbuilder + +import com.twitter.ann.common.Distance +import com.twitter.ann.common.EntityEmbedding +import com.twitter.ann.common.Metric +import com.twitter.ann.faiss.FaissIndexer +import com.twitter.cortex.ml.embeddings.common.EmbeddingFormat +import com.twitter.ml.api.embedding.Embedding +import com.twitter.ml.featurestore.lib.UserId +import com.twitter.scalding.Execution +import com.twitter.search.common.file.AbstractFile +import com.twitter.util.logging.Logging + +object IndexBuilder extends FaissIndexer with Logging { + def run[T <: UserId, D <: Distance[D]]( + embeddingFormat: EmbeddingFormat[T], + embeddingLimit: Option[Int], + sampleRate: Float, + factoryString: String, + metric: Metric[D], + outputDirectory: AbstractFile, + numDimensions: Int + ): Execution[Unit] = { + val embeddingsPipe = embeddingFormat.getEmbeddings + val limitedEmbeddingsPipe = embeddingLimit + .map { limit => + embeddingsPipe.limit(limit) + }.getOrElse(embeddingsPipe) + + val annEmbeddingPipe = limitedEmbeddingsPipe.map { embedding => + val embeddingSize = embedding.embedding.length + assert( + embeddingSize == numDimensions, + s"Specified number of dimensions $numDimensions does not match the dimensions of the " + + s"embedding $embeddingSize" + ) + EntityEmbedding[Long](embedding.entityId.userId, Embedding(embedding.embedding.toArray)) + } + + build(annEmbeddingPipe, sampleRate, factoryString, metric, outputDirectory) + } +} diff --git a/ann/src/main/scala/com/twitter/ann/scalding/offline/faissindexbuilder/IndexBuilderApp.scala b/ann/src/main/scala/com/twitter/ann/scalding/offline/faissindexbuilder/IndexBuilderApp.scala new file mode 100644 index 0000000000..9857572716 --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/scalding/offline/faissindexbuilder/IndexBuilderApp.scala @@ -0,0 +1,75 @@ +package com.twitter.ann.scalding.offline.faissindexbuilder + +import com.twitter.ann.common.Distance +import com.twitter.ann.common.Metric +import com.twitter.cortex.ml.embeddings.common._ +import com.twitter.ml.featurestore.lib.UserId +import com.twitter.scalding.Args +import com.twitter.scalding.DateOps +import com.twitter.scalding.DateParser +import com.twitter.scalding.DateRange +import com.twitter.scalding.Execution +import com.twitter.scalding_internal.job.TwitterExecutionApp +import com.twitter.search.common.file.FileUtils +import com.twitter.util.logging.Logging +import java.util.Calendar +import java.util.TimeZone + +trait IndexBuilderExecutable extends Logging { + // This method is used to cast the entityKind and the metric to have parameters. + def indexBuilderExecution[T <: UserId, D <: Distance[D]]( + args: Args + ): Execution[Unit] = { + // parse the arguments for this job + val uncastEntityKind = EntityKind.getEntityKind(args("entity_kind")) + val uncastMetric = Metric.fromString(args("metric")) + val entityKind = uncastEntityKind.asInstanceOf[EntityKind[T]] + val metric = uncastMetric.asInstanceOf[Metric[D]] + val uncastDateRange = args.list("embedding_date_range") + val embeddingDateRange = if (uncastDateRange.nonEmpty) { + Some(DateRange.parse(uncastDateRange)(DateOps.UTC, DateParser.default)) + } else { + None + } + val embeddingFormat = + entityKind.parser.getEmbeddingFormat(args, "input", providedDateRange = embeddingDateRange) + val numDimensions = args.int("num_dimensions") + val embeddingLimit = args.optional("embedding_limit").map(_.toInt) + val outputDirectory = FileUtils.getFileHandle(args("output_dir")) + val factoryString = args.optional("factory_string").get + val sampleRate = args.float("training_sample_rate", 0.05f) + + logger.debug(s"Job args: ${args.toString}") + + val finalOutputDirectory = embeddingDateRange + .map { range => + val cal = Calendar.getInstance(TimeZone.getTimeZone("UTC")) + cal.setTime(range.end) + outputDirectory + .getChild(s"${cal.get(Calendar.YEAR)}") + .getChild(f"${cal.get(Calendar.MONTH) + 1}%02d") + .getChild(f"${cal.get(Calendar.DAY_OF_MONTH)}%02d") + }.getOrElse(outputDirectory) + + logger.info(s"Final output directory is ${finalOutputDirectory.getPath}") + + IndexBuilder + .run( + embeddingFormat, + embeddingLimit, + sampleRate, + factoryString, + metric, + finalOutputDirectory, + numDimensions + ).onComplete { _ => + Unit + } + } +} + +object IndexBuilderApp extends TwitterExecutionApp with IndexBuilderExecutable { + override def job: Execution[Unit] = Execution.getArgs.flatMap { args: Args => + indexBuilderExecution(args) + } +} diff --git a/ann/src/main/scala/com/twitter/ann/scalding/offline/indexbuilder/BUILD.bazel b/ann/src/main/scala/com/twitter/ann/scalding/offline/indexbuilder/BUILD.bazel new file mode 100644 index 0000000000..bd421443ec --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/scalding/offline/indexbuilder/BUILD.bazel @@ -0,0 +1,37 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = [ + "bazel-compatible", + "bazel-only", + ], + dependencies = [ + "3rdparty/src/jvm/com/twitter/scalding:args", + "3rdparty/src/jvm/com/twitter/scalding:core", + "ann/src/main/scala/com/twitter/ann/annoy", + "ann/src/main/scala/com/twitter/ann/brute_force", + "ann/src/main/scala/com/twitter/ann/common", + "ann/src/main/scala/com/twitter/ann/hnsw", + "ann/src/main/scala/com/twitter/ann/serialization", + "ann/src/main/scala/com/twitter/ann/util", + "src/scala/com/twitter/cortex/ml/embeddings/common:Helpers", + "src/scala/com/twitter/scalding_internal/job", + ], +) + +hadoop_binary( + name = "indexbuilder-deploy", + main = "com.twitter.ann.scalding.offline.indexbuilder.IndexBuilderApp", + platform = "java8", + runtime_platform = "java8", + tags = [ + "bazel-compatible", + "bazel-compatible:migrated", + "bazel-only", + ], + dependencies = [ + ":indexbuilder", + "3rdparty/jvm/org/slf4j:slf4j-jdk14", + ], +) diff --git a/ann/src/main/scala/com/twitter/ann/scalding/offline/indexbuilder/IndexBuilder.scala b/ann/src/main/scala/com/twitter/ann/scalding/offline/indexbuilder/IndexBuilder.scala new file mode 100644 index 0000000000..9e4d49da70 --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/scalding/offline/indexbuilder/IndexBuilder.scala @@ -0,0 +1,53 @@ +package com.twitter.ann.scalding.offline.indexbuilder + +import com.twitter.ann.common.Appendable +import com.twitter.ann.common.Distance +import com.twitter.ann.common.EntityEmbedding +import com.twitter.ann.common.Serialization +import com.twitter.ann.util.IndexBuilderUtils +import com.twitter.cortex.ml.embeddings.common.EmbeddingFormat +import com.twitter.ml.api.embedding.Embedding +import com.twitter.ml.featurestore.lib.EntityId +import com.twitter.scalding.Execution +import com.twitter.scalding_internal.job.FutureHelper +import com.twitter.search.common.file.AbstractFile +import com.twitter.util.logging.Logger + +object IndexBuilder { + private[this] val Log = Logger.apply[IndexBuilder.type] + + def run[T <: EntityId, _, D <: Distance[D]]( + embeddingFormat: EmbeddingFormat[T], + embeddingLimit: Option[Int], + index: Appendable[T, _, D] with Serialization, + concurrencyLevel: Int, + outputDirectory: AbstractFile, + numDimensions: Int + ): Execution[Unit] = { + val embeddingsPipe = embeddingFormat.getEmbeddings + val limitedEmbeddingsPipe = embeddingLimit + .map { limit => + embeddingsPipe.limit(limit) + }.getOrElse(embeddingsPipe) + + val annEmbeddingPipe = limitedEmbeddingsPipe.map { embedding => + val embeddingSize = embedding.embedding.length + assert( + embeddingSize == numDimensions, + s"Specified number of dimensions $numDimensions does not match the dimensions of the " + + s"embedding $embeddingSize" + ) + EntityEmbedding[T](embedding.entityId, Embedding(embedding.embedding.toArray)) + } + + annEmbeddingPipe.toIterableExecution.flatMap { annEmbeddings => + val future = IndexBuilderUtils.addToIndex(index, annEmbeddings.toStream, concurrencyLevel) + val result = future.map { numberUpdates => + Log.info(s"Performed $numberUpdates updates") + index.toDirectory(outputDirectory) + Log.info(s"Finished writing to $outputDirectory") + } + FutureHelper.executionFrom(result).unit + } + } +} diff --git a/ann/src/main/scala/com/twitter/ann/scalding/offline/indexbuilder/IndexBuilderApp.scala b/ann/src/main/scala/com/twitter/ann/scalding/offline/indexbuilder/IndexBuilderApp.scala new file mode 100644 index 0000000000..9ab98d30cb --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/scalding/offline/indexbuilder/IndexBuilderApp.scala @@ -0,0 +1,91 @@ +package com.twitter.ann.scalding.offline.indexbuilder + +import com.twitter.ann.annoy.TypedAnnoyIndex +import com.twitter.ann.brute_force.SerializableBruteForceIndex +import com.twitter.ann.common.Distance +import com.twitter.ann.common.Metric +import com.twitter.ann.common.ReadWriteFuturePool +import com.twitter.ann.hnsw.TypedHnswIndex +import com.twitter.ann.serialization.thriftscala.PersistedEmbedding +import com.twitter.ann.serialization.PersistedEmbeddingInjection +import com.twitter.ann.serialization.ThriftIteratorIO +import com.twitter.cortex.ml.embeddings.common._ +import com.twitter.ml.featurestore.lib.EntityId +import com.twitter.scalding.Args +import com.twitter.scalding.Execution +import com.twitter.scalding_internal.job.TwitterExecutionApp +import com.twitter.search.common.file.FileUtils +import com.twitter.util.FuturePool +import java.util.concurrent.Executors + +trait IndexBuilderExecutable { + // This method is used to cast the entityKind and the metric to have parameters. + def indexBuilderExecution[T <: EntityId, D <: Distance[D]]( + args: Args + ): Execution[Unit] = { + // parse the arguments for this job + val uncastEntityKind = EntityKind.getEntityKind(args("entity_kind")) + val uncastMetric = Metric.fromString(args("metric")) + val entityKind = uncastEntityKind.asInstanceOf[EntityKind[T]] + val metric = uncastMetric.asInstanceOf[Metric[D]] + val embeddingFormat = entityKind.parser.getEmbeddingFormat(args, "input") + val injection = entityKind.byteInjection + val numDimensions = args.int("num_dimensions") + val embeddingLimit = args.optional("embedding_limit").map(_.toInt) + val concurrencyLevel = args.int("concurrency_level") + val outputDirectory = FileUtils.getFileHandle(args("output_dir")) + + println(s"Job args: ${args.toString}") + val threadPool = Executors.newFixedThreadPool(concurrencyLevel) + + val serialization = args("algo") match { + case "brute_force" => + val PersistedEmbeddingIO = new ThriftIteratorIO[PersistedEmbedding](PersistedEmbedding) + SerializableBruteForceIndex[T, D]( + metric, + FuturePool.apply(threadPool), + new PersistedEmbeddingInjection[T](injection), + PersistedEmbeddingIO + ) + case "annoy" => + TypedAnnoyIndex.indexBuilder[T, D]( + numDimensions, + args.int("annoy_num_trees"), + metric, + injection, + FuturePool.apply(threadPool) + ) + case "hnsw" => + val efConstruction = args.int("ef_construction") + val maxM = args.int("max_m") + val expectedElements = args.int("expected_elements") + TypedHnswIndex.serializableIndex[T, D]( + numDimensions, + metric, + efConstruction, + maxM, + expectedElements, + injection, + ReadWriteFuturePool(FuturePool.apply(threadPool)) + ) + } + IndexBuilder + .run( + embeddingFormat, + embeddingLimit, + serialization, + concurrencyLevel, + outputDirectory, + numDimensions + ).onComplete { _ => + threadPool.shutdown() + Unit + } + } +} + +object IndexBuilderApp extends TwitterExecutionApp with IndexBuilderExecutable { + override def job: Execution[Unit] = Execution.getArgs.flatMap { args: Args => + indexBuilderExecution(args) + } +} diff --git a/ann/src/main/scala/com/twitter/ann/scalding/offline/indexbuilder/README.rst b/ann/src/main/scala/com/twitter/ann/scalding/offline/indexbuilder/README.rst new file mode 100644 index 0000000000..c58e9620a3 --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/scalding/offline/indexbuilder/README.rst @@ -0,0 +1,132 @@ +******** +Overview +******** +This job reads embedding data from HDFS in the embedding formats supported by the cortex MLX team. It converts that data into the ANN format and adds it to an ANN index. The ANN index is serialized and save to disk. + +***************** +Running In Aurora +***************** + +Set up example +============== +This job builds an ANN index based on hnsw algorithm using user embeddings available in hdfs. + +.. code-block:: bash + + $ export JOB_NAME=ann_index_builder + $ export OUTPUT_PATH=hdfs:///user/$USER/${JOB_NAME}_test + + $ CPU=32 RAM_GB=150 DISK_GB=60 aurora job create smf1/$USER/devel/$JOB_NAME ann/src/main/aurora/index_builder/aurora_builder.aurora \ + --bind=profile.name=$JOB_NAME \ + --bind=profile.role=$USER \ + --bind=profile.output_dir=$OUTPUT_PATH \ + --bind=profile.entity_kind=user \ + --bind=profile.embedding_args='--input.embedding_format tab --input.embedding_path /user/cortex-mlx/official_examples/ann/non_pii_random_user_embeddings_tab_format' \ + --bind=profile.num_dimensions=300 \ + --bind=profile.algo=hnsw \ + --bind=profile.ef_construction=200 \ + --bind=profile.max_m=16 \ + --bind=profile.expected_elements=10000000 \ + --bind=profile.metric=InnerProduct \ + --bind=profile.concurrency_level=32 \ + --bind=profile.hadoop_cluster=dw2-smf1 + +This job builds an ANN index based on hnsw algorithm using producer embeddings (Major version 1546473691) available in feature store. + +.. code-block:: bash + + $ export JOB_NAME=ann_index_builder + $ export OUTPUT_PATH=hdfs:///user/$USER/${JOB_NAME}_test + + $ CPU=32 RAM_GB=150 DISK_GB=60 aurora job create smf1/$USER/devel/$JOB_NAME ann/src/main/aurora/index_builder/aurora_builder.aurora \ + --bind=profile.name=$JOB_NAME \ + --bind=profile.role=$USER \ + --bind=profile.output_dir=$OUTPUT_PATH \ + --bind=profile.entity_kind=user \ + --bind=profile.embedding_args='--input.feature_store_embedding ProducerFollowEmbedding300Dataset --input.feature_store_major_version 1546473691 --input.date_range 2019-01-02' \ + --bind=profile.num_dimensions=300 \ + --bind=profile.algo=hnsw \ + --bind=profile.ef_construction=200 \ + --bind=profile.max_m=16 \ + --bind=profile.expected_elements=10000000 \ + --bind=profile.metric=InnerProduct \ + --bind=profile.concurrency_level=32 \ + --bind=profile.hadoop_cluster=dw2-smf1 + + +************* +Job arguments +************* + +Enviroment variables (resources): +============== +- **CPU** Number of cpu cores (default: 32) +- **RAM_GB** RAM in gigabytes (default: 150) +- **DISK_GB** Disk in gigabytes (default: 60) + +General arguments (specified as **--profile.{options}**): +============== +- **name** Aurora job name +- **role** Aurora role +- **hadoop_cluster** Hadoop cluster for data. dw2-smf1/proc-atla. +- **input_dir** Path of saved embeddings in hdfs without prefixing `hdfs://` +- **entity_kind** The type of entity id that is use with the embeddings. Possible options: + + - word + - url + - user + - tweet + - tfwId + +- **embedding_args** Embedding format args. See the documentation in `com.twitter.cortex.ml.embeddings.common.EmbeddingFormatArgsParser` for a full explanation of the input options. Possible options: + + 1. **input.embedding_format** Format of the serialized embedding. + + - usertensor + - usercontinuous + - comma + - tab + + 2. **input.embedding_path** Path of saved embeddings in hdfs without prefixing `hdfs://` + + 3. **input.{feature_store_args}** For feature store related args like `feature_store_embedding`, `feature_store_major_version`, `date_range`: + +- **output_dir** Where to save the produced serialized ann index. Save to HDFS by specifying the full URI. e.g `hdfs://hadoop-dw2-nn.smf1.twitter.com/user//index_file` or using the default cluster `hdfs:///user//index_file`. +- **num_dimensions** Dimension of embedding in the input data. An exception will be thrown if any entry does not have a number of dimensions equal to this number. +- **metric** Distance metric (InnerProduct/Cosine/L2) +- **concurrency_level** Specifies how many parallel inserts happen to the index. This should probably be set to the number of cores on the machine. +- **algo** The kind of index you want to ouput. The supported options right now are: + + 1. **hnsw** (Metric supported: Cosine, L2, InnerProduct) + + .. _hnsw: https://arxiv.org/abs/1603.09320 + + - **ef\_construction** : Larger value increases build time but will give better recall. Good start value : 200 + - **max\_m** : Larger value increases will increase the index size but will give better recall. Optimal Range : 6-48. Good starting value 16. + - **expected\_elements** : Approximate number of elements that will be indexed. + + 2. **annoy** (Metric supported: Cosine, L2) + + .. _annoy: https://github.com/spotify/annoy + + - **annoy\_num\_trees** This parameter is required for annoy. From the annoy documentation: num_trees is provided during build time and affects the build time and the index size. A larger value will give more accurate results, but larger indexes. + + 3. **brute_force** (Metric supported: Cosine, L2, InnerProduct) + + +Developing locally +=================== + +For building and testing custom ann index builder job, +You can create job bundle locally, upload to packer and then it can be used with the job using `profile.packer_package` for name, `profile.packer_role` for role and `profile.packer_version` for bundle version. + +.. code-block:: bash + + ./bazel bundle ann/src/main/scala/com/twitter/ann/scalding/offline/indexbuilder:indexbuilder-deploy \ + --bundle-jvm-archive=zip + +.. code-block:: bash + + packer add_version --cluster=atla dist/indexbuilder-deploy.zip + + diff --git a/ann/src/main/scala/com/twitter/ann/scalding/offline/indexbuilderfrombq/BUILD.bazel b/ann/src/main/scala/com/twitter/ann/scalding/offline/indexbuilderfrombq/BUILD.bazel new file mode 100644 index 0000000000..992325fa2e --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/scalding/offline/indexbuilderfrombq/BUILD.bazel @@ -0,0 +1,37 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = [ + "bazel-compatible", + "bazel-only", + ], + dependencies = [ + "3rdparty/src/jvm/com/twitter/scalding:args", + "3rdparty/src/jvm/com/twitter/scalding:core", + "ann/src/main/scala/com/twitter/ann/annoy", + "ann/src/main/scala/com/twitter/ann/brute_force", + "ann/src/main/scala/com/twitter/ann/common", + "ann/src/main/scala/com/twitter/ann/hnsw", + "ann/src/main/scala/com/twitter/ann/serialization", + "ann/src/main/scala/com/twitter/ann/util", + "src/scala/com/twitter/cortex/ml/embeddings/common:Helpers", + "src/scala/com/twitter/scalding_internal/bigquery", + "src/scala/com/twitter/scalding_internal/job", + ], +) + +hadoop_binary( + name = "ann-index-builder", + main = "com.twitter.ann.scalding.offline.indexbuilderfrombq.IndexBuilderFromBQApp", + platform = "java8", + runtime_platform = "java8", + tags = [ + "bazel-compatible", + "bazel-compatible:migrated", + "bazel-only", + ], + dependencies = [ + ":indexbuilderfrombq", + ], +) diff --git a/ann/src/main/scala/com/twitter/ann/scalding/offline/indexbuilderfrombq/IndexBuilderFromBQ.scala b/ann/src/main/scala/com/twitter/ann/scalding/offline/indexbuilderfrombq/IndexBuilderFromBQ.scala new file mode 100644 index 0000000000..fe82a07929 --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/scalding/offline/indexbuilderfrombq/IndexBuilderFromBQ.scala @@ -0,0 +1,53 @@ +package com.twitter.ann.scalding.offline.indexbuilderfrombq + +import com.twitter.ann.common.Appendable +import com.twitter.ann.common.Distance +import com.twitter.ann.common.EntityEmbedding +import com.twitter.ann.common.Serialization +import com.twitter.ann.util.IndexBuilderUtils +import com.twitter.ml.api.embedding.Embedding +import com.twitter.ml.featurestore.lib.embedding.EmbeddingWithEntity +import com.twitter.ml.featurestore.lib.EntityId +import com.twitter.scalding.Execution +import com.twitter.scalding.TypedPipe +import com.twitter.scalding_internal.job.FutureHelper +import com.twitter.search.common.file.AbstractFile +import com.twitter.util.logging.Logger + +object IndexBuilder { + private[this] val Log = Logger.apply[IndexBuilder.type] + + def run[T <: EntityId, _, D <: Distance[D]]( + embeddingsPipe: TypedPipe[EmbeddingWithEntity[T]], + embeddingLimit: Option[Int], + index: Appendable[T, _, D] with Serialization, + concurrencyLevel: Int, + outputDirectory: AbstractFile, + numDimensions: Int + ): Execution[Unit] = { + val limitedEmbeddingsPipe = embeddingLimit + .map { limit => + embeddingsPipe.limit(limit) + }.getOrElse(embeddingsPipe) + + val annEmbeddingPipe = limitedEmbeddingsPipe.map { embedding => + val embeddingSize = embedding.embedding.length + assert( + embeddingSize == numDimensions, + s"Specified number of dimensions $numDimensions does not match the dimensions of the " + + s"embedding $embeddingSize" + ) + EntityEmbedding[T](embedding.entityId, Embedding(embedding.embedding.toArray)) + } + + annEmbeddingPipe.toIterableExecution.flatMap { annEmbeddings => + val future = IndexBuilderUtils.addToIndex(index, annEmbeddings.toStream, concurrencyLevel) + val result = future.map { numberUpdates => + Log.info(s"Performed $numberUpdates updates") + index.toDirectory(outputDirectory) + Log.info(s"Finished writing to $outputDirectory") + } + FutureHelper.executionFrom(result).unit + } + } +} diff --git a/ann/src/main/scala/com/twitter/ann/scalding/offline/indexbuilderfrombq/IndexBuilderFromBQApp.scala b/ann/src/main/scala/com/twitter/ann/scalding/offline/indexbuilderfrombq/IndexBuilderFromBQApp.scala new file mode 100644 index 0000000000..18079ac96f --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/scalding/offline/indexbuilderfrombq/IndexBuilderFromBQApp.scala @@ -0,0 +1,194 @@ +package com.twitter.ann.scalding.offline.indexbuilderfrombq + +import com.google.auth.oauth2.ServiceAccountCredentials +import com.google.cloud.bigquery.BigQueryOptions +import com.google.cloud.bigquery.QueryJobConfiguration +import com.twitter.ann.annoy.TypedAnnoyIndex +import com.twitter.ann.brute_force.SerializableBruteForceIndex +import com.twitter.ann.common.Distance +import com.twitter.ann.common.Metric +import com.twitter.ann.common.ReadWriteFuturePool +import com.twitter.ann.hnsw.TypedHnswIndex +import com.twitter.ann.serialization.PersistedEmbeddingInjection +import com.twitter.ann.serialization.ThriftIteratorIO +import com.twitter.ann.serialization.thriftscala.PersistedEmbedding +import com.twitter.cortex.ml.embeddings.common._ +import com.twitter.ml.api.embedding.Embedding +import com.twitter.ml.featurestore.lib._ +import com.twitter.ml.featurestore.lib.embedding.EmbeddingWithEntity +import com.twitter.scalding.Args +import com.twitter.scalding.Execution +import com.twitter.scalding.typed.TypedPipe +import com.twitter.scalding_internal.bigquery.BigQueryConfig +import com.twitter.scalding_internal.bigquery.BigQuerySource +import com.twitter.scalding_internal.job.TwitterExecutionApp +import com.twitter.scalding_internal.multiformat.format.keyval.KeyVal +import com.twitter.search.common.file.FileUtils +import com.twitter.util.FuturePool +import java.io.FileInputStream +import java.time.LocalDateTime +import java.time.ZoneOffset +import java.util.concurrent.Executors +import org.apache.avro.generic.GenericRecord +import scala.collection.JavaConverters._ + +/** + * Scalding execution app for building ANN index from embeddings present in BigQuery table. + * The output index is written to a GCS file. + * + * Note: + * - Assumes input data has the fields entityId + * - Assumes input data has the fields embedding + * + * Command for running the app (from source repo root): + * scalding remote run \ + * --target ann/src/main/scala/com/twitter/ann/scalding/offline/indexbuilderfrombq:ann-index-builder-binary + */ +trait IndexBuilderFromBQExecutable { + // This method is used to cast the entityKind and the metric to have parameters. + def indexBuilderExecution[T <: EntityId, D <: Distance[D]]( + args: Args + ): Execution[Unit] = { + // parse the arguments for this job + val uncastEntityKind = EntityKind.getEntityKind(args("entity_kind")) + val uncastMetric = Metric.fromString(args("metric")) + val entityKind = uncastEntityKind.asInstanceOf[EntityKind[T]] + val metric = uncastMetric.asInstanceOf[Metric[D]] + val injection = entityKind.byteInjection + val numDimensions = args.int("num_dimensions") + val embeddingLimit = args.optional("embedding_limit").map(_.toInt) + val concurrencyLevel = args.int("concurrency_level") + + val bigQuery = + BigQueryOptions + .newBuilder().setProjectId(args.required("bq_gcp_job_project")).setCredentials( + ServiceAccountCredentials.fromStream( + new FileInputStream(args.required("gcp_service_account_key_json")))).build().getService + + // Query to get the latest partition of the BigQuery table. + val query = + s"SELECT MAX(ts) AS RecentPartition FROM ${args.required("bq_gcp_table_project")}.${args + .required("bq_dataset")}.${args.required("bq_table")}" + val queryConfig = QueryJobConfiguration + .newBuilder(query) + .setUseLegacySql(false) + .build + val recentPartition = + bigQuery + .query(queryConfig).iterateAll().asScala.map(field => { + field.get(0).getStringValue + }).toArray.apply(0) + + // Query to extract the embeddings from the latest partition of the BigQuery table + val bigQueryConfig = BigQueryConfig( + args.required("bq_gcp_table_project"), + args + .required("bq_dataset"), + args.required("bq_table")) + .withServiceAccountKey(args.required("gcp_service_account_key_json")) + + val bqFilter = Some( + s"ts >= '${recentPartition}' AND DATE(TIMESTAMP_MILLIS(createdAt)) >= DATE_SUB(DATE('${recentPartition}'), INTERVAL 1 DAY) AND DATE(TIMESTAMP_MILLIS(createdAt)) <= DATE('${recentPartition}')") + val withFilterBigQueryConfig = bqFilter + .map { filter: String => + bigQueryConfig.withFilter(filter) + }.getOrElse(bigQueryConfig) + val source = new BigQuerySource(withFilterBigQueryConfig) + .andThen(avroMapper) + + val sourcePipe = TypedPipe + .from(source) + .map(transform[T](entityKind)) + + println(s"Job args: ${args.toString}") + val threadPool = Executors.newFixedThreadPool(concurrencyLevel) + + val serialization = args("algo") match { + case "brute_force" => + val PersistedEmbeddingIO = new ThriftIteratorIO[PersistedEmbedding](PersistedEmbedding) + SerializableBruteForceIndex[T, D]( + metric, + FuturePool.apply(threadPool), + new PersistedEmbeddingInjection[T](injection), + PersistedEmbeddingIO + ) + case "annoy" => + TypedAnnoyIndex.indexBuilder[T, D]( + numDimensions, + args.int("annoy_num_trees"), + metric, + injection, + FuturePool.apply(threadPool) + ) + case "hnsw" => + val efConstruction = args.int("ef_construction") + val maxM = args.int("max_m") + val expectedElements = args.int("expected_elements") + TypedHnswIndex.serializableIndex[T, D]( + numDimensions, + metric, + efConstruction, + maxM, + expectedElements, + injection, + ReadWriteFuturePool(FuturePool.apply(threadPool)) + ) + } + + // Output directory for the ANN index. We place the index under a timestamped directory which + // will be used by the ANN service to read the latest index + val timestamp = LocalDateTime.now().toEpochSecond(ZoneOffset.UTC) + val outputDirectory = FileUtils.getFileHandle(args("output_dir") + "/" + timestamp) + IndexBuilder + .run( + sourcePipe, + embeddingLimit, + serialization, + concurrencyLevel, + outputDirectory, + numDimensions + ).onComplete { _ => + threadPool.shutdown() + Unit + } + + } + + def avroMapper(row: GenericRecord): KeyVal[Long, java.util.List[Double]] = { + val entityId = row.get("entityId") + val embedding = row.get("embedding") + + KeyVal( + entityId.toString.toLong, + embedding.asInstanceOf[java.util.List[Double]] + ) + } + + def transform[T <: EntityId]( + entityKind: EntityKind[T] + )( + bqRecord: KeyVal[Long, java.util.List[Double]] + ): EmbeddingWithEntity[T] = { + val embeddingArray = bqRecord.value.asScala.map(_.floatValue()).toArray + val entity_id = entityKind match { + case UserKind => UserId(bqRecord.key).toThrift + case TweetKind => TweetId(bqRecord.key).toThrift + case TfwKind => TfwId(bqRecord.key).toThrift + case SemanticCoreKind => SemanticCoreId(bqRecord.key).toThrift + case _ => throw new IllegalArgumentException(s"Unsupported embedding kind: $entityKind") + } + EmbeddingWithEntity[T]( + EntityId.fromThrift(entity_id).asInstanceOf[T], + Embedding(embeddingArray)) + } +} + +/* +scalding remote run \ +--target ann/src/main/scala/com/twitter/ann/scalding/offline/indexbuilderfrombq:ann-index-builder-binary + */ +object IndexBuilderFromBQApp extends TwitterExecutionApp with IndexBuilderFromBQExecutable { + override def job: Execution[Unit] = Execution.getArgs.flatMap { args: Args => + indexBuilderExecution(args) + } +} diff --git a/ann/src/main/scala/com/twitter/ann/serialization/BUILD b/ann/src/main/scala/com/twitter/ann/serialization/BUILD new file mode 100644 index 0000000000..d2d6bf7542 --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/serialization/BUILD @@ -0,0 +1,14 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/com/twitter/bijection:core", + "ann/src/main/scala/com/twitter/ann/common", + "ann/src/main/thrift/com/twitter/ann/serialization:serialization-scala", + "mediaservices/commons", + "scrooge/scrooge-core", + "src/scala/com/twitter/scalding_internal/multiformat/format", + ], +) diff --git a/ann/src/main/scala/com/twitter/ann/serialization/DummyANNIndexInjection.scala b/ann/src/main/scala/com/twitter/ann/serialization/DummyANNIndexInjection.scala new file mode 100644 index 0000000000..c937472344 --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/serialization/DummyANNIndexInjection.scala @@ -0,0 +1,12 @@ +package com.twitter.ann.serialization + +import com.twitter.scalding_internal.multiformat.format.keyval.KeyValInjection +import com.twitter.scalding_internal.multiformat.format.keyval.KeyValInjection.Long2BigEndian + +/** +Dummy injection required to writeup dummy dal dataset to ANN folder. +**/ +object DummyANNIndexInjection { + val injection: KeyValInjection[Long, Long] = + KeyValInjection[Long, Long](Long2BigEndian, Long2BigEndian) +} diff --git a/ann/src/main/scala/com/twitter/ann/serialization/PersistedEmbeddingInjection.scala b/ann/src/main/scala/com/twitter/ann/serialization/PersistedEmbeddingInjection.scala new file mode 100644 index 0000000000..50917fe568 --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/serialization/PersistedEmbeddingInjection.scala @@ -0,0 +1,28 @@ +package com.twitter.ann.serialization + +import com.twitter.ann.common.EntityEmbedding +import com.twitter.ann.common.EmbeddingType._ +import com.twitter.ann.serialization.thriftscala.PersistedEmbedding +import com.twitter.bijection.Injection +import com.twitter.mediaservices.commons.codec.ArrayByteBufferCodec +import java.nio.ByteBuffer +import scala.util.Try + +/** + * Injection that converts from the ann.common.Embedding to the thrift PersistedEmbedding. + */ +class PersistedEmbeddingInjection[T]( + idByteInjection: Injection[T, Array[Byte]]) + extends Injection[EntityEmbedding[T], PersistedEmbedding] { + override def apply(entity: EntityEmbedding[T]): PersistedEmbedding = { + val byteBuffer = ByteBuffer.wrap(idByteInjection(entity.id)) + PersistedEmbedding(byteBuffer, embeddingSerDe.toThrift(entity.embedding)) + } + + override def invert(persistedEmbedding: PersistedEmbedding): Try[EntityEmbedding[T]] = { + val idTry = idByteInjection.invert(ArrayByteBufferCodec.decode(persistedEmbedding.id)) + idTry.map { id => + EntityEmbedding(id, embeddingSerDe.fromThrift(persistedEmbedding.embedding)) + } + } +} diff --git a/ann/src/main/scala/com/twitter/ann/serialization/ThriftIteratorIO.scala b/ann/src/main/scala/com/twitter/ann/serialization/ThriftIteratorIO.scala new file mode 100644 index 0000000000..6f7389723f --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/serialization/ThriftIteratorIO.scala @@ -0,0 +1,57 @@ +package com.twitter.ann.serialization + +import com.twitter.scrooge.{ThriftStruct, ThriftStructCodec} +import java.io.{InputStream, OutputStream} +import org.apache.thrift.protocol.TBinaryProtocol +import org.apache.thrift.transport.{TIOStreamTransport, TTransportException} + +/** + * Class that can serialize and deserialize an iterator of thrift objects. + * This class can do things lazily so there is no need to have all the object into memory. + */ +class ThriftIteratorIO[T <: ThriftStruct]( + codec: ThriftStructCodec[T]) { + def toOutputStream( + iterator: Iterator[T], + outputStream: OutputStream + ): Unit = { + val protocol = (new TBinaryProtocol.Factory).getProtocol(new TIOStreamTransport(outputStream)) + iterator.foreach { thriftObject => + codec.encode(thriftObject, protocol) + } + } + + /** + * Returns an iterator that lazily reads from an inputStream. + * @return + */ + def fromInputStream( + inputStream: InputStream + ): Iterator[T] = { + ThriftIteratorIO.getIterator(codec, inputStream) + } +} + +object ThriftIteratorIO { + private def getIterator[T <: ThriftStruct]( + codec: ThriftStructCodec[T], + inputStream: InputStream + ): Iterator[T] = { + val protocol = (new TBinaryProtocol.Factory).getProtocol(new TIOStreamTransport(inputStream)) + + def getNext: Option[T] = + try { + Some(codec.decode(protocol)) + } catch { + case e: TTransportException if e.getType == TTransportException.END_OF_FILE => + inputStream.close() + None + } + + Iterator + .continually[Option[T]](getNext) + .takeWhile(_.isDefined) + // It should be safe to call get on here since we are only take the defined ones. + .map(_.get) + } +} diff --git a/ann/src/main/scala/com/twitter/ann/service/loadtest/AnnLoadTest.scala b/ann/src/main/scala/com/twitter/ann/service/loadtest/AnnLoadTest.scala new file mode 100644 index 0000000000..ed7754b1bc --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/service/loadtest/AnnLoadTest.scala @@ -0,0 +1,66 @@ +package com.twitter.ann.service.loadtest + +import com.twitter.ann.common.EmbeddingType.EmbeddingVector +import com.twitter.ann.common.{Appendable, Distance, EntityEmbedding, Queryable, RuntimeParams} +import com.twitter.util.logging.Logger +import com.twitter.util.{Duration, Future} + +class AnnIndexQueryLoadTest( + worker: AnnLoadTestWorker = new AnnLoadTestWorker()) { + lazy val logger = Logger(getClass.getName) + + def performQueries[T, P <: RuntimeParams, D <: Distance[D]]( + queryable: Queryable[T, P, D], + qps: Int, + duration: Duration, + queries: Seq[Query[T]], + concurrencyLevel: Int, + runtimeConfigurations: Seq[QueryTimeConfiguration[T, P]] + ): Future[Unit] = { + logger.info(s"Query set: ${queries.size}") + val res = Future.traverseSequentially(runtimeConfigurations) { config => + logger.info(s"Run load test with runtime config $config") + worker.runWithQps( + queryable, + queries, + qps, + duration, + config, + concurrencyLevel + ) + } + res.onSuccess { _ => + logger.info(s"Done loadtest with $qps for ${duration.inMilliseconds / 1000} sec") + } + res.unit + } +} + +/** + * @param embedding Embedding vector + * @param trueNeighbours List of true neighbour ids. Empty in case true neighbours dataset not available + * @tparam T Type of neighbour + */ +case class Query[T](embedding: EmbeddingVector, trueNeighbours: Seq[T] = Seq.empty) + +class AnnIndexBuildLoadTest( + buildRecorder: LoadTestBuildRecorder, + embeddingIndexer: EmbeddingIndexer = new EmbeddingIndexer()) { + lazy val logger = Logger(getClass.getName) + def indexEmbeddings[T, P <: RuntimeParams, D <: Distance[D]]( + appendable: Appendable[T, P, D], + indexSet: Seq[EntityEmbedding[T]], + concurrencyLevel: Int + ): Future[Queryable[T, P, D]] = { + logger.info(s"Index set: ${indexSet.size}") + val queryable = embeddingIndexer + .indexEmbeddings( + appendable, + buildRecorder, + indexSet, + concurrencyLevel + ).onSuccess(_ => logger.info(s"Done indexing..")) + + queryable + } +} diff --git a/ann/src/main/scala/com/twitter/ann/service/loadtest/AnnLoadTestMain.scala b/ann/src/main/scala/com/twitter/ann/service/loadtest/AnnLoadTestMain.scala new file mode 100644 index 0000000000..23f18d7af3 --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/service/loadtest/AnnLoadTestMain.scala @@ -0,0 +1,379 @@ +package com.twitter.ann.service.loadtest + +import com.twitter.ann.annoy.AnnoyCommon +import com.twitter.ann.annoy.AnnoyRuntimeParams +import com.twitter.ann.annoy.TypedAnnoyIndex +import com.twitter.ann.common._ +import com.twitter.ann.common.thriftscala.{Distance => ServiceDistance} +import com.twitter.ann.common.thriftscala.{RuntimeParams => ServiceRuntimeParams} +import com.twitter.ann.faiss.FaissCommon +import com.twitter.ann.faiss.FaissParams +import com.twitter.ann.hnsw.HnswCommon +import com.twitter.ann.hnsw.HnswParams +import com.twitter.ann.hnsw.TypedHnswIndex +import com.twitter.bijection.Injection +import com.twitter.cortex.ml.embeddings.common.EntityKind +import com.twitter.finagle.mtls.authentication.ServiceIdentifier +import com.twitter.finagle.util.DefaultTimer +import com.twitter.finatra.mtls.modules.ServiceIdentifierModule +import com.twitter.inject.server.TwitterServer +import com.twitter.util._ +import java.util.concurrent.TimeUnit + +/** + * To build and upload: + * $ ./bazel bundle ann/src/main/scala/com/twitter/ann/service/loadtest:bin --bundle-jvm-archive=zip + * $ packer add_version --cluster=smf1 $USER ann-loadtest dist/ann-loadtest.zip + */ +object AnnLoadTestMain extends TwitterServer { + private[this] val algo = + flag[String]("algo", "load test server types: [annoy/hnsw]") + private[this] val targetQPS = + flag[Int]("qps", "target QPS for load test") + private[this] val queryIdType = + flag[String]( + "query_id_type", + "query id type for load test: [long/string/int/user/tweet/word/url/tfwId]") + private[this] val indexIdType = + flag[String]( + "index_id_type", + "index id type for load test: [long/string/int/user/tweet/word/url/tfwId]") + private[this] val metric = + flag[String]("metric", "metric type for load test: [Cosine/L2/InnerProduct]") + private[this] val durationSec = + flag[Int]("duration_sec", "duration for the load test in sec") + private[this] val numberOfNeighbors = + flag[Seq[Int]]("number_of_neighbors", Seq(), "number of neighbors") + private[this] val dimension = flag[Int]("embedding_dimension", "dimension of embeddings") + private[this] val querySetDir = + flag[String]("query_set_dir", "", "Directory containing the queries") + private[this] val indexSetDir = + flag[String]( + "index_set_dir", + "", + "Directory containing the embeddings to be indexed" + ) + private[this] val truthSetDir = + flag[String]("truth_set_dir", "", "Directory containing the truth data") + private[this] val loadTestType = + flag[String]("loadtest_type", "Load test type [server/local]") + private[this] val serviceDestination = + flag[String]("service_destination", "wily address of remote query service") + private[this] val concurrencyLevel = + flag[Int]("concurrency_level", 8, "number of concurrent operations on the index") + + // Queries with random embeddings + private[this] val withRandomQueries = + flag[Boolean]("with_random_queries", false, "query with random embeddings") + private[this] val randomQueriesCount = + flag[Int]("random_queries_count", 50000, "total random queries") + private[this] val randomEmbeddingMinValue = + flag[Float]("random_embedding_min_value", -1.0f, "Min value of random embeddings") + private[this] val randomEmbeddingMaxValue = + flag[Float]("random_embedding_max_value", 1.0f, "Max value of random embeddings") + + // parameters for annoy + private[this] val numOfNodesToExplore = + flag[Seq[Int]]("annoy_num_of_nodes_to_explore", Seq(), "number of nodes to explore") + private[this] val numOfTrees = + flag[Int]("annoy_num_trees", 0, "number of trees to build") + + // parameters for HNSW + private[this] val efConstruction = flag[Int]("hnsw_ef_construction", "ef for Hnsw construction") + private[this] val ef = flag[Seq[Int]]("hnsw_ef", Seq(), "ef for Hnsw query") + private[this] val maxM = flag[Int]("hnsw_max_m", "maxM for Hnsw") + + // FAISS + private[this] val nprobe = flag[Seq[Int]]("faiss_nprobe", Seq(), "nprobe for faiss query") + private[this] val quantizerEf = + flag[Seq[Int]]("faiss_quantizerEf", Seq(0), "quantizerEf for faiss query") + private[this] val quantizerKfactorRF = + flag[Seq[Int]]("faiss_quantizerKfactorRF", Seq(0), "quantizerEf for faiss query") + private[this] val quantizerNprobe = + flag[Seq[Int]]("faiss_quantizerNprobe", Seq(0), "quantizerNprobe for faiss query") + private[this] val ht = + flag[Seq[Int]]("faiss_ht", Seq(0), "ht for faiss query") + + implicit val timer: Timer = DefaultTimer + + override def start(): Unit = { + logger.info("Starting load test..") + logger.info(flag.getAll().mkString("\t")) + + assert(numberOfNeighbors().nonEmpty, "number_of_neighbors not defined") + assert(dimension() > 0, s"Invalid dimension ${dimension()}") + + val inMemoryBuildRecorder = new InMemoryLoadTestBuildRecorder + + val queryableFuture = buildQueryable(inMemoryBuildRecorder) + val queryConfig = getQueryRuntimeConfig + val result = queryableFuture.flatMap { queryable => + performQueries(queryable, queryConfig, getQueries) + } + + Await.result(result) + System.out.println(s"Target QPS: ${targetQPS()}") + System.out.println(s"Duration per test: ${durationSec()}") + System.out.println(s"Concurrency Level: ${concurrencyLevel()}") + + LoadTestUtils + .printResults(inMemoryBuildRecorder, queryConfig) + .foreach(System.out.println) + + Await.result(close()) + System.exit(0) + } + + private[this] def getQueries[Q, I]: Seq[Query[I]] = { + if (withRandomQueries()) { + assert( + truthSetDir().isEmpty, + "Cannot use truth set when query with random embeddings enabled" + ) + val queries = LoadTestUtils.getRandomQuerySet( + dimension(), + randomQueriesCount(), + randomEmbeddingMinValue(), + randomEmbeddingMaxValue() + ) + + queries.map(Query[I](_)) + } else { + assert(querySetDir().nonEmpty, "Query set path is empty") + assert(queryIdType().nonEmpty, "Query id type is empty") + val queries = LoadTestUtils.getEmbeddingsSet[Q](querySetDir(), queryIdType()) + + if (truthSetDir().nonEmpty) { + // Join the queries with truth set data. + assert(indexIdType().nonEmpty, "Index id type is empty") + val truthSetMap = + LoadTestUtils.getTruthSetMap[Q, I](truthSetDir(), queryIdType(), indexIdType()) + queries.map(entity => Query[I](entity.embedding, truthSetMap(entity.id))) + } else { + queries.map(entity => Query[I](entity.embedding)) + } + } + } + + private[this] def getQueryRuntimeConfig[ + T, + P <: RuntimeParams + ]: Seq[QueryTimeConfiguration[T, P]] = { + val queryTimeConfig = algo() match { + case "annoy" => + assert(numOfNodesToExplore().nonEmpty, "Must specify the num_of_nodes_to_explore") + logger.info(s"Querying annoy index with num_of_nodes_to_explore ${numOfNodesToExplore()}") + for { + numNodes <- numOfNodesToExplore() + numOfNeighbors <- numberOfNeighbors() + } yield { + buildQueryTimeConfig[T, AnnoyRuntimeParams]( + numOfNeighbors, + AnnoyRuntimeParams(Some(numNodes)), + Map( + "numNodes" -> numNodes.toString, + "numberOfNeighbors" -> numOfNeighbors.toString + ) + ).asInstanceOf[QueryTimeConfiguration[T, P]] + } + case "hnsw" => + assert(ef().nonEmpty, "Must specify ef") + logger.info(s"Querying hnsw index with ef ${ef()}") + for { + ef <- ef() + numOfNeighbors <- numberOfNeighbors() + } yield { + buildQueryTimeConfig[T, HnswParams]( + numOfNeighbors, + HnswParams(ef), + Map( + "efConstruction" -> ef.toString, + "numberOfNeighbors" -> numOfNeighbors.toString + ) + ).asInstanceOf[QueryTimeConfiguration[T, P]] + } + case "faiss" => + assert(nprobe().nonEmpty, "Must specify nprobe") + def toNonZeroOptional(x: Int): Option[Int] = if (x != 0) Some(x) else None + for { + numOfNeighbors <- numberOfNeighbors() + runNProbe <- nprobe() + runQEF <- quantizerEf() + runKFactorEF <- quantizerKfactorRF() + runQNProbe <- quantizerNprobe() + runHT <- ht() + } yield { + val params = FaissParams( + Some(runNProbe), + toNonZeroOptional(runQEF), + toNonZeroOptional(runKFactorEF), + toNonZeroOptional(runQNProbe), + toNonZeroOptional(runHT)) + buildQueryTimeConfig[T, FaissParams]( + numOfNeighbors, + params, + Map( + "nprobe" -> params.nprobe.toString, + "quantizer_efSearch" -> params.quantizerEf.toString, + "quantizer_k_factor_rf" -> params.quantizerKFactorRF.toString, + "quantizer_nprobe" -> params.quantizerNprobe.toString, + "ht" -> params.ht.toString, + "numberOfNeighbors" -> numOfNeighbors.toString, + ) + ).asInstanceOf[QueryTimeConfiguration[T, P]] + } + case _ => throw new IllegalArgumentException(s"server type: $algo is not supported yet") + } + + queryTimeConfig + } + + private def buildQueryable[T, P <: RuntimeParams, D <: Distance[D]]( + inMemoryBuildRecorder: InMemoryLoadTestBuildRecorder + ): Future[Queryable[T, P, D]] = { + val queryable = loadTestType() match { + case "remote" => { + assert(serviceDestination().nonEmpty, "Service destination not defined") + logger.info(s"Running load test with remote service ${serviceDestination()}") + LoadTestUtils.buildRemoteServiceQueryClient[T, P, D]( + serviceDestination(), + "ann-load-test", + statsReceiver, + injector.instance[ServiceIdentifier], + getRuntimeParamInjection[P], + getDistanceInjection[D], + getIndexIdInjection[T] + ) + } + case "local" => { + logger.info("Running load test locally..") + assert(indexSetDir().nonEmpty, "Index set path is empty") + val statsLoadTestBuildRecorder = new StatsLoadTestBuildRecorder(statsReceiver) + val buildRecorder = + new ComposedLoadTestBuildRecorder(Seq(inMemoryBuildRecorder, statsLoadTestBuildRecorder)) + indexEmbeddingsAndGetQueryable[T, P, D]( + buildRecorder, + LoadTestUtils.getEmbeddingsSet(indexSetDir(), indexIdType()) + ) + } + } + queryable + } + + private def indexEmbeddingsAndGetQueryable[T, P <: RuntimeParams, D <: Distance[D]]( + buildRecorder: LoadTestBuildRecorder, + indexSet: Seq[EntityEmbedding[T]] + ): Future[Queryable[T, P, D]] = { + logger.info(s"Indexing entity embeddings in index set with size ${indexSet.size}") + val metric = getDistanceMetric[D] + val indexIdInjection = getIndexIdInjection[T] + val indexBuilder = new AnnIndexBuildLoadTest(buildRecorder) + val appendable = algo() match { + case "annoy" => + assert(numOfTrees() > 0, "Must specify the number of trees for annoy") + logger.info( + s"Creating annoy index locally with num_of_trees: ${numOfTrees()}" + ) + TypedAnnoyIndex + .indexBuilder( + dimension(), + numOfTrees(), + metric, + indexIdInjection, + FuturePool.interruptibleUnboundedPool + ) + case "hnsw" => + assert(efConstruction() > 0 && maxM() > 0, "Must specify ef_construction and max_m") + logger.info( + s"Creating hnsw index locally with max_m: ${maxM()} and ef_construction: ${efConstruction()}" + ) + TypedHnswIndex + .index[T, D]( + dimension(), + metric, + efConstruction(), + maxM(), + indexSet.size, + ReadWriteFuturePool(FuturePool.interruptibleUnboundedPool) + ) + } + + indexBuilder + .indexEmbeddings(appendable, indexSet, concurrencyLevel()) + .asInstanceOf[Future[Queryable[T, P, D]]] + } + + private[this] def performQueries[T, P <: RuntimeParams, D <: Distance[D]]( + queryable: Queryable[T, P, D], + queryTimeConfig: Seq[QueryTimeConfiguration[T, P]], + queries: Seq[Query[T]] + ): Future[Unit] = { + val indexQuery = new AnnIndexQueryLoadTest() + val duration = Duration(durationSec().toLong, TimeUnit.SECONDS) + indexQuery.performQueries( + queryable, + targetQPS(), + duration, + queries, + concurrencyLevel(), + queryTimeConfig + ) + } + + // provide index id injection based on argument + private[this] def getIndexIdInjection[T]: Injection[T, Array[Byte]] = { + val injection = indexIdType() match { + case "long" => AnnInjections.LongInjection + case "string" => AnnInjections.StringInjection + case "int" => AnnInjections.IntInjection + case entityKind => EntityKind.getEntityKind(entityKind).byteInjection + } + injection.asInstanceOf[Injection[T, Array[Byte]]] + } + + private[this] def getRuntimeParamInjection[ + P <: RuntimeParams + ]: Injection[P, ServiceRuntimeParams] = { + val injection = algo() match { + case "annoy" => AnnoyCommon.RuntimeParamsInjection + case "hnsw" => HnswCommon.RuntimeParamsInjection + case "faiss" => FaissCommon.RuntimeParamsInjection + } + + injection.asInstanceOf[Injection[P, ServiceRuntimeParams]] + } + + // provide distance injection based on argument + private[this] def getDistanceInjection[D <: Distance[D]]: Injection[D, ServiceDistance] = { + Metric.fromString(metric()).asInstanceOf[Injection[D, ServiceDistance]] + } + + private[this] def getDistanceMetric[D <: Distance[D]]: Metric[D] = { + Metric.fromString(metric()).asInstanceOf[Metric[D]] + } + + private[this] def buildQueryTimeConfig[T, P <: RuntimeParams]( + numOfNeighbors: Int, + params: P, + config: Map[String, String] + ): QueryTimeConfiguration[T, P] = { + val printableQueryRecorder = new InMemoryLoadTestQueryRecorder[T]() + val scope = config.flatMap { case (key, value) => Seq(key, value.toString) }.toSeq + val statsLoadTestQueryRecorder = new StatsLoadTestQueryRecorder[T]( + // Put the run time params in the stats receiver names so that we can tell the difference when + // we look at them later. + statsReceiver.scope(algo()).scope(scope: _*) + ) + val queryRecorder = new ComposedLoadTestQueryRecorder( + Seq(printableQueryRecorder, statsLoadTestQueryRecorder) + ) + QueryTimeConfiguration( + queryRecorder, + params, + numOfNeighbors, + printableQueryRecorder + ) + } + + override protected def modules: Seq[com.google.inject.Module] = Seq(ServiceIdentifierModule) +} diff --git a/ann/src/main/scala/com/twitter/ann/service/loadtest/AnnLoadTestWorker.scala b/ann/src/main/scala/com/twitter/ann/service/loadtest/AnnLoadTestWorker.scala new file mode 100644 index 0000000000..87ed6a1b32 --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/service/loadtest/AnnLoadTestWorker.scala @@ -0,0 +1,116 @@ +package com.twitter.ann.service.loadtest + +import com.google.common.annotations.VisibleForTesting +import com.twitter.ann.common.Distance +import com.twitter.ann.common.Queryable +import com.twitter.ann.common.RuntimeParams +import com.twitter.concurrent.AsyncMeter +import com.twitter.concurrent.AsyncStream +import com.twitter.finagle.util.DefaultTimer +import com.twitter.logging.Logger +import com.twitter.util.Duration +import com.twitter.util.Future +import com.twitter.util.Stopwatch +import com.twitter.util.Timer +import com.twitter.util.Try +import java.util.concurrent.atomic.AtomicInteger + +object QueryTimeConfiguration { + val ResultHeader = + "params\tnumNeighbors\trecall@1\trecall@10\trecall\tavgLatencyMicros\tp50LatencyMicros\tp90LatencyMicros\tp99LatencyMicros\tavgRPS" +} + +case class QueryTimeConfiguration[T, P <: RuntimeParams]( + recorder: LoadTestQueryRecorder[T], + param: P, + numberOfNeighbors: Int, + private val results: InMemoryLoadTestQueryRecorder[T]) { + override def toString: String = + s"QueryTimeConfiguration(param = $param, numberOfNeighbors = $numberOfNeighbors)" + + def printResults: String = { + val snapshot = results.computeSnapshot() + s"$param\t$numberOfNeighbors\t${results.top1Recall}\t${results.top10Recall}\t${results.recall}\t${snapshot.avgQueryLatencyMicros}\t${snapshot.p50QueryLatencyMicros}\t${snapshot.p90QueryLatencyMicros}\t${snapshot.p99QueryLatencyMicros}\t${results.avgRPS}" + } +} + +/** + * Basic worker for ANN benchmark, send query with configured QPS and record results + */ +class AnnLoadTestWorker( + timer: Timer = DefaultTimer) { + private val logger = Logger() + + /** + * @param queries List of tuple of query embedding and corresponding list of truth set of ids associated with the embedding + * @param qps the maximum number of request per second to send to the queryable. Note that if you + * do not set the concurrency level high enough you may not be able to achieve this. + * @param duration how long to perform the load test. + * @param configuration Query configuration encapsulating runtime params and recorder. + * @param concurrencyLevel The maximum number of concurrent requests to the queryable at a time. + * Note that you may not be able to achieve this number of concurrent + * requests if you do not have the QPS set high enough. + * + * @return a Future that completes when the load test is over. It contains the number of requests + * sent. + */ + def runWithQps[T, P <: RuntimeParams, D <: Distance[D]]( + queryable: Queryable[T, P, D], + queries: Seq[Query[T]], + qps: Int, + duration: Duration, + configuration: QueryTimeConfiguration[T, P], + concurrencyLevel: Int + ): Future[Int] = { + val elapsed = Stopwatch.start() + val atomicInteger = new AtomicInteger(0) + val fullStream = Stream.continually { + if (elapsed() <= duration) { + logger.ifDebug(s"running with config: $configuration") + Some(atomicInteger.getAndIncrement() % queries.size) + } else { + logger.ifDebug(s"stopping with config: $configuration") + None + } + } + val limitedStream = fullStream.takeWhile(_.isDefined).flatten + // at most we will have concurrencyLevel concurrent requests. So we should never have more than + // concurrency level waiters. + val asyncMeter = AsyncMeter.perSecond(qps, concurrencyLevel)(timer) + + Future.Unit.before { + AsyncStream + .fromSeq(limitedStream).mapConcurrent(concurrencyLevel) { index => + asyncMeter.await(1).flatMap { _ => + performQuery(configuration, queryable, queries(index)) + } + }.size + } + } + + @VisibleForTesting + private[loadtest] def performQuery[T, P <: RuntimeParams, D <: Distance[D]]( + configuration: QueryTimeConfiguration[T, P], + queryable: Queryable[T, P, D], + query: Query[T] + ): Future[Try[Unit]] = { + val elapsed = Stopwatch.start() + queryable + .query(query.embedding, configuration.numberOfNeighbors, configuration.param) + .onSuccess { res: List[T] => + // underneath LoadTestRecorder will record results for load test + // knnMap should be truncated to be same size as query result + configuration.recorder.recordQueryResult( + query.trueNeighbours, + res, + elapsed.apply() + ) + logger.ifDebug(s"Successful query for $query") + } + .onFailure { e => + logger.error(s"Failed query for $query: " + e) + } + .unit + .liftToTry + } +} diff --git a/ann/src/main/scala/com/twitter/ann/service/loadtest/BUILD b/ann/src/main/scala/com/twitter/ann/service/loadtest/BUILD new file mode 100644 index 0000000000..9aa6982428 --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/service/loadtest/BUILD @@ -0,0 +1,31 @@ +scala_library( + name = "loadtest", + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + tags = ["bazel-compatible"], + dependencies = [ + "ann/src/main/resources", + "ann/src/main/scala/com/twitter/ann/annoy", + "ann/src/main/scala/com/twitter/ann/common", + "ann/src/main/scala/com/twitter/ann/faiss", + "ann/src/main/scala/com/twitter/ann/hnsw", + "ann/src/main/scala/com/twitter/ann/util", + "ann/src/main/thrift/com/twitter/ann/common:ann-common-scala", + "finatra/inject/inject-server/src/main/scala", + "src/scala/com/twitter/cortex/ml/embeddings/common:Helpers", + "twitter-server-internal/src/main/scala", + "util/util-logging/src/main/scala", + ], +) + +jvm_binary( + name = "bin", + basename = "ann-loadtest", + main = "com.twitter.ann.service.loadtest.AnnLoadTestMain", + runtime_platform = "java11", + dependencies = [ + ":loadtest", + "3rdparty/jvm/org/slf4j:slf4j-jdk14", + "twitter-server/slf4j-jdk14/src/main/scala", + ], +) diff --git a/ann/src/main/scala/com/twitter/ann/service/loadtest/EmbeddingIndexer.scala b/ann/src/main/scala/com/twitter/ann/service/loadtest/EmbeddingIndexer.scala new file mode 100644 index 0000000000..b963c3eb79 --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/service/loadtest/EmbeddingIndexer.scala @@ -0,0 +1,28 @@ +package com.twitter.ann.service.loadtest + +import com.twitter.ann.common.{Appendable, Distance, EntityEmbedding, Queryable, RuntimeParams} +import com.twitter.ann.util.IndexBuilderUtils +import com.twitter.util.{Future, Stopwatch} + +class EmbeddingIndexer { + // Index embeddings into Appendable and return the (appendable, latency) pair + // we need to return appendable itself here because for Annoy, we need to build + // appendable and serialize it first, and then we could query with index directory + // once we are confident to remove Annoy, should clean up this method. + def indexEmbeddings[T, P <: RuntimeParams, D <: Distance[D]]( + appendable: Appendable[T, P, D], + recorder: LoadTestBuildRecorder, + indexSet: Seq[EntityEmbedding[T]], + concurrencyLevel: Int + ): Future[Queryable[T, P, D]] = { + val indexBuildingTimeElapsed = Stopwatch.start() + val future = IndexBuilderUtils.addToIndex(appendable, indexSet, concurrencyLevel) + future.map { _ => + val indexBuildingTime = indexBuildingTimeElapsed() + val toQueryableElapsed = Stopwatch.start() + val queryable = appendable.toQueryable + recorder.recordIndexCreation(indexSet.size, indexBuildingTime, toQueryableElapsed()) + queryable + } + } +} diff --git a/ann/src/main/scala/com/twitter/ann/service/loadtest/LoadTestRecorder.scala b/ann/src/main/scala/com/twitter/ann/service/loadtest/LoadTestRecorder.scala new file mode 100644 index 0000000000..0bcdc9be81 --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/service/loadtest/LoadTestRecorder.scala @@ -0,0 +1,231 @@ +package com.twitter.ann.service.loadtest + +import com.google.common.util.concurrent.AtomicDouble +import com.twitter.finagle.stats.{MetricsBucketedHistogram, Snapshot, StatsReceiver} +import com.twitter.util.{Duration, Stopwatch} +import java.util.concurrent.atomic.{AtomicInteger, AtomicReference} + +trait LoadTestQueryRecorder[T] { + def recordQueryResult( + trueNeighbors: Seq[T], + foundNeighbors: Seq[T], + queryLatency: Duration + ): Unit +} + +case class LoadTestQueryResults( + numResults: Int, + top1Recall: Float, + top10Recall: Option[Float], + overallRecall: Float) + +private object LoadTestQueryRecorder { + def recordQueryResult[T]( + trueNeighbors: Seq[T], + foundNeighbors: Seq[T] + ): LoadTestQueryResults = { + // record number of results returned + val numResults = foundNeighbors.size + if (trueNeighbors.isEmpty) { + LoadTestQueryResults( + numResults, + 0f, + Option.empty, + 0f + ) + } else { + // record top 1, top 10 and overall recall + // recall here is computed as number of true neighbors within the returned points set + // divides by the number of required neighbors + val top1Recall = foundNeighbors.intersect(Seq(trueNeighbors.head)).size + val top10Recall = if (numResults >= 10 && trueNeighbors.size >= 10) { + Some( + trueNeighbors.take(10).intersect(foundNeighbors).size.toFloat / 10 + ) + } else { + None + } + + val overallRecall = trueNeighbors + .take(foundNeighbors.size).intersect(foundNeighbors).size.toFloat / + Math.min(foundNeighbors.size, trueNeighbors.size) + + LoadTestQueryResults( + numResults, + top1Recall, + top10Recall, + overallRecall + ) + } + } +} + +class StatsLoadTestQueryRecorder[T]( + statsReceiver: StatsReceiver) + extends LoadTestQueryRecorder[T] { + private[this] val numResultsStats = statsReceiver.stat("number_of_results") + private[this] val recallStats = statsReceiver.stat("recall") + private[this] val top1RecallStats = statsReceiver.stat("top_1_recall") + private[this] val top10RecallStats = statsReceiver.stat("top_10_recall") + private[this] val queryLatencyMicrosStats = statsReceiver.stat("query_latency_micros") + + override def recordQueryResult( + trueNeighbors: Seq[T], + foundNeighbors: Seq[T], + queryLatency: Duration + ): Unit = { + val results = LoadTestQueryRecorder.recordQueryResult(trueNeighbors, foundNeighbors) + numResultsStats.add(results.numResults) + recallStats.add(results.overallRecall * 100) + results.top10Recall.foreach { top10Recall => + top10RecallStats.add(top10Recall * 100) + } + top1RecallStats.add(results.top1Recall * 100) + queryLatencyMicrosStats.add(queryLatency.inMicroseconds) + } +} + +trait LoadTestBuildRecorder { + def recordIndexCreation( + indexSize: Int, + indexLatency: Duration, + toQueryableLatency: Duration + ): Unit +} + +class StatsLoadTestBuildRecorder( + statsReceiver: StatsReceiver) + extends LoadTestBuildRecorder { + private[this] val indexLatencyGauge = statsReceiver.addGauge("index_latency_ms")(_) + private[this] val indexSizeGauge = statsReceiver.addGauge("index_size")(_) + private[this] val toQueryableGauge = statsReceiver.addGauge("to_queryable_latency_ms")(_) + + override def recordIndexCreation( + indexSize: Int, + indexLatency: Duration, + toQueryableLatency: Duration + ): Unit = { + indexLatencyGauge(indexLatency.inMillis) + indexSizeGauge(indexSize) + toQueryableGauge(toQueryableLatency.inMillis) + } +} + +class QueryRecorderSnapshot(snapshot: Snapshot) { + def avgQueryLatencyMicros: Double = snapshot.average + def p50QueryLatencyMicros: Double = + snapshot.percentiles.find(_.quantile == .5).get.value + def p90QueryLatencyMicros: Double = + snapshot.percentiles.find(_.quantile == .9).get.value + def p99QueryLatencyMicros: Double = + snapshot.percentiles.find(_.quantile == .99).get.value +} + +class InMemoryLoadTestQueryRecorder[T]( + // You have to specify a name of the histogram even though it is not used + // Use latch period of bottom. We will compute a new snapshot every time we call computeSnapshot + private[this] val latencyHistogram: MetricsBucketedHistogram = + new MetricsBucketedHistogram("latencyhistogram", latchPeriod = Duration.Bottom)) + extends LoadTestQueryRecorder[T] { + private[this] val counter = new AtomicInteger(0) + private[this] val countMoreThan10Results = new AtomicInteger(0) + private[this] val recallSum = new AtomicDouble(0.0) + private[this] val top1RecallSum = new AtomicDouble(0.0) + private[this] val top10RecallSum = new AtomicDouble(0.0) + private[this] val elapsedTimeFun = new AtomicReference[(Stopwatch.Elapsed, Duration)]() + private[this] val elapsedTime = new AtomicReference[Duration](Duration.Zero) + + /** + * Compute a snapshot of what happened between the time that this was called and the previous time + * it was called. + * @return + */ + def computeSnapshot(): QueryRecorderSnapshot = { + new QueryRecorderSnapshot(latencyHistogram.snapshot()) + } + + def recall: Double = + if (counter.get() != 0) { + recallSum.get * 100 / counter.get() + } else { 0 } + + def top1Recall: Double = + if (counter.get() != 0) { + top1RecallSum.get * 100 / counter.get() + } else { 0 } + def top10Recall: Double = + if (countMoreThan10Results.get() != 0) { + top10RecallSum.get * 100 / countMoreThan10Results.get() + } else { 0 } + + def avgRPS: Double = + if (elapsedTime.get() != Duration.Zero) { + (counter.get().toDouble * 1e9) / elapsedTime.get().inNanoseconds + } else { 0 } + + override def recordQueryResult( + trueNeighbors: Seq[T], + foundNeighbors: Seq[T], + queryLatency: Duration + ): Unit = { + elapsedTimeFun.compareAndSet(null, (Stopwatch.start(), queryLatency)) + val results = LoadTestQueryRecorder.recordQueryResult(trueNeighbors, foundNeighbors) + top1RecallSum.addAndGet(results.top1Recall) + results.top10Recall.foreach { top10Recall => + top10RecallSum.addAndGet(top10Recall) + countMoreThan10Results.incrementAndGet() + } + recallSum.addAndGet(results.overallRecall) + latencyHistogram.add(queryLatency.inMicroseconds) + counter.incrementAndGet() + // Requests are assumed to have started around the time time of the first time record was called + // plus the time it took for that query to hhave completed. + val (elapsedSinceFirstCall, firstQueryLatency) = elapsedTimeFun.get() + val durationSoFar = elapsedSinceFirstCall() + firstQueryLatency + elapsedTime.set(durationSoFar) + } +} + +class InMemoryLoadTestBuildRecorder extends LoadTestBuildRecorder { + var indexLatency: Duration = Duration.Zero + var indexSize: Int = 0 + var toQueryableLatency: Duration = Duration.Zero + + override def recordIndexCreation( + size: Int, + indexLatencyArg: Duration, + toQueryableLatencyArg: Duration + ): Unit = { + indexLatency = indexLatencyArg + indexSize = size + toQueryableLatency = toQueryableLatencyArg + } +} + +/** + * A LoadTestRecorder that be composed by other recorders + */ +class ComposedLoadTestQueryRecorder[T]( + recorders: Seq[LoadTestQueryRecorder[T]]) + extends LoadTestQueryRecorder[T] { + override def recordQueryResult( + trueNeighbors: Seq[T], + foundNeighbors: Seq[T], + queryLatency: Duration + ): Unit = recorders.foreach { + _.recordQueryResult(trueNeighbors, foundNeighbors, queryLatency) + } +} + +/** + * A LoadTestRecorder that be composed by other recorders + */ +class ComposedLoadTestBuildRecorder( + recorders: Seq[LoadTestBuildRecorder]) + extends LoadTestBuildRecorder { + override def recordIndexCreation( + indexSize: Int, + indexLatency: Duration, + toQueryableLatency: Duration + ): Unit = recorders.foreach { _.recordIndexCreation(indexSize, indexLatency, toQueryableLatency) } +} diff --git a/ann/src/main/scala/com/twitter/ann/service/loadtest/LoadTestUtils.scala b/ann/src/main/scala/com/twitter/ann/service/loadtest/LoadTestUtils.scala new file mode 100644 index 0000000000..b1009cc623 --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/service/loadtest/LoadTestUtils.scala @@ -0,0 +1,200 @@ +package com.twitter.ann.service.loadtest + +import com.google.common.annotations.VisibleForTesting +import com.twitter.ann.common.EmbeddingType.EmbeddingVector +import com.twitter.ann.common.thriftscala.AnnQueryService +import com.twitter.ann.common.thriftscala.NearestNeighborQuery +import com.twitter.ann.common.thriftscala.NearestNeighborResult +import com.twitter.ann.common.thriftscala.{Distance => ServiceDistance} +import com.twitter.ann.common.thriftscala.{RuntimeParams => ServiceRuntimeParams} +import com.twitter.ann.common.Distance +import com.twitter.ann.common.EntityEmbedding +import com.twitter.ann.common.Queryable +import com.twitter.ann.common.RuntimeParams +import com.twitter.ann.common.ServiceClientQueryable +import com.twitter.bijection.Injection +import com.twitter.cortex.ml.embeddings.common.EntityKind +import com.twitter.finagle.builder.ClientBuilder +import com.twitter.finagle.mtls.authentication.ServiceIdentifier +import com.twitter.finagle.mtls.client.MtlsStackClient.MtlsThriftMuxClientSyntax +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.finagle.thrift.ClientId +import com.twitter.finagle.Service +import com.twitter.finagle.ThriftMux +import com.twitter.ml.api.embedding.Embedding +import com.twitter.search.common.file.AbstractFile.Filter +import com.twitter.search.common.file.AbstractFile +import com.twitter.search.common.file.FileUtils +import com.twitter.search.common.file.LocalFile +import com.twitter.util.Future +import com.twitter.util.logging.Logger +import java.io.File +import scala.collection.JavaConversions._ +import scala.collection.mutable +import scala.util.Random + +object LoadTestUtils { + lazy val Log = Logger(getClass.getName) + + private[this] val LocalPath = "." + private[this] val RNG = new Random(100) + + private[loadtest] def getTruthSetMap[Q, I]( + directory: String, + queryIdType: String, + indexIdType: String + ): Map[Q, Seq[I]] = { + Log.info(s"Loading truth set from ${directory}") + val queryConverter = getKeyConverter[Q](queryIdType) + val indexConverter = getKeyConverter[I](indexIdType) + val res = loadKnnDirFileToMap( + getLocalFileHandle(directory), + // Knn truth file tsv format: [id neighbor:distance neighbor:distance ...] + arr => { arr.map(str => indexConverter(str.substring(0, str.lastIndexOf(":")))).toSeq }, + queryConverter + ) + assert(res.nonEmpty, s"Must have some something in the truth set ${directory}") + res + } + + private[this] def getLocalFileHandle( + directory: String + ): AbstractFile = { + val fileHandle = FileUtils.getFileHandle(directory) + if (fileHandle.isInstanceOf[LocalFile]) { + fileHandle + } else { + val localFileHandle = + FileUtils.getFileHandle(s"${LocalPath}${File.separator}${fileHandle.getName}") + fileHandle.copyTo(localFileHandle) + localFileHandle + } + } + + private[loadtest] def getEmbeddingsSet[T]( + directory: String, + idType: String + ): Seq[EntityEmbedding[T]] = { + Log.info(s"Loading embeddings from ${directory}") + val res = loadKnnDirFileToMap( + getLocalFileHandle(directory), + arr => { arr.map(_.toFloat) }, + getKeyConverter[T](idType) + ).map { case (key, value) => EntityEmbedding[T](key, Embedding(value.toArray)) }.toSeq + assert(res.nonEmpty, s"Must have some something in the embeddings set ${directory}") + res + } + + private[this] def loadKnnDirFileToMap[K, V]( + directory: AbstractFile, + f: Array[String] => Seq[V], + converter: String => K + ): Map[K, Seq[V]] = { + val map = mutable.HashMap[K, Seq[V]]() + directory + .listFiles(new Filter { + override def accept(file: AbstractFile): Boolean = + file.getName != AbstractFile.SUCCESS_FILE_NAME + }).foreach { file => + asScalaBuffer(file.readLines()).foreach { line => + addToMapFromKnnString(line, f, map, converter) + } + } + map.toMap + } + + // Generating random float with value range bounded between minValue and maxValue + private[loadtest] def getRandomQuerySet( + dimension: Int, + totalQueries: Int, + minValue: Float, + maxValue: Float + ): Seq[EmbeddingVector] = { + Log.info( + s"Generating $totalQueries random queries for dimension $dimension with value between $minValue and $maxValue...") + assert(totalQueries > 0, s"Total random queries $totalQueries should be greater than 0") + assert( + maxValue > minValue, + s"Random embedding max value should be greater than min value. min: $minValue max: $maxValue") + (1 to totalQueries).map { _ => + val embedding = Array.fill(dimension)(minValue + (maxValue - minValue) * RNG.nextFloat()) + Embedding(embedding) + } + } + + private[this] def getKeyConverter[T](idType: String): String => T = { + val converter = idType match { + case "long" => + (s: String) => s.toLong + case "string" => + (s: String) => s + case "int" => + (s: String) => s.toInt + case entityKind => + (s: String) => EntityKind.getEntityKind(entityKind).stringInjection.invert(s).get + } + converter.asInstanceOf[String => T] + } + + private[loadtest] def buildRemoteServiceQueryClient[T, P <: RuntimeParams, D <: Distance[D]]( + destination: String, + clientId: String, + statsReceiver: StatsReceiver, + serviceIdentifier: ServiceIdentifier, + runtimeParamInjection: Injection[P, ServiceRuntimeParams], + distanceInjection: Injection[D, ServiceDistance], + indexIdInjection: Injection[T, Array[Byte]] + ): Future[Queryable[T, P, D]] = { + val client: AnnQueryService.MethodPerEndpoint = new AnnQueryService.FinagledClient( + service = ClientBuilder() + .reportTo(statsReceiver) + .dest(destination) + .stack(ThriftMux.client.withMutualTls(serviceIdentifier).withClientId(ClientId(clientId))) + .build(), + stats = statsReceiver + ) + + val service = new Service[NearestNeighborQuery, NearestNeighborResult] { + override def apply(request: NearestNeighborQuery): Future[NearestNeighborResult] = + client.query(request) + } + + Future.value( + new ServiceClientQueryable[T, P, D]( + service, + runtimeParamInjection, + distanceInjection, + indexIdInjection + ) + ) + } + + // helper method to convert a line in KNN file output format into map + @VisibleForTesting + def addToMapFromKnnString[K, V]( + line: String, + f: Array[String] => Seq[V], + map: mutable.HashMap[K, Seq[V]], + converter: String => K + ): Unit = { + val items = line.split("\t") + map += converter(items(0)) -> f(items.drop(1)) + } + + def printResults( + inMemoryBuildRecorder: InMemoryLoadTestBuildRecorder, + queryTimeConfigurations: Seq[QueryTimeConfiguration[_, _]] + ): Seq[String] = { + val queryTimeConfigStrings = queryTimeConfigurations.map { config => + config.printResults + } + + Seq( + "Build results", + "indexingTimeSecs\ttoQueryableTimeMs\tindexSize", + s"${inMemoryBuildRecorder.indexLatency.inSeconds}\t${inMemoryBuildRecorder.toQueryableLatency.inMilliseconds}\t${inMemoryBuildRecorder.indexSize}", + "Query results", + QueryTimeConfiguration.ResultHeader + ) ++ queryTimeConfigStrings + } +} diff --git a/ann/src/main/scala/com/twitter/ann/service/loadtest/README.md b/ann/src/main/scala/com/twitter/ann/service/loadtest/README.md new file mode 100644 index 0000000000..b668788eab --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/service/loadtest/README.md @@ -0,0 +1,218 @@ +# Loadtest ANN query service with random embeddings + +An ANN query service can be load-tested with random embeddings as queries, generated automatically by loadtest tool. +Example script to load test a ANN query service with random embeddings: + +```bash +$ aurora job create smf1//staging/ann-loadtest-service ann/src/main/aurora/loadtest/loadtest.aurora \ + --bind=profile.name=ann-loadtest-service \ + --bind=profile.role= \ + --bind=profile.duration_sec=10 \ + --bind=profile.number_of_neighbors=10 \ + --bind=profile.qps=200 \ + --bind=profile.algo=hnsw \ + --bind=profile.metric=Cosine \ + --bind=profile.index_id_type=int \ + --bind=profile.hnsw_ef=400,600,800 \ + --bind=profile.embedding_dimension=3 \ + --bind=profile.concurrency_level=8 \ + --bind=profile.loadtest_type=remote \ + --bind=profile.service_destination=/srv#/staging/local/apoorvs/ann-server-test \ + --bind=profile.with_random_queries=True \ + --bind=profile.random_queries_count=50000 \ + --bind=profile.random_embedding_min_value=-10.0 \ + --bind=profile.random_embedding_max_value=10.0 +``` + +It will run the loadtest with `50000` random embeddings, where each embedding value will be range bounded between `random_embedding_min_value` and `random_embedding_max_value`. +In the above the case it will be bounded between `-10.0` and `10.0`. +If `random_embedding_min_value` and `random_embedding_max_value` are not supplied default value of `-1.0` and `1.0` will be used. + +## Results + +Load test results will be printed to stdout of an aurora job. + +# Loadtest ANN query service with query set + +An ANN query service can be load-tested with sample queries drawn from the embeddings dataset. +For creating sample queries i.e `query_set` refer this [section](#query-set-generator). + +Test is run with `live` version of loadtest binary that is already available in packer. +Example script to load test a ANN query service: + +```bash +$ aurora job create smf1//staging/ann-loadtest-service ann/src/main/aurora/loadtest/loadtest.aurora \ + --bind=profile.name=ann-loadtest-service \ + --bind=profile.role= \ + --bind=profile.duration_sec=10 \ + --bind=profile.query_set_dir=hdfs:///user/cortex/ann_example/dataset/search/query_knn/query_set \ + --bind=profile.number_of_neighbors=10 \ + --bind=profile.qps=200 \ + --bind=profile.algo=hnsw \ + --bind=profile.query_id_type=string \ + --bind=profile.index_id_type=string \ + --bind=profile.metric=Cosine \ + --bind=profile.hnsw_ef=400,600,800 \ + --bind=profile.embedding_dimension=100 \ + --bind=profile.concurrency_level=8 \ + --bind=profile.loadtest_type=remote \ + --bind=profile.service_destination=/srv#/staging/local/apoorvs/ann-server-test +``` + +# In-Memory based loadtest for measuring recall + +Load test can be with the above created dataset in memory. +For running in in-memory mode, index is created in memory, and for that you need `query_set/index_set/truth_set`. +For creating this dataset refer this [section](#knn-truth-set-generator). + +Test is run with `live` version loadtest binary that is already available in packer. +Example script In-Memory index building and benchmarking: + +```bash +$ aurora job create smf1//staging/ann-loadtest ann/src/main/aurora/loadtest/loadtest.aurora \ + --bind=profile.name=ann-loadtest \ + --bind=profile.role= \ + --bind=profile.duration_sec=10 \ + --bind=profile.truth_set_dir=hdfs:///user/cortex/ann_example/dataset/search/query_knn/true_knn \ + --bind=profile.query_set_dir=hdfs:///user/cortex/ann_example/dataset/search/query_knn/query_set \ + --bind=profile.index_set_dir=hdfs:///user/cortex/ann_example/dataset/search/query_knn/index_set \ + --bind=profile.number_of_neighbors=10 \ + --bind=profile.qps=200 \ + --bind=profile.algo=hnsw \ + --bind=profile.query_id_type=string \ + --bind=profile.index_id_type=string \ + --bind=profile.metric=Cosine \ + --bind=profile.hnsw_ef_construction=15 \ + --bind=profile.hnsw_max_m=10 \ + --bind=profile.hnsw_ef=400,600,800 \ + --bind=profile.embedding_dimension=100 \ + --bind=profile.concurrency_level=8 \ + --bind=profile.loadtest_type=local +``` + +# Loadtest faiss + +```bash +$ aurora job create smf1//staging/ann-loadtest-service ann/src/main/aurora/loadtest/loadtest.aurora \ + --bind=profile.name=ann-loadtest-service \ + --bind=profile.role= \ + --bind=profile.duration_sec=10 \ + --bind=profile.number_of_neighbors=10 \ + --bind=profile.qps=200 \ + --bind=profile.algo=faiss \ # Changed to faiss + --bind=profile.faiss_nprobe=1,3,9,27,81,128,256,512 \ # Added + --bind=profile.faiss_quantizerKfactorRF=1,2 \ # Pass a list to do grid search + --bind=profile.faiss_quantizerNprobe=128 \ # Added + --bind=profile.metric=Cosine \ + --bind=profile.index_id_type=int \ + --bind=profile.embedding_dimension=3 \ + --bind=profile.concurrency_level=8 \ + --bind=profile.loadtest_type=remote \ + --bind=profile.service_destination=/srv#/staging/local/apoorvs/ann-server-test \ + --bind=profile.with_random_queries=True \ + --bind=profile.random_queries_count=50000 \ + --bind=profile.random_embedding_min_value=-10.0 \ + --bind=profile.random_embedding_max_value=10.0 +``` + +Full list of faiss specific parameters. [Exact definition of all available parameters](https://github.com/facebookresearch/faiss/blob/36f2998a6469280cef3b0afcde2036935a29aa1f/faiss/AutoTune.cpp#L444). Please reach out if you need to use parameters which aren't shown below + +``` +faiss_nprobe = Default(String, '1') +faiss_quantizerEf = Default(String, '0') +faiss_quantizerKfactorRF = Default(String, '0') +faiss_quantizerNprobe = Default(String, '0') +faiss_ht = Default(String, '0') +``` + +# Query Set Generator + +Sample queries can be generated from the embeddings dataset and can be used directly with load test in tab format. +To generate sample queries `EmbeddingSamplingJob` can be used as follows. + +```bash +$ ./bazel bundle cortex-core/entity-embeddings/src/scala/main/com/twitter/scalding/util/EmbeddingFormat:embeddingformat-deploy + +$ export INPUT_PATH=/user/cortex/embeddings/user/tfwproducersg/embedding_datarecords_on_data/2018/05/01 +$ export ENTITY_KIND=user +$ export EMBEDDING_INPUT_FORMAT=usertensor +$ export OUTPUT_PATH=/user/$USER/sample_embeddings +$ export SAMPLE_PERCENT=0.1 + +$ oscar hdfs \ + --screen --tee log.txt \ + --hadoop-client-memory 6000 \ + --hadoop-properties "yarn.app.mapreduce.am.resource.mb=6000;yarn.app.mapreduce.am.command-opts='-Xmx7500m';mapreduce.map.memory.mb=7500;mapreduce.reduce.java.opts='-Xmx6000m';mapreduce.reduce.memory.mb=7500;mapred.task.timeout=36000000;" \ + --min-split-size 284217728 \ + --bundle embeddingformat-deploy \ + --host hadoopnest1.smf1.twitter.com \ + --tool com.twitter.scalding.entityembeddings.util.EmbeddingFormat.EmbeddingSamplingJob -- \ + --entity_kind $ENTITY_KIND \ + --input.embedding_path $INPUT_PATH \ + --input.embedding_format $EMBEDDING_INPUT_FORMAT \ + --output.embedding_path $OUTPUT_PATH \ + --output.embedding_format tab \ + --sample_percent $SAMPLE_PERCENT +``` + +It will sample 0.1% of embeddings and store them in `tab` format to hdfs that can be direcly used as `query_set` for loadtest. + +# Knn Truth Set Generator + +To use load test framework to benchmark recall, you need to split your data set into index_set, query_set and knn_truth + +- index_set: data that will be indexed for ann +- query_set: data that will be used for queries +- truth_set: the real nearest neighbor used as truth to compute recall + +And also you need to figure out the dimension for your embedding vectors. + +KnnTruthSetGenerator can help to prepare data sets: + +```bash +$ ./bazel bundle ann/src/main/scala/com/twitter/ann/scalding/offline:ann-offline-deploy + +$ export QUERY_EMBEDDINGS_PATH=/user/cortex-mlx/official_examples/ann/non_pii_random_user_embeddings_tab_format +$ export INDEX_EMBEDDINGS_PATH=/user/cortex-mlx/official_examples/ann/non_pii_random_user_embeddings_tab_format +$ export TRUTH_SET_PATH=/user/$USER/truth_set +$ export INDEX_SET_PATH=/user/$USER/index_set +$ export QUERY_SET_PATH=/user/$USER/query_set +$ export METRIC=InnerProduct +$ export QUERY_ENTITY_KIND=user +$ export INDEX_ENTITY_KIND=user +$ export NEIGHBOURS=10 + +$ oscar hdfs \ + --screen --tee log.txt \ + --hadoop-client-memory 6000 \ + --hadoop-properties "yarn.app.mapreduce.am.resource.mb=6000;yarn.app.mapreduce.am.command-opts='-Xmx7500m';mapreduce.map.memory.mb=7500;mapreduce.reduce.java.opts='-Xmx6000m';mapreduce.reduce.memory.mb=7500;mapred.task.timeout=36000000;" \ + --bundle ann-offline-deploy \ + --min-split-size 284217728 \ + --host hadoopnest1.smf1.twitter.com \ + --tool com.twitter.ann.scalding.offline.KnnTruthSetGenerator -- \ + --neighbors $NEIGHBOURS \ + --metric $METRIC \ + --query_entity_kind $QUERY_ENTITY_KIND \ + --query.embedding_path $QUERY_EMBEDDINGS_PATH \ + --query.embedding_format tab \ + --query_sample_percent 50.0 \ + --index_entity_kind $INDEX_ENTITY_KIND \ + --index.embedding_path $INDEX_EMBEDDINGS_PATH \ + --index.embedding_format tab \ + --index_sample_percent 90.0 \ + --query_set_output.embedding_path $QUERY_SET_PATH \ + --query_set_output.embedding_format tab \ + --index_set_output.embedding_path $INDEX_SET_PATH \ + --index_set_output.embedding_format tab \ + --truth_set_output_path $TRUTH_SET_PATH \ + --reducers 100 +``` + +It will sample 90% of index set embeddings and 50% of query embeddings from total and then it will generate 3 datasets from the same that are index set, query set and true nearest neighbours from query to index in the tab format. +`Note`: The reason for using high sample percent is due to the fact the sample embeddings dataset is small. For real use cases query set should be really small. +Set `--reducers` according to the embeddings dataset size. + +# FAQ + +There are multiple type of `query_id_type` and `index_id_type` that can be used. Some native types like string/int/long or related to entity embeddings +like tweet/word/user/url... for more info: [Link](https://cgit.twitter.biz/source/tree/src/scala/com/twitter/cortex/ml/embeddings/common/EntityKind.scala#n8) diff --git a/ann/src/main/scala/com/twitter/ann/service/query_server/common/BUILD b/ann/src/main/scala/com/twitter/ann/service/query_server/common/BUILD new file mode 100644 index 0000000000..026be56990 --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/service/query_server/common/BUILD @@ -0,0 +1,63 @@ +scala_library( + name = "common", + sources = [ + "BaseQueryIndexServer.scala", + "Exceptions.scala", + "QueryIndexThriftController.scala", + "QueryableProvider.scala", + "RefreshableQueryable.scala", + "UnsafeQueryIndexServer.scala", + ], + compiler_option_sets = ["fatal_warnings"], + tags = ["bazel-compatible"], + dependencies = [ + ":index_path_provider", + "3rdparty/jvm/com/google/guava", + "3rdparty/jvm/com/google/inject:guice", + "3rdparty/jvm/com/google/inject/extensions:guice-assistedinject", + "3rdparty/jvm/javax/inject:javax.inject", + "3rdparty/jvm/net/codingwell:scala-guice", + "ann/src/main/scala/com/twitter/ann/common", + "ann/src/main/thrift/com/twitter/ann/common:ann-common-scala", + "finagle/finagle-core/src/main", + "finagle/finagle-zipkin-scribe", + "finatra-internal/decider", + "finatra-internal/mtls-thriftmux/src/main/scala", + "finatra/inject/inject-core/src/main/scala", + "mediaservices/commons", + "scrooge/scrooge-core/src/main/scala", + "src/scala/com/twitter/cortex/ml/embeddings/common:Helpers", + "util/util-app/src/main/scala", + "util/util-logging/src/main/scala", + ], +) + +scala_library( + name = "index_path_provider", + sources = [ + "IndexPathProvider.scala", + "QueryServerUtil.scala", + ], + compiler_option_sets = ["fatal_warnings"], + tags = ["bazel-compatible"], + dependencies = [ + "ann/src/main/scala/com/twitter/ann/hnsw", + "src/java/com/twitter/search/common/file", + "util/util-logging/src/main/scala", + "util/util-stats/src/main/scala/com/twitter/finagle/stats", + ], +) + +scala_library( + name = "faiss_index_path_provider", + sources = ["FaissIndexPathProvider.scala"], + compiler_option_sets = ["fatal_warnings"], + tags = ["bazel-compatible"], + dependencies = [ + ":index_path_provider", + "ann/src/main/scala/com/twitter/ann/faiss", + "src/java/com/twitter/search/common/file", + "util/util-logging/src/main/scala", + "util/util-stats/src/main/scala/com/twitter/finagle/stats", + ], +) diff --git a/ann/src/main/scala/com/twitter/ann/service/query_server/common/BaseQueryIndexServer.scala b/ann/src/main/scala/com/twitter/ann/service/query_server/common/BaseQueryIndexServer.scala new file mode 100644 index 0000000000..bac537d273 --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/service/query_server/common/BaseQueryIndexServer.scala @@ -0,0 +1,53 @@ +package com.twitter.ann.service.query_server.common + +import com.google.inject.Module +import com.twitter.ann.common.thriftscala.AnnQueryService +import com.twitter.app.Flag +import com.twitter.finatra.decider.modules.DeciderModule +import com.twitter.finatra.thrift.ThriftServer +import com.twitter.finatra.mtls.thriftmux.Mtls +import com.twitter.finatra.mtls.thriftmux.modules.MtlsThriftWebFormsModule +import com.twitter.finatra.thrift.filters.{ + AccessLoggingFilter, + LoggingMDCFilter, + StatsFilter, + ThriftMDCFilter, + TraceIdMDCFilter +} +import com.twitter.finatra.thrift.routing.ThriftRouter + +/** + * This class provides most of the configuration needed for logging, stats, deciders etc. + */ +abstract class BaseQueryIndexServer extends ThriftServer with Mtls { + + protected val environment: Flag[String] = flag[String]("environment", "service environment") + + /** + * Override with method to provide more module to guice. + */ + protected def additionalModules: Seq[Module] + + /** + * Override this method to add the controller to the thrift router. BaseQueryIndexServer takes + * care of most of the other configuration for you. + * @param router + */ + protected def addController(router: ThriftRouter): Unit + + override protected final lazy val modules: Seq[Module] = Seq( + DeciderModule, + new MtlsThriftWebFormsModule[AnnQueryService.MethodPerEndpoint](this) + ) ++ additionalModules + + override protected final def configureThrift(router: ThriftRouter): Unit = { + router + .filter[LoggingMDCFilter] + .filter[TraceIdMDCFilter] + .filter[ThriftMDCFilter] + .filter[AccessLoggingFilter] + .filter[StatsFilter] + + addController(router) + } +} diff --git a/ann/src/main/scala/com/twitter/ann/service/query_server/common/Exceptions.scala b/ann/src/main/scala/com/twitter/ann/service/query_server/common/Exceptions.scala new file mode 100644 index 0000000000..60c585e33f --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/service/query_server/common/Exceptions.scala @@ -0,0 +1,15 @@ +package com.twitter.ann.service.query_server.common + +import com.twitter.ann.common.thriftscala.BadRequest +import com.twitter.mediaservices.commons._ + +object RuntimeExceptionTransform extends ExceptionTransformer { + override def transform = { + case e: BadRequest => + MisuseExceptionInfo(e) + } + + override def getStatName: PartialFunction[Exception, String] = { + case e: BadRequest => exceptionName(e, e.code.name) + } +} diff --git a/ann/src/main/scala/com/twitter/ann/service/query_server/common/FaissIndexPathProvider.scala b/ann/src/main/scala/com/twitter/ann/service/query_server/common/FaissIndexPathProvider.scala new file mode 100644 index 0000000000..0286023c72 --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/service/query_server/common/FaissIndexPathProvider.scala @@ -0,0 +1,20 @@ +package com.twitter.ann.service.query_server.common + +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.logging.Logger +import com.twitter.search.common.file.AbstractFile + +case class FaissIndexPathProvider( + override val minIndexSizeBytes: Long, + override val maxIndexSizeBytes: Long, + override val statsReceiver: StatsReceiver) + extends BaseIndexPathProvider { + + override val log = Logger.get("FAISSIndexPathProvider") + + override def isValidIndex(dir: AbstractFile): Boolean = { + dir.isDirectory && + dir.hasSuccessFile && + dir.getChild("faiss.index").exists() + } +} diff --git a/ann/src/main/scala/com/twitter/ann/service/query_server/common/IndexPathProvider.scala b/ann/src/main/scala/com/twitter/ann/service/query_server/common/IndexPathProvider.scala new file mode 100644 index 0000000000..b6193b90a6 --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/service/query_server/common/IndexPathProvider.scala @@ -0,0 +1,179 @@ +package com.twitter.ann.service.query_server.common + +import com.twitter.ann.common.IndexOutputFile +import com.twitter.ann.hnsw.HnswCommon._ +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.logging.Logger +import com.twitter.search.common.file.AbstractFile +import com.twitter.search.common.file.AbstractFile.Filter +import com.twitter.search.common.file.PathUtils +import com.twitter.util.Try +import java.io.IOException +import java.util.concurrent.atomic.AtomicReference +import scala.collection.JavaConverters._ +import scala.math.Ordering.comparatorToOrdering + +abstract class IndexPathProvider { + def provideIndexPath(rootPath: AbstractFile, group: Boolean = false): Try[AbstractFile] + def provideIndexPathWithGroups(rootPath: AbstractFile): Try[Seq[AbstractFile]] +} + +abstract class BaseIndexPathProvider extends IndexPathProvider { + protected val minIndexSizeBytes: Long + protected val maxIndexSizeBytes: Long + protected val statsReceiver: StatsReceiver + protected val log: Logger + private val invalidPathCounter = statsReceiver.counter("invalid_index") + private val failToLocateDirectoryCounter = statsReceiver.counter("find_latest_path_fail") + private val successProvidePathCounter = statsReceiver.counter("provide_path_success") + + private val latestGroupCount = new AtomicReference(0f) + private val latestIndexTimestamp = new AtomicReference(0f) + private val latestValidIndexTimestamp = new AtomicReference(0f) + + private val INDEX_METADATA_FILE = "ANN_INDEX_METADATA" + + private val latestIndexGauge = statsReceiver.addGauge("latest_index_timestamp")( + latestIndexTimestamp.get() + ) + private val latestValidIndexGauge = statsReceiver.addGauge("latest_valid_index_timestamp")( + latestValidIndexTimestamp.get() + ) + private val latestGroupCountGauge = statsReceiver.addGauge("latest_group_count")( + latestGroupCount.get() + ) + + private val latestTimeStampDirectoryFilter = new AbstractFile.Filter { + + /** Determines which files should be accepted when listing a directory. */ + override def accept(file: AbstractFile): Boolean = { + val name = file.getName + PathUtils.TIMESTAMP_PATTERN.matcher(name).matches() + } + } + + private def findLatestTimeStampValidSuccessDirectory( + path: AbstractFile, + group: Boolean + ): AbstractFile = { + log.info(s"Calling findLatestTimeStampValidSuccessDirectory with ${path.getPath}") + // Get all the timestamp directories + val dateDirs = path.listFiles(latestTimeStampDirectoryFilter).asScala.toSeq + + if (dateDirs.nonEmpty) { + // Validate the indexes + val latestValidPath = { + if (group) { + // For grouped, check all the individual group indexes and stop as soon as a valid index + // is found. + dateDirs + .sorted(comparatorToOrdering(PathUtils.NEWEST_FIRST_COMPARATOR)).find(file => { + val indexMetadataFile = file.getChild(INDEX_METADATA_FILE) + val indexes = file.listFiles().asScala.filter(_.isDirectory) + val isValid = if (indexMetadataFile.exists()) { + // Metadata file exists. Check the number of groups and verify the index is + // complete + val indexMetadata = new IndexOutputFile(indexMetadataFile).loadIndexMetadata() + if (indexMetadata.numGroups.get != indexes.size) { + log.info( + s"Grouped index ${file.getPath} should have ${indexMetadata.numGroups.get} groups but had ${indexes.size}") + } + indexMetadata.numGroups.get == indexes.size + } else { + // True if the file doesn't exist. This is to make this change backwards + // compatible for clients using the old version of the dataflow job + true + } + + isValid && indexes.forall(index => { + index.hasSuccessFile && isValidIndex(index) && QueryServerUtil + .isValidIndexDirSize(index, minIndexSizeBytes, maxIndexSizeBytes) + }) + }) + } else { + // For non-grouped, find the first valid index. + dateDirs + .sorted(comparatorToOrdering(PathUtils.NEWEST_FIRST_COMPARATOR)).find(file => { + file.hasSuccessFile && QueryServerUtil + .isValidIndexDirSize(file, minIndexSizeBytes, maxIndexSizeBytes) + }) + } + } + + if (latestValidPath.nonEmpty) { + // Log the results + successProvidePathCounter.incr() + if (group) { + latestGroupCount.set(latestValidPath.get.listFiles().asScala.count(_.isDirectory)) + log.info( + s"findLatestTimeStampValidSuccessDirectory latestValidPath ${latestValidPath.get.getPath} and number of groups $latestGroupCount") + } else { + val latestValidPathSize = + latestValidPath.get.listFiles(true).asScala.map(_.getSizeInBytes).sum + log.info( + s"findLatestTimeStampValidSuccessDirectory latestValidPath ${latestValidPath.get.getPath} and size $latestValidPathSize") + } + return latestValidPath.get + } + } + + // Fail if no index or no valid index. + failToLocateDirectoryCounter.incr() + throw new IOException(s"Cannot find any valid directory with SUCCESS file at ${path.getName}") + } + + def isValidIndex(index: AbstractFile): Boolean + + override def provideIndexPath( + rootPath: AbstractFile, + group: Boolean = false + ): Try[AbstractFile] = { + Try { + val latestValidPath = findLatestTimeStampValidSuccessDirectory(rootPath, group) + if (!group) { + val latestPath = PathUtils.findLatestTimeStampSuccessDirectory(rootPath) + // since latestValidPath does not throw exception, latestPath must exist + assert(latestPath.isPresent) + val latestPathSize = latestPath.get.listFiles(true).asScala.map(_.getSizeInBytes).sum + log.info(s"provideIndexPath latestPath ${latestPath + .get() + .getPath} and size $latestPathSize") + latestIndexTimestamp.set(latestPath.get().getName.toFloat) + // latest directory is not valid, update counter for alerts + if (latestPath.get() != latestValidPath) { + invalidPathCounter.incr() + } + } else { + latestIndexTimestamp.set(latestValidPath.getName.toFloat) + } + latestValidIndexTimestamp.set(latestValidPath.getName.toFloat) + latestValidPath + } + } + + override def provideIndexPathWithGroups( + rootPath: AbstractFile + ): Try[Seq[AbstractFile]] = { + val latestValidPath = provideIndexPath(rootPath, true) + latestValidPath.map { path => + path + .listFiles(new Filter { + override def accept(file: AbstractFile): Boolean = + file.isDirectory && file.hasSuccessFile + }).asScala.toSeq + } + } +} + +case class ValidatedIndexPathProvider( + override val minIndexSizeBytes: Long, + override val maxIndexSizeBytes: Long, + override val statsReceiver: StatsReceiver) + extends BaseIndexPathProvider { + + override val log = Logger.get("ValidatedIndexPathProvider") + + override def isValidIndex(dir: AbstractFile): Boolean = { + isValidHnswIndex(dir) + } +} diff --git a/ann/src/main/scala/com/twitter/ann/service/query_server/common/QueryIndexThriftController.scala b/ann/src/main/scala/com/twitter/ann/service/query_server/common/QueryIndexThriftController.scala new file mode 100644 index 0000000000..4a10500198 --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/service/query_server/common/QueryIndexThriftController.scala @@ -0,0 +1,92 @@ +package com.twitter.ann.service.query_server.common + +import com.twitter.ann.common._ +import com.twitter.ann.common.EmbeddingType._ +import com.twitter.ann.common.thriftscala.AnnQueryService.Query +import com.twitter.ann.common.thriftscala.AnnQueryService +import com.twitter.ann.common.thriftscala.NearestNeighbor +import com.twitter.ann.common.thriftscala.NearestNeighborResult +import com.twitter.ann.common.thriftscala.{Distance => ServiceDistance} +import com.twitter.ann.common.thriftscala.{RuntimeParams => ServiceRuntimeParams} +import com.twitter.bijection.Injection +import com.twitter.finagle.Service +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.finatra.thrift.Controller +import com.twitter.mediaservices.commons.{ThriftServer => TServer} +import java.nio.ByteBuffer +import javax.inject.Inject + +class QueryIndexThriftController[T, P <: RuntimeParams, D <: Distance[D]] @Inject() ( + statsReceiver: StatsReceiver, + queryable: Queryable[T, P, D], + runtimeParamInjection: Injection[P, ServiceRuntimeParams], + distanceInjection: Injection[D, ServiceDistance], + idInjection: Injection[T, Array[Byte]]) + extends Controller(AnnQueryService) { + + private[this] val thriftServer = new TServer(statsReceiver, Some(RuntimeExceptionTransform)) + + val trackingStatName = "ann_query" + + private[this] val stats = statsReceiver.scope(trackingStatName) + private[this] val numOfNeighboursRequested = stats.stat("num_of_neighbours_requested") + private[this] val numOfNeighboursResponse = stats.stat("num_of_neighbours_response") + private[this] val queryKeyNotFound = stats.stat("query_key_not_found") + + /** + * Implements AnnQueryService.query, returns nearest neighbours for a given query + */ + val query: Service[Query.Args, Query.SuccessType] = { args: Query.Args => + thriftServer.track(trackingStatName) { + val query = args.query + val key = query.key + val embedding = embeddingSerDe.fromThrift(query.embedding) + val numOfNeighbours = query.numberOfNeighbors + val withDistance = query.withDistance + val runtimeParams = runtimeParamInjection.invert(query.runtimeParams).get + numOfNeighboursRequested.add(numOfNeighbours) + + val result = if (withDistance) { + val nearestNeighbors = if (queryable.isInstanceOf[QueryableGrouped[T, P, D]]) { + queryable + .asInstanceOf[QueryableGrouped[T, P, D]] + .queryWithDistance(embedding, numOfNeighbours, runtimeParams, key) + } else { + queryable + .queryWithDistance(embedding, numOfNeighbours, runtimeParams) + } + + nearestNeighbors.map { list => + list.map { nn => + NearestNeighbor( + ByteBuffer.wrap(idInjection.apply(nn.neighbor)), + Some(distanceInjection.apply(nn.distance)) + ) + } + } + } else { + + val nearestNeighbors = if (queryable.isInstanceOf[QueryableGrouped[T, P, D]]) { + queryable + .asInstanceOf[QueryableGrouped[T, P, D]] + .query(embedding, numOfNeighbours, runtimeParams, key) + } else { + queryable + .query(embedding, numOfNeighbours, runtimeParams) + } + + nearestNeighbors + .map { list => + list.map { nn => + NearestNeighbor(ByteBuffer.wrap(idInjection.apply(nn))) + } + } + } + + result.map(NearestNeighborResult(_)).onSuccess { r => + numOfNeighboursResponse.add(r.nearestNeighbors.size) + } + } + } + handle(Query) { query } +} diff --git a/ann/src/main/scala/com/twitter/ann/service/query_server/common/QueryServerUtil.scala b/ann/src/main/scala/com/twitter/ann/service/query_server/common/QueryServerUtil.scala new file mode 100644 index 0000000000..5aef64a547 --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/service/query_server/common/QueryServerUtil.scala @@ -0,0 +1,34 @@ +package com.twitter.ann.service.query_server.common + +import com.twitter.logging.Logger +import com.twitter.search.common.file.AbstractFile +import scala.collection.JavaConverters._ + +object QueryServerUtil { + + private val log = Logger.get("QueryServerUtil") + + /** + * Validate if the abstract file (directory) size is within the defined limits. + * @param dir Hdfs/Local directory + * @param minIndexSizeBytes minimum size of file in bytes (Exclusive) + * @param maxIndexSizeBytes minimum size of file in bytes (Exclusive) + * @return true if file size within minIndexSizeBytes and maxIndexSizeBytes else false + */ + def isValidIndexDirSize( + dir: AbstractFile, + minIndexSizeBytes: Long, + maxIndexSizeBytes: Long + ): Boolean = { + val recursive = true + val dirSize = dir.listFiles(recursive).asScala.map(_.getSizeInBytes).sum + + log.debug(s"Ann index directory ${dir.getPath} size in bytes $dirSize") + + val isValid = (dirSize > minIndexSizeBytes) && (dirSize < maxIndexSizeBytes) + if (!isValid) { + log.info(s"Ann index directory is invalid ${dir.getPath} size in bytes $dirSize") + } + isValid + } +} diff --git a/ann/src/main/scala/com/twitter/ann/service/query_server/common/QueryableProvider.scala b/ann/src/main/scala/com/twitter/ann/service/query_server/common/QueryableProvider.scala new file mode 100644 index 0000000000..0c579c591d --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/service/query_server/common/QueryableProvider.scala @@ -0,0 +1,8 @@ +package com.twitter.ann.service.query_server.common + +import com.twitter.ann.common.{Distance, Queryable, RuntimeParams} +import com.twitter.search.common.file.AbstractFile + +trait QueryableProvider[T, P <: RuntimeParams, D <: Distance[D]] { + def provideQueryable(indexDir: AbstractFile): Queryable[T, P, D] +} diff --git a/ann/src/main/scala/com/twitter/ann/service/query_server/common/RefreshableQueryable.scala b/ann/src/main/scala/com/twitter/ann/service/query_server/common/RefreshableQueryable.scala new file mode 100644 index 0000000000..1ce301c5e1 --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/service/query_server/common/RefreshableQueryable.scala @@ -0,0 +1,212 @@ +package com.twitter.ann.service.query_server.common + +import com.google.common.annotations.VisibleForTesting +import com.google.common.util.concurrent.ThreadFactoryBuilder +import com.twitter.ann.common.EmbeddingType.EmbeddingVector +import com.twitter.ann.common.Distance +import com.twitter.ann.common.NeighborWithDistance +import com.twitter.ann.common.Queryable +import com.twitter.ann.common.QueryableGrouped +import com.twitter.ann.common.RuntimeParams +import com.twitter.conversions.DurationOps._ +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.logging.Logger +import com.twitter.search.common.file.AbstractFile +import com.twitter.util.Duration +import com.twitter.util.Future +import java.util.concurrent.atomic.AtomicReference +import java.util.concurrent.Executors +import java.util.concurrent.TimeUnit +import scala.util.Random +import scala.util.control.NonFatal + +class RefreshableQueryable[T, P <: RuntimeParams, D <: Distance[D]]( + grouped: Boolean, + rootDir: AbstractFile, + queryableProvider: QueryableProvider[T, P, D], + indexPathProvider: IndexPathProvider, + statsReceiver: StatsReceiver, + updateInterval: Duration = 10.minutes) + extends QueryableGrouped[T, P, D] { + + private val log = Logger.get("RefreshableQueryable") + + private val loadCounter = statsReceiver.counter("load") + private val loadFailCounter = statsReceiver.counter("load_error") + private val newIndexCounter = statsReceiver.counter("new_index") + protected val random = new Random(System.currentTimeMillis()) + + private val threadFactory = new ThreadFactoryBuilder() + .setNameFormat("refreshable-queryable-update-%d") + .build() + // single thread to check and load index + private val executor = Executors.newScheduledThreadPool(1, threadFactory) + + private[common] val indexPathRef: AtomicReference[AbstractFile] = + new AtomicReference(indexPathProvider.provideIndexPath(rootDir, grouped).get()) + private[common] val queryableRef: AtomicReference[Map[Option[String], Queryable[T, P, D]]] = { + if (grouped) { + val mapping = getGroupMapping + + new AtomicReference(mapping) + } else { + new AtomicReference(Map(None -> buildIndex(indexPathRef.get()))) + } + } + + private val servingIndexGauge = statsReceiver.addGauge("serving_index_timestamp") { + indexPathRef.get().getName.toFloat + } + + log.info("System.gc() before start") + System.gc() + + private val reloadTask = new Runnable { + override def run(): Unit = { + innerLoad() + } + } + + def start(): Unit = { + executor.scheduleWithFixedDelay( + reloadTask, + // init reloading with random delay + computeRandomInitDelay().inSeconds, + updateInterval.inSeconds, + TimeUnit.SECONDS + ) + } + + private def buildIndex(indexPath: AbstractFile): Queryable[T, P, D] = { + log.info(s"build index from ${indexPath.getPath}") + queryableProvider.provideQueryable(indexPath) + } + + @VisibleForTesting + private[common] def innerLoad(): Unit = { + log.info("Check and load for new index") + loadCounter.incr() + try { + // Find the latest directory + val latestPath = indexPathProvider.provideIndexPath(rootDir, grouped).get() + if (indexPathRef.get() != latestPath) { + log.info(s"loading index from: ${latestPath.getName}") + newIndexCounter.incr() + if (grouped) { + val mapping = getGroupMapping + queryableRef.set(mapping) + } else { + val queryable = buildIndex(latestPath) + queryableRef.set(Map(None -> queryable)) + } + indexPathRef.set(latestPath) + } else { + log.info(s"Current index already up to date: ${indexPathRef.get.getName}") + } + } catch { + case NonFatal(err) => + loadFailCounter.incr() + log.error(s"Failed to load index: $err") + } + log.info(s"Current index loaded from ${indexPathRef.get().getPath}") + } + + @VisibleForTesting + private[common] def computeRandomInitDelay(): Duration = { + val bound = 5.minutes + val nextUpdateSec = updateInterval + Duration.fromSeconds( + random.nextInt(bound.inSeconds) + ) + nextUpdateSec + } + + /** + * ANN query for ids with key as group id + * @param embedding: Embedding/Vector to be queried with. + * @param numOfNeighbors: Number of neighbours to be queried for. + * @param runtimeParams: Runtime params associated with index to control accuracy/latency etc. + * @param key: Optional key to lookup specific ANN index and perform query there + * @return List of approximate nearest neighbour ids. + */ + override def query( + embedding: EmbeddingVector, + numOfNeighbors: Int, + runtimeParams: P, + key: Option[String] + ): Future[List[T]] = { + val mapping = queryableRef.get() + + if (!mapping.contains(key)) { + Future.value(List()) + } else { + mapping.get(key).get.query(embedding, numOfNeighbors, runtimeParams) + } + } + + /** + * ANN query for ids with key as group id with distance + * @param embedding: Embedding/Vector to be queried with. + * @param numOfNeighbors: Number of neighbours to be queried for. + * @param runtimeParams: Runtime params associated with index to control accuracy/latency etc. + * @param key: Optional key to lookup specific ANN index and perform query there + * @return List of approximate nearest neighbour ids with distance from the query embedding. + */ + override def queryWithDistance( + embedding: EmbeddingVector, + numOfNeighbors: Int, + runtimeParams: P, + key: Option[String] + ): Future[List[NeighborWithDistance[T, D]]] = { + val mapping = queryableRef.get() + + if (!mapping.contains(key)) { + Future.value(List()) + } else { + mapping.get(key).get.queryWithDistance(embedding, numOfNeighbors, runtimeParams) + } + } + + private def getGroupMapping(): Map[Option[String], Queryable[T, P, D]] = { + val groupDirs = indexPathProvider.provideIndexPathWithGroups(rootDir).get() + val mapping = groupDirs.map { groupDir => + val queryable = buildIndex(groupDir) + Option(groupDir.getName) -> queryable + }.toMap + + mapping + } + + /** + * ANN query for ids. + * + * @param embedding : Embedding/Vector to be queried with. + * @param numOfNeighbors : Number of neighbours to be queried for. + * @param runtimeParams : Runtime params associated with index to control accuracy/latency etc. + * + * @return List of approximate nearest neighbour ids. + */ + override def query( + embedding: EmbeddingVector, + numOfNeighbors: Int, + runtimeParams: P + ): Future[List[T]] = { + query(embedding, numOfNeighbors, runtimeParams, None) + } + + /** + * ANN query for ids with distance. + * + * @param embedding : Embedding/Vector to be queried with. + * @param numOfNeighbors : Number of neighbours to be queried for. + * @param runtimeParams : Runtime params associated with index to control accuracy/latency etc. + * + * @return List of approximate nearest neighbour ids with distance from the query embedding. + */ + override def queryWithDistance( + embedding: EmbeddingVector, + numOfNeighbors: Int, + runtimeParams: P + ): Future[List[NeighborWithDistance[T, D]]] = { + queryWithDistance(embedding, numOfNeighbors, runtimeParams, None) + } +} diff --git a/ann/src/main/scala/com/twitter/ann/service/query_server/common/UnsafeQueryIndexServer.scala b/ann/src/main/scala/com/twitter/ann/service/query_server/common/UnsafeQueryIndexServer.scala new file mode 100644 index 0000000000..a50dd53e6d --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/service/query_server/common/UnsafeQueryIndexServer.scala @@ -0,0 +1,109 @@ +package com.twitter.ann.service.query_server.common + +import com.google.common.util.concurrent.MoreExecutors +import com.google.inject.Module +import com.twitter.ann.common._ +import com.twitter.ann.common.thriftscala.{Distance => ServiceDistance} +import com.twitter.ann.common.thriftscala.{RuntimeParams => ServiceRuntimeParams} +import com.twitter.app.Flag +import com.twitter.bijection.Injection +import com.twitter.cortex.ml.embeddings.common.EntityKind +import com.twitter.finatra.thrift.routing.ThriftRouter +import java.util.concurrent.ExecutorService +import java.util.concurrent.Executors +import java.util.concurrent.ThreadPoolExecutor +import java.util.concurrent.TimeUnit + +/** + * This class is used when you do not know the generic parameters of the Server at compile time. + * If you want compile time checks that your parameters are correct use QueryIndexServer instead. + * In particular, when you want to have these id, distance and the runtime params as cli options you + * should extend this class. + */ +abstract class UnsafeQueryIndexServer[R <: RuntimeParams] extends BaseQueryIndexServer { + private[this] val metricName = flag[String]("metric", "metric") + private[this] val idType = flag[String]("id_type", "type of ids to use") + private[query_server] val queryThreads = + flag[Int]( + "threads", + System + .getProperty("mesos.resources.cpu", s"${Runtime.getRuntime.availableProcessors()}").toInt, + "Size of thread pool for concurrent querying" + ) + private[query_server] val dimension = flag[Int]("dimension", "dimension") + private[query_server] val indexDirectory = flag[String]("index_directory", "index directory") + private[query_server] val refreshable = + flag[Boolean]("refreshable", false, "if index is refreshable or not") + private[query_server] val refreshableInterval = + flag[Int]("refreshable_interval_minutes", 10, "refreshable index update interval") + private[query_server] val sharded = + flag[Boolean]("sharded", false, "if index is sharded") + private[query_server] val shardedHours = + flag[Int]("sharded_hours", "how many shards load at once") + private[query_server] val shardedWatchLookbackIndexes = + flag[Int]("sharded_watch_lookback_indexes", "how many indexes backwards to watch") + private[query_server] val shardedWatchIntervalMinutes = + flag[Int]("sharded_watch_interval_minutes", "interval at which hdfs is watched for changes") + private[query_server] val minIndexSizeBytes = + flag[Long]("min_index_size_byte", 0, "min index size in bytes") + private[query_server] val maxIndexSizeBytes = + flag[Long]("max_index_size_byte", Long.MaxValue, "max index size in bytes") + private[query_server] val grouped = + flag[Boolean]("grouped", false, "if indexes are grouped") + private[query_server] val qualityFactorEnabled = + flag[Boolean]( + "quality_factor_enabled", + false, + "Enable dynamically reducing search complexity when cgroups container is throttled. Useful to disable when load testing" + ) + private[query_server] val warmup_enabled: Flag[Boolean] = + flag("warmup", false, "Enable warmup before the query server starts up") + + // Time to wait for the executor to finish before terminating the JVM + private[this] val terminationTimeoutMs = 100 + protected lazy val executor: ExecutorService = MoreExecutors.getExitingExecutorService( + Executors.newFixedThreadPool(queryThreads()).asInstanceOf[ThreadPoolExecutor], + terminationTimeoutMs, + TimeUnit.MILLISECONDS + ) + + protected lazy val unsafeMetric: Metric[_] with Injection[_, ServiceDistance] = { + Metric.fromString(metricName()) + } + + override protected val additionalModules: Seq[Module] = Seq() + + override final def addController(router: ThriftRouter): Unit = { + router.add(queryIndexThriftController) + } + + protected def unsafeQueryableMap[T, D <: Distance[D]]: Queryable[T, R, D] + protected val runtimeInjection: Injection[R, ServiceRuntimeParams] + + private[this] def queryIndexThriftController[ + T, + D <: Distance[D] + ]: QueryIndexThriftController[T, R, D] = { + val controller = new QueryIndexThriftController[T, R, D]( + statsReceiver.scope("ann_server"), + unsafeQueryableMap.asInstanceOf[Queryable[T, R, D]], + runtimeInjection, + unsafeMetric.asInstanceOf[Injection[D, ServiceDistance]], + idInjection[T]() + ) + + logger.info("QueryIndexThriftController created....") + controller + } + + protected final def idInjection[T](): Injection[T, Array[Byte]] = { + val idInjection = idType() match { + case "string" => AnnInjections.StringInjection + case "long" => AnnInjections.LongInjection + case "int" => AnnInjections.IntInjection + case entityKind => EntityKind.getEntityKind(entityKind).byteInjection + } + + idInjection.asInstanceOf[Injection[T, Array[Byte]]] + } +} diff --git a/ann/src/main/scala/com/twitter/ann/service/query_server/common/throttling/AuroraCPUStatsReader.scala b/ann/src/main/scala/com/twitter/ann/service/query_server/common/throttling/AuroraCPUStatsReader.scala new file mode 100644 index 0000000000..5b461ca1d0 --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/service/query_server/common/throttling/AuroraCPUStatsReader.scala @@ -0,0 +1,28 @@ +package com.twitter.ann.service.query_server.common.throttling + +import com.twitter.server.filter.CgroupsCpu + +class AuroraCPUStatsReader() { + + val cgroupsCpu = new CgroupsCpu() + + def throttledTimeNanos(): Option[Long] = cgroupsCpu.cpuStat.map { cs => + cs.throttledTimeNanos + } + + /** + * Read assigned cpu number from Mesos files + * + * @return positive number is the number of CPUs (can be fractional). + * -1 means file read failed or it's not a valid Mesos environment. + */ + def cpuQuota: Double = cgroupsCpu.cfsPeriodMicros match { + case -1L => -1.0 + case 0L => 0.0 // avoid divide by 0 + case periodMicros => + cgroupsCpu.cfsQuotaMicros match { + case -1L => -1.0 + case quotaMicros => quotaMicros.toDouble / periodMicros.toDouble + } + } +} diff --git a/ann/src/main/scala/com/twitter/ann/service/query_server/common/throttling/BUILD b/ann/src/main/scala/com/twitter/ann/service/query_server/common/throttling/BUILD new file mode 100644 index 0000000000..5d0776d7a9 --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/service/query_server/common/throttling/BUILD @@ -0,0 +1,12 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + tags = ["bazel-compatible"], + dependencies = [ + "ann/src/main/scala/com/twitter/ann/common", + "ann/src/main/scala/com/twitter/ann/faiss", + "ann/src/main/scala/com/twitter/ann/hnsw", + "twitter-server-internal/src/main/scala", + "util/util-stats/src/main/scala/com/twitter/finagle/stats", + ], +) diff --git a/ann/src/main/scala/com/twitter/ann/service/query_server/common/throttling/ThrottlingBasedQualityTask.scala b/ann/src/main/scala/com/twitter/ann/service/query_server/common/throttling/ThrottlingBasedQualityTask.scala new file mode 100644 index 0000000000..35d893f8d2 --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/service/query_server/common/throttling/ThrottlingBasedQualityTask.scala @@ -0,0 +1,73 @@ +package com.twitter.ann.service.query_server.common.throttling + +import com.twitter.ann.common.RuntimeParams +import com.twitter.ann.common.Task +import com.twitter.ann.faiss.FaissParams +import com.twitter.ann.hnsw.HnswParams +import com.twitter.ann.service.query_server.common.throttling.ThrottlingBasedQualityTask.SAMPLING_INTERVAL +import com.twitter.conversions.DurationOps.richDurationFromInt +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.util.Duration +import com.twitter.util.Future +import com.twitter.util.logging.Logging + +object ThrottlingBasedQualityTask { + private[throttling] val SAMPLING_INTERVAL = 100.milliseconds +} + +class ThrottlingBasedQualityTask( + override val statsReceiver: StatsReceiver, + // Parameters are taken from OverloadAdmissionController + instrument: ThrottlingInstrument = new WindowedThrottlingInstrument(SAMPLING_INTERVAL, 5, + new AuroraCPUStatsReader())) + extends Task + with Logging { + import ThrottlingBasedQualityTask._ + + // [0, 1] where 1 is fully throttled + // Quickly throttle, but dampen recovery to make sure we won't enter throttle/GC death spiral + @volatile private var dampenedThrottlingPercentage: Double = 0 + + protected[throttling] def task(): Future[Unit] = { + if (!instrument.disabled) { + instrument.sample() + + val delta = instrument.percentageOfTimeSpentThrottling - dampenedThrottlingPercentage + if (delta > 0) { + // We want to start shedding load, do it quickly + dampenedThrottlingPercentage += delta + } else { + // Recover much slower + // At the rate of 100ms per sample, lookback is 2 minutes + val samplesToConverge = 1200.toDouble + dampenedThrottlingPercentage = + dampenedThrottlingPercentage + delta * (2 / (samplesToConverge + 1)) + } + + statsReceiver.stat("dampened_throttling").add(dampenedThrottlingPercentage.toFloat * 100) + } + + Future.Unit + } + + protected def taskInterval: Duration = SAMPLING_INTERVAL + + def discountParams[T <: RuntimeParams](params: T): T = { + // [0, 1] where 1 is best quality and lowest speed + // It's expected to run @1 majority of time + val qualityFactor = math.min(1, math.max(0, 1 - dampenedThrottlingPercentage)) + def applyQualityFactor(param: Int) = math.max(1, math.ceil(param * qualityFactor).toInt) + + params match { + case HnswParams(ef) => HnswParams(applyQualityFactor(ef)).asInstanceOf[T] + case FaissParams(nprobe, quantizerEf, quantizerKFactorRF, quantizerNprobe, ht) => + FaissParams( + nprobe.map(applyQualityFactor), + quantizerEf.map(applyQualityFactor), + quantizerKFactorRF.map(applyQualityFactor), + quantizerNprobe.map(applyQualityFactor), + ht.map(applyQualityFactor) + ).asInstanceOf[T] + } + } +} diff --git a/ann/src/main/scala/com/twitter/ann/service/query_server/common/throttling/WindowedStats.scala b/ann/src/main/scala/com/twitter/ann/service/query_server/common/throttling/WindowedStats.scala new file mode 100644 index 0000000000..71971574e1 --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/service/query_server/common/throttling/WindowedStats.scala @@ -0,0 +1,22 @@ +package com.twitter.ann.service.query_server.common.throttling + +/** + * A simple ring buffer that keeps track of long values over `window`. + */ +private[throttling] class WindowedStats(window: Int) { + private[this] val buffer = new Array[Long](window) + private[this] var index = 0 + private[this] var sumValue = 0L + private[this] var count = 0 + + def add(v: Long): Unit = { + count = math.min(count + 1, window) + val old = buffer(index) + buffer(index) = v + index = (index + 1) % window + sumValue += v - old + } + + def avg: Double = { sumValue.toDouble / count } + def sum: Long = { sumValue } +} diff --git a/ann/src/main/scala/com/twitter/ann/service/query_server/common/throttling/WindowedThrottlingInstrument.scala b/ann/src/main/scala/com/twitter/ann/service/query_server/common/throttling/WindowedThrottlingInstrument.scala new file mode 100644 index 0000000000..59ec936e9f --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/service/query_server/common/throttling/WindowedThrottlingInstrument.scala @@ -0,0 +1,50 @@ +package com.twitter.ann.service.query_server.common.throttling + +import com.twitter.util.Duration + +trait ThrottlingInstrument { + def sample(): Unit + def percentageOfTimeSpentThrottling(): Double + def disabled: Boolean +} + +class WindowedThrottlingInstrument( + stepFrequency: Duration, + windowLengthInFrequencySteps: Int, + reader: AuroraCPUStatsReader) + extends ThrottlingInstrument { + private[this] val throttlingChangeHistory: WindowedStats = new WindowedStats( + windowLengthInFrequencySteps) + + private[this] val cpuQuota: Double = reader.cpuQuota + + // The total number of allotted CPU time per step (in nanos). + private[this] val assignedCpu: Duration = stepFrequency * cpuQuota + private[this] val assignedCpuNs: Long = assignedCpu.inNanoseconds + + @volatile private[this] var previousThrottledTimeNs: Long = 0 + + /** + * If there isn't a limit on how much cpu the container can use, aurora + * throttling will never kick in. + */ + final def disabled: Boolean = cpuQuota <= 0 + + def sample(): Unit = sampleThrottling() match { + case Some(load) => + throttlingChangeHistory.add(load) + case None => () + } + + private[this] def sampleThrottling(): Option[Long] = reader.throttledTimeNanos().map { + throttledTimeNs => + val throttlingChange = throttledTimeNs - previousThrottledTimeNs + previousThrottledTimeNs = throttledTimeNs + throttlingChange + } + + // Time spent throttling over windowLength, normalized by number of CPUs + def percentageOfTimeSpentThrottling(): Double = { + math.min(1, throttlingChangeHistory.sum.toDouble / assignedCpuNs) + } +} diff --git a/ann/src/main/scala/com/twitter/ann/service/query_server/common/warmup/BUILD b/ann/src/main/scala/com/twitter/ann/service/query_server/common/warmup/BUILD new file mode 100644 index 0000000000..84706056b3 --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/service/query_server/common/warmup/BUILD @@ -0,0 +1,10 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + tags = ["bazel-compatible"], + dependencies = [ + "ann/src/main/scala/com/twitter/ann/common", + "util/util-core:scala", + "util/util-slf4j-api/src/main/scala/com/twitter/util/logging", + ], +) diff --git a/ann/src/main/scala/com/twitter/ann/service/query_server/common/warmup/Warmup.scala b/ann/src/main/scala/com/twitter/ann/service/query_server/common/warmup/Warmup.scala new file mode 100644 index 0000000000..349bc262e2 --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/service/query_server/common/warmup/Warmup.scala @@ -0,0 +1,50 @@ +package com.twitter.ann.service.query_server.common.warmup + +import com.twitter.ann.common.EmbeddingType.EmbeddingVector +import com.twitter.ml.api.embedding.Embedding +import com.twitter.util.Await +import com.twitter.util.Duration +import com.twitter.util.Future +import com.twitter.util.Return +import com.twitter.util.Throw +import com.twitter.util.Try +import com.twitter.util.logging.Logging +import scala.annotation.tailrec +import scala.util.Random + +trait Warmup extends Logging { + protected def minSuccessfulTries: Int + protected def maxTries: Int + protected def randomQueryDimension: Int + protected def timeout: Duration + + @tailrec + final protected def run( + iteration: Int = 0, + successes: Int = 0, + name: String, + f: => Future[_] + ): Unit = { + if (successes == minSuccessfulTries || iteration == maxTries) { + info(s"Warmup finished after ${iteration} iterations with ${successes} successes") + } else { + Try(Await.result(f.liftToTry, timeout)) match { + case Return(Return(_)) => + debug(s"[$name] Iteration $iteration Success") + run(iteration + 1, successes + 1, name, f) + case Return(Throw(e)) => + warn(s"[$name] Iteration $iteration has failed: ${e.getMessage}. ", e) + run(iteration + 1, successes, name, f) + case Throw(e) => + info(s"[$name] Iteration $iteration was too slow: ${e.getMessage}. ", e) + run(iteration + 1, successes, name, f) + } + } + } + + private val rng = new Random() + protected def randomQuery(): EmbeddingVector = + Embedding(Array.fill(randomQueryDimension)(-1 + 2 * rng.nextFloat())) + + def warmup(): Unit +} diff --git a/ann/src/main/scala/com/twitter/ann/service/query_server/faiss/BUILD b/ann/src/main/scala/com/twitter/ann/service/query_server/faiss/BUILD new file mode 100644 index 0000000000..ffdb2eb9c4 --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/service/query_server/faiss/BUILD @@ -0,0 +1,33 @@ +scala_library( + name = "server", + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + tags = ["bazel-compatible"], + dependencies = [ + "ann/src/main/java/com/twitter/ann/faiss", + "ann/src/main/scala/com/twitter/ann/common", + "ann/src/main/scala/com/twitter/ann/faiss", + "ann/src/main/scala/com/twitter/ann/service/query_server/common", + "ann/src/main/scala/com/twitter/ann/service/query_server/common:faiss_index_path_provider", + "ann/src/main/scala/com/twitter/ann/service/query_server/common/throttling", + "ann/src/main/scala/com/twitter/ann/service/query_server/common/warmup", + "src/java/com/twitter/search/common/file", + ], +) + +jvm_binary( + name = "faiss-query-server", + main = "com.twitter.ann.service.query_server.faiss.FaissQueryIndexServer", + compiler_option_sets = ["fatal_warnings"], + runtime_platform = "java11", + tags = ["bazel-compatible"], + dependencies = [ + ":server", + "3rdparty/jvm/ch/qos/logback:logback-classic", + "3rdparty/jvm/org/slf4j:jcl-over-slf4j", + "3rdparty/jvm/org/slf4j:jul-to-slf4j", + "3rdparty/jvm/org/slf4j:log4j-over-slf4j", + "ann/src/main/resources", + "loglens/loglens-logback/src/main/scala/com/twitter/loglens/logback", + ], +) diff --git a/ann/src/main/scala/com/twitter/ann/service/query_server/faiss/FaissQueryIndexServer.scala b/ann/src/main/scala/com/twitter/ann/service/query_server/faiss/FaissQueryIndexServer.scala new file mode 100644 index 0000000000..3ca46f8e4e --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/service/query_server/faiss/FaissQueryIndexServer.scala @@ -0,0 +1,149 @@ +package com.twitter.ann.service.query_server.faiss + +import com.twitter.ann.common.Distance +import com.twitter.ann.common.QueryableOperations.Map +import com.twitter.ann.common._ +import com.twitter.ann.common.thriftscala.{RuntimeParams => ServiceRuntimeParams} +import com.twitter.ann.faiss.FaissCommon +import com.twitter.ann.faiss.FaissIndex +import com.twitter.ann.faiss.FaissParams +import com.twitter.ann.faiss.HourlyShardedIndex +import com.twitter.ann.service.query_server.common.QueryableProvider +import com.twitter.ann.service.query_server.common.RefreshableQueryable +import com.twitter.ann.service.query_server.common.UnsafeQueryIndexServer +import com.twitter.ann.service.query_server.common.FaissIndexPathProvider +import com.twitter.ann.service.query_server.common.throttling.ThrottlingBasedQualityTask +import com.twitter.ann.service.query_server.common.warmup.Warmup +import com.twitter.bijection.Injection +import com.twitter.conversions.DurationOps.richDurationFromInt +import com.twitter.search.common.file.AbstractFile +import com.twitter.search.common.file.FileUtils +import com.twitter.util.Duration +import java.util.concurrent.TimeUnit + +object FaissQueryIndexServer extends FaissQueryableServer + +class FaissQueryableServer extends UnsafeQueryIndexServer[FaissParams] { + // given a directory, how to load it as a queryable index + def queryableProvider[T, D <: Distance[D]]: QueryableProvider[T, FaissParams, D] = + new QueryableProvider[T, FaissParams, D] { + override def provideQueryable( + directory: AbstractFile + ): Queryable[T, FaissParams, D] = { + FaissIndex.loadIndex[T, D]( + dimension(), + unsafeMetric.asInstanceOf[Metric[D]], + directory + ) + } + } + + private def buildSimpleQueryable[T, D <: Distance[D]]( + dir: AbstractFile + ): Queryable[T, FaissParams, D] = { + val queryable = if (refreshable()) { + logger.info(s"build refreshable queryable") + val updatableQueryable = new RefreshableQueryable( + false, + dir, + queryableProvider.asInstanceOf[QueryableProvider[T, FaissParams, D]], + FaissIndexPathProvider( + minIndexSizeBytes(), + maxIndexSizeBytes(), + statsReceiver.scope("validated_index_provider") + ), + statsReceiver.scope("refreshable_queryable"), + updateInterval = refreshableInterval().minutes + ) + // init first load of index and also schedule the following reloads + updatableQueryable.start() + updatableQueryable.asInstanceOf[QueryableGrouped[T, FaissParams, D]] + } else { + logger.info(s"build non-refreshable queryable") + + logger.info(s"Loading ${dir}") + queryableProvider.provideQueryable(dir).asInstanceOf[Queryable[T, FaissParams, D]] + } + + logger.info("Faiss queryable created....") + queryable + } + + private def buildShardedQueryable[T, D <: Distance[D]]( + dir: AbstractFile + ): Queryable[T, FaissParams, D] = { + logger.info(s"build sharded queryable") + + val queryable = HourlyShardedIndex.loadIndex[T, D]( + dimension(), + unsafeMetric.asInstanceOf[Metric[D]], + dir, + shardedHours(), + Duration(shardedWatchIntervalMinutes(), TimeUnit.MINUTES), + shardedWatchLookbackIndexes(), + statsReceiver.scope("hourly_sharded_index") + ) + + logger.info("Faiss sharded queryable created....") + + closeOnExit(queryable) + queryable.startImmediately() + + logger.info("Directory watching is scheduled") + + queryable + } + + // Readings come incorrect if reader is created too early in the lifecycle of a server + // hence lazy + private lazy val throttleSamplingTask = new ThrottlingBasedQualityTask( + statsReceiver.scope("throttling_task")) + + override def unsafeQueryableMap[T, D <: Distance[D]]: Queryable[T, FaissParams, D] = { + val dir = FileUtils.getFileHandle(indexDirectory()) + + val queryable = if (sharded()) { + require(shardedHours() > 0, "Number of hourly shards must be specified") + require(shardedWatchIntervalMinutes() > 0, "Shard watch interval must be specified") + require(shardedWatchLookbackIndexes() > 0, "Index lookback must be specified") + buildShardedQueryable[T, D](dir) + } else { + buildSimpleQueryable[T, D](dir) + } + + if (qualityFactorEnabled()) { + logger.info("Quality Factor throttling is enabled") + closeOnExit(throttleSamplingTask) + throttleSamplingTask.jitteredStart() + + queryable.mapRuntimeParameters(throttleSamplingTask.discountParams) + } else { + queryable + } + } + + override val runtimeInjection: Injection[FaissParams, ServiceRuntimeParams] = + FaissCommon.RuntimeParamsInjection + + protected override def warmup(): Unit = + if (warmup_enabled()) + new FaissWarmup(unsafeQueryableMap, dimension()).warmup() +} + +class FaissWarmup(faiss: Queryable[_, FaissParams, _], dimension: Int) extends Warmup { + protected def minSuccessfulTries: Int = 100 + protected def maxTries: Int = 1000 + protected def timeout: Duration = 50.milliseconds + protected def randomQueryDimension: Int = dimension + + def warmup(): Unit = { + run( + name = "queryWithDistance", + f = faiss + .queryWithDistance( + randomQuery(), + 100, + FaissParams(nprobe = Some(128), None, None, None, None)) + ) + } +} diff --git a/ann/src/main/scala/com/twitter/ann/service/query_server/hnsw/BUILD b/ann/src/main/scala/com/twitter/ann/service/query_server/hnsw/BUILD new file mode 100644 index 0000000000..a67bf445d8 --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/service/query_server/hnsw/BUILD @@ -0,0 +1,32 @@ +scala_library( + name = "server", + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + tags = ["bazel-compatible"], + dependencies = [ + "ann/src/main/scala/com/twitter/ann/common", + "ann/src/main/scala/com/twitter/ann/hnsw", + "ann/src/main/scala/com/twitter/ann/service/query_server/common", + "ann/src/main/scala/com/twitter/ann/service/query_server/common/warmup", + "src/java/com/twitter/search/common/file", + ], +) + +jvm_binary( + name = "hnsw-query-server", + main = "com.twitter.ann.service.query_server.hnsw.HnswQueryIndexServer", + compiler_option_sets = ["fatal_warnings"], + runtime_platform = "java11", + tags = [ + "bazel-compatible", + ], + dependencies = [ + ":server", + "3rdparty/jvm/ch/qos/logback:logback-classic", + "3rdparty/jvm/org/slf4j:jcl-over-slf4j", + "3rdparty/jvm/org/slf4j:jul-to-slf4j", + "3rdparty/jvm/org/slf4j:log4j-over-slf4j", + "ann/src/main/resources", + "loglens/loglens-logback/src/main/scala/com/twitter/loglens/logback", + ], +) diff --git a/ann/src/main/scala/com/twitter/ann/service/query_server/hnsw/HnswQueryIndexServer.scala b/ann/src/main/scala/com/twitter/ann/service/query_server/hnsw/HnswQueryIndexServer.scala new file mode 100644 index 0000000000..3c0efde40b --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/service/query_server/hnsw/HnswQueryIndexServer.scala @@ -0,0 +1,98 @@ +package com.twitter.ann.service.query_server.hnsw + +import com.twitter.ann.common.Distance +import com.twitter.ann.common._ +import com.twitter.ann.common.thriftscala.{RuntimeParams => ServiceRuntimeParams} +import com.twitter.ann.hnsw.HnswCommon +import com.twitter.ann.hnsw.HnswParams +import com.twitter.ann.hnsw.TypedHnswIndex +import com.twitter.ann.service.query_server.common.QueryableProvider +import com.twitter.ann.service.query_server.common.RefreshableQueryable +import com.twitter.ann.service.query_server.common.UnsafeQueryIndexServer +import com.twitter.ann.service.query_server.common.ValidatedIndexPathProvider +import com.twitter.ann.service.query_server.common.warmup.Warmup +import com.twitter.bijection.Injection +import com.twitter.conversions.DurationOps.richDurationFromInt +import com.twitter.search.common.file.AbstractFile +import com.twitter.search.common.file.FileUtils +import com.twitter.util.Duration +import com.twitter.util.FuturePool + +// Creating a separate hnsw query server object, since unit test require non singleton server. +object HnswQueryIndexServer extends HnswQueryableServer + +class HnswQueryableServer extends UnsafeQueryIndexServer[HnswParams] { + private val IndexGroupPrefix = "group_" + + // given a directory, how to load it as a queryable index + def queryableProvider[T, D <: Distance[D]]: QueryableProvider[T, HnswParams, D] = + new QueryableProvider[T, HnswParams, D] { + override def provideQueryable( + dir: AbstractFile + ): Queryable[T, HnswParams, D] = { + TypedHnswIndex.loadIndex[T, D]( + dimension(), + unsafeMetric.asInstanceOf[Metric[D]], + idInjection[T](), + ReadWriteFuturePool(FuturePool.interruptible(executor)), + dir + ) + } + } + + private def buildQueryable[T, D <: Distance[D]]( + dir: AbstractFile, + grouped: Boolean + ): Queryable[T, HnswParams, D] = { + val queryable = if (refreshable()) { + logger.info(s"build refreshable queryable") + val updatableQueryable = new RefreshableQueryable( + grouped, + dir, + queryableProvider.asInstanceOf[QueryableProvider[T, HnswParams, D]], + ValidatedIndexPathProvider( + minIndexSizeBytes(), + maxIndexSizeBytes(), + statsReceiver.scope("validated_index_provider") + ), + statsReceiver.scope("refreshable_queryable"), + updateInterval = refreshableInterval().minutes + ) + // init first load of index and also schedule the following reloads + updatableQueryable.start() + updatableQueryable.asInstanceOf[QueryableGrouped[T, HnswParams, D]] + } else { + logger.info(s"build non-refreshable queryable") + queryableProvider.provideQueryable(dir).asInstanceOf[Queryable[T, HnswParams, D]] + } + + logger.info("Hnsw queryable created....") + queryable + } + + override def unsafeQueryableMap[T, D <: Distance[D]]: Queryable[T, HnswParams, D] = { + val dir = FileUtils.getFileHandle(indexDirectory()) + buildQueryable(dir, grouped()) + } + + override val runtimeInjection: Injection[HnswParams, ServiceRuntimeParams] = + HnswCommon.RuntimeParamsInjection + + protected override def warmup(): Unit = + if (warmup_enabled()) new HNSWWarmup(unsafeQueryableMap, dimension()).warmup() +} + +class HNSWWarmup(hnsw: Queryable[_, HnswParams, _], dimension: Int) extends Warmup { + protected def minSuccessfulTries: Int = 100 + protected def maxTries: Int = 1000 + protected def timeout: Duration = 50.milliseconds + protected def randomQueryDimension: Int = dimension + + def warmup(): Unit = { + run( + name = "queryWithDistance", + f = hnsw + .queryWithDistance(randomQuery(), 100, HnswParams(ef = 800)) + ) + } +} diff --git a/ann/src/main/scala/com/twitter/ann/util/BUILD b/ann/src/main/scala/com/twitter/ann/util/BUILD new file mode 100644 index 0000000000..6635d23331 --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/util/BUILD @@ -0,0 +1,9 @@ +scala_library( + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "ann/src/main/scala/com/twitter/ann/common", + "util/util-logging", + ], +) diff --git a/ann/src/main/scala/com/twitter/ann/util/IndexBuilderUtils.scala b/ann/src/main/scala/com/twitter/ann/util/IndexBuilderUtils.scala new file mode 100644 index 0000000000..b0245f48aa --- /dev/null +++ b/ann/src/main/scala/com/twitter/ann/util/IndexBuilderUtils.scala @@ -0,0 +1,31 @@ +package com.twitter.ann.util + +import com.twitter.ann.common.{Appendable, EntityEmbedding} +import com.twitter.concurrent.AsyncStream +import com.twitter.logging.Logger +import com.twitter.util.Future +import java.util.concurrent.atomic.AtomicInteger + +object IndexBuilderUtils { + val Log = Logger.apply() + + def addToIndex[T]( + appendable: Appendable[T, _, _], + embeddings: Seq[EntityEmbedding[T]], + concurrencyLevel: Int + ): Future[Int] = { + val count = new AtomicInteger() + // Async stream allows us to procss at most concurrentLevel futures at a time. + Future.Unit.before { + val stream = AsyncStream.fromSeq(embeddings) + val appendStream = stream.mapConcurrent(concurrencyLevel) { annEmbedding => + val processed = count.incrementAndGet() + if (processed % 10000 == 0) { + Log.info(s"Performed $processed updates") + } + appendable.append(annEmbedding) + } + appendStream.size + } + } +} diff --git a/ann/src/main/thrift/com/twitter/ann/common/BUILD b/ann/src/main/thrift/com/twitter/ann/common/BUILD new file mode 100644 index 0000000000..5a0b1897cb --- /dev/null +++ b/ann/src/main/thrift/com/twitter/ann/common/BUILD @@ -0,0 +1,18 @@ +create_thrift_libraries( + base_name = "ann-common", + sources = ["*.thrift"], + platform = "java8", + tags = ["bazel-compatible"], + dependency_roots = [ + "mediaservices/commons/src/main/thrift", + "src/thrift/com/twitter/ml/api:embedding", + ], + generate_languages = [ + "java", + "python", + "scala", + "strato", + ], + provides_java_name = "ann-common-thrift-java", + provides_scala_name = "ann-common-thrift-scala", +) diff --git a/ann/src/main/thrift/com/twitter/ann/common/ann_common.thrift b/ann/src/main/thrift/com/twitter/ann/common/ann_common.thrift new file mode 100644 index 0000000000..b39629f184 --- /dev/null +++ b/ann/src/main/thrift/com/twitter/ann/common/ann_common.thrift @@ -0,0 +1,169 @@ +namespace java com.twitter.ann.common.thriftjava +#@namespace scala com.twitter.ann.common.thriftscala +#@namespace strato com.twitter.ann.common +namespace py gen.twitter.ann.common + +include "com/twitter/mediaservices/commons/ServerCommon.thrift" +include "com/twitter/ml/api/embedding.thrift" + +/** +* Thrift schema for storing file based Index mapping +*/ +struct FileBasedIndexIdStore { + 1: optional map indexIdMap +} + +enum DistanceMetric { + L2, Cosine, InnerProduct, + RESERVED_4, RESERVED_5, RESERVED_6, RESERVED_7, EditDistance +} (persisted = 'true', strato.graphql.typename='DistanceMetric') + +struct AnnoyIndexMetadata { + 1: i32 dimension + 2: DistanceMetric distanceMetric + 3: i32 numOfTrees + 4: i64 numOfVectorsIndexed +} (persisted = 'true', strato.graphql.typename='AnnoyIndexMetadata') + +struct AnnoyRuntimeParam { + /* Number of vectors to evaluate while searching. A larger value will give more accurate results, but will take longer time to return. + * Default value would be numberOfTrees*numberOfNeigboursRequested + */ + 1: optional i32 numOfNodesToExplore +} + +struct HnswRuntimeParam { + // More the value of ef better the recall with but at cost of latency. + // Set it greater than equal to number of neighbours required. + 1: i32 ef +} + +// These options are subset of all possible parameters, defined by +// https://github.com/facebookresearch/faiss/blob/36f2998a6469280cef3b0afcde2036935a29aa1f/faiss/AutoTune.cpp#L444 +// quantizer_ prefix changes IndexIVF.quantizer parameters instead +struct FaissRuntimeParam { + // How many cells to visit in IVFPQ. Higher is slower / more precise. + 1: optional i32 nprobe + // Depth of search in HNSW. Higher is slower / more precise. + 2: optional i32 quantizer_ef + // How many times more neighbours are requested from underlying index by IndexRefine. + 3: optional i32 quantizer_kfactor_rf + // Same as 1: but for quantizer + 4: optional i32 quantizer_nprobe + // Hamming distance threshold to filter neighbours when searching. + 5: optional i32 ht +} + +// Every ANN index will have this metadata and it'll be used by the query service for validation. +struct AnnIndexMetadata { + 1: optional i64 timestamp + 2: optional i32 index_size + 3: optional bool withGroups + 4: optional i32 numGroups +} (persisted = 'true') + +struct HnswIndexMetadata { + 1: i32 dimension + 2: DistanceMetric distanceMetric + 3: i32 numElements +} (persisted = 'true') + +struct HnswInternalIndexMetadata { + 1: i32 maxLevel + 2: optional binary entryPoint + 3: i32 efConstruction + 4: i32 maxM + 5: i32 numElements +} (persisted = 'true') + +struct HnswGraphEntry { + 1: i32 level + 2: binary key + 3: list neighbours +} (persisted = 'true', strato.graphql.typename='HnswGraphEntry') + +enum IndexType { + TWEET, + USER, + WORD, + LONG, + INT, + STRING, + RESERVED_7, RESERVED_8, RESERVED_9, RESERVED_10 +} (persisted = 'true', strato.graphql.typename='IndexType') + +struct CosineDistance { + 1: required double distance +} + +struct L2Distance { + 1: required double distance +} + +struct InnerProductDistance { + 1: required double distance +} + +struct EditDistance { + 1: required i32 distance +} + +union Distance { + 1: CosineDistance cosineDistance + 2: L2Distance l2Distance + 3: InnerProductDistance innerProductDistance + 4: EditDistance editDistance +} + +struct NearestNeighbor { + 1: required binary id + 2: optional Distance distance +} + +struct NearestNeighborResult { + // This list is ordered from nearest to furthest neighbor + 1: required list nearestNeighbors +} + +// Different runtime/tuning params while querying for indexes to control accuracy/latency etc.. +union RuntimeParams { + 1: AnnoyRuntimeParam annoyParam + 2: HnswRuntimeParam hnswParam + 3: FaissRuntimeParam faissParam +} + +struct NearestNeighborQuery { + 1: required embedding.Embedding embedding + 2: required bool with_distance + 3: required RuntimeParams runtimeParams, + 4: required i32 numberOfNeighbors, + // The purpose of the key here is to load the index in memory as a map of Option[key] to index + // If the key is not specified in the query, the map value corresponding to None key will be used + // as the queryable index to perform Nearest Neighbor search on + 5: optional string key +} + +enum BadRequestCode { + VECTOR_DIMENSION_MISMATCH, + RESERVED_2, + RESERVED_3, + RESERVED_4, + RESERVED_5, + RESERVED_6, + RESERVED_7, + RESERVED_8, + RESERVED_9 +} + +exception BadRequest { + 1: string message + 2: required BadRequestCode code +} + +service AnnQueryService { + /** + * Get approximate nearest neighbor for a given vector + */ + NearestNeighborResult query(1: NearestNeighborQuery query) + throws (1: ServerCommon.ServerError serverError, 2: BadRequest badRequest) +} diff --git a/ann/src/main/thrift/com/twitter/ann/knn/BUILD b/ann/src/main/thrift/com/twitter/ann/knn/BUILD new file mode 100644 index 0000000000..5c7c1c7847 --- /dev/null +++ b/ann/src/main/thrift/com/twitter/ann/knn/BUILD @@ -0,0 +1,11 @@ +create_thrift_libraries( + base_name = "thrift", + sources = ["*.thrift"], + platform = "java8", + tags = ["bazel-compatible"], + dependency_roots = ["src/thrift/com/twitter/ml/featurestore:ml-feature-store"], + generate_languages = [ + "java", + "scala", + ], +) diff --git a/ann/src/main/thrift/com/twitter/ann/knn/knn.thrift b/ann/src/main/thrift/com/twitter/ann/knn/knn.thrift new file mode 100644 index 0000000000..bf4aba184e --- /dev/null +++ b/ann/src/main/thrift/com/twitter/ann/knn/knn.thrift @@ -0,0 +1,15 @@ +namespace java com.twitter.ann.knn.thriftjava +#@namespace scala com.twitter.ann.knn.thriftscala +namespace py gen.twitter.ann.knn + +include "com/twitter/ml/featurestore/entity.thrift" + +struct Neighbor { + 1: required double distance + 2: required entity.EntityId id +} (persisted = "true") + +struct Knn { + 1: required entity.EntityId queryId + 2: required list neighbors +}(persisted='true') diff --git a/ann/src/main/thrift/com/twitter/ann/serialization/BUILD b/ann/src/main/thrift/com/twitter/ann/serialization/BUILD new file mode 100644 index 0000000000..aa35395326 --- /dev/null +++ b/ann/src/main/thrift/com/twitter/ann/serialization/BUILD @@ -0,0 +1,13 @@ +create_thrift_libraries( + base_name = "serialization", + sources = ["*.thrift"], + platform = "java8", + tags = ["bazel-compatible"], + dependency_roots = [ + "src/thrift/com/twitter/ml/api:embedding", + ], + generate_languages = [ + "java", + "scala", + ], +) diff --git a/ann/src/main/thrift/com/twitter/ann/serialization/serialization.thrift b/ann/src/main/thrift/com/twitter/ann/serialization/serialization.thrift new file mode 100644 index 0000000000..fcfa47a56b --- /dev/null +++ b/ann/src/main/thrift/com/twitter/ann/serialization/serialization.thrift @@ -0,0 +1,10 @@ +#@namespace scala com.twitter.ann.serialization.thriftscala + +include "com/twitter/ml/api/embedding.thrift" +/** +* Thrift schema for storing embeddings in a file +*/ +struct PersistedEmbedding { + 1: required binary id + 2: required embedding.Embedding embedding +}(persisted = 'true') diff --git a/ci/ci.sh b/ci/ci.sh new file mode 100755 index 0000000000..c52d3c26b3 --- /dev/null +++ b/ci/ci.sh @@ -0,0 +1,3 @@ +#!/bin/sh + +exit 0 diff --git a/cr-mixer/BUILD.bazel b/cr-mixer/BUILD.bazel new file mode 100644 index 0000000000..75890d1336 --- /dev/null +++ b/cr-mixer/BUILD.bazel @@ -0,0 +1,24 @@ +jvm_binary( + name = "bin", + basename = "cr-mixer", + main = "com.twitter.cr_mixer.CrMixerServerMain", + runtime_platform = "java11", + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/ch/qos/logback:logback-classic", + "cr-mixer/server/src/main/scala/com/twitter/cr_mixer", + "finagle/finagle-zipkin-scribe/src/main/scala", + "finatra/inject/inject-logback/src/main/scala", + "loglens/loglens-logback/src/main/scala/com/twitter/loglens/logback", + "twitter-server-internal/src/main/scala", + "twitter-server/logback-classic/src/main/scala", + ], +) + +# Aurora Workflows build phase convention requires a jvm_app named with ${project-name}-app +jvm_app( + name = "cr-mixer-app", + archive = "zip", + binary = ":bin", + tags = ["bazel-compatible"], +) diff --git a/cr-mixer/README.md b/cr-mixer/README.md new file mode 100644 index 0000000000..75c0a1553d --- /dev/null +++ b/cr-mixer/README.md @@ -0,0 +1,7 @@ +# CR-Mixer + +CR-Mixer is a candidate generation service proposed as part of the Personalization Strategy vision for Twitter. Its aim is to speed up the iteration and development of candidate generation and light ranking. The service acts as a lightweight coordinating layer that delegates candidate generation tasks to underlying compute services. It focuses on Twitter's candidate generation use cases and offers a centralized platform for fetching, mixing, and managing candidate sources and light rankers. The overarching goal is to increase the speed and ease of testing and developing candidate generation pipelines, ultimately delivering more value to Twitter users. + +CR-Mixer act as a configurator and delegator, providing abstractions for the challenging parts of candidate generation and handling performance issues. It will offer a 1-stop-shop for fetching and mixing candidate sources, a managed and shared performant platform, a light ranking layer, a common filtering layer, a version control system, a co-owned feature switch set, and peripheral tooling. + +CR-Mixer's pipeline consists of 4 steps: source signal extraction, candidate generation, filtering, and ranking. It also provides peripheral tooling like scribing, debugging, and monitoring. The service fetches source signals externally from stores like UserProfileService and RealGraph, calls external candidate generation services, and caches results. Filters are applied for deduping and pre-ranking, and a light ranking step follows. \ No newline at end of file diff --git a/cr-mixer/server/src/main/resources/BUILD.bazel b/cr-mixer/server/src/main/resources/BUILD.bazel new file mode 100644 index 0000000000..8f96f402c1 --- /dev/null +++ b/cr-mixer/server/src/main/resources/BUILD.bazel @@ -0,0 +1,8 @@ +resources( + sources = [ + "*.xml", + "*.yml", + "config/*.yml", + ], + tags = ["bazel-compatible"], +) diff --git a/cr-mixer/server/src/main/resources/config/decider.yml b/cr-mixer/server/src/main/resources/config/decider.yml new file mode 100644 index 0000000000..a0d55b9b4e --- /dev/null +++ b/cr-mixer/server/src/main/resources/config/decider.yml @@ -0,0 +1,146 @@ +# The keys in this file correspond to the DeciderValues defined in +# https://sourcegraph.twitter.biz/git.twitter.biz/source/-/blob/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/decider/DeciderKey.scala + +dark_traffic_filter: + comment: Proportion of the requests that are forwarded as dark traffic to the proxy + default_availability: 0 + +enable_tweet_recommendations_home_product: + comment: Proportion of requests where we return an actual response for TweetRecommendations Home product + default_availability: 10000 + +enable_tweet_health_score: + comment: "Enable the calculation for health scores in tweetInfo. By enabling this decider, we will compute TweetHealthModelScore" + default_availability: 10000 + +enable_user_agatha_score: + comment: "Enable the calculation for health scores in tweetInfo. By enabling this decider, we will compute UserHealthModelScore" + default_availability: 10000 + +enable_user_tweet_entity_graph_traffic: + comment: "Enable the traffic to user entity tweet graph to fetch liked-by tweets candidates" + default_availability: 10000 + +enable_user_tweet_graph_traffic: + comment: "Enable the traffic to user tweet graph to fetch similar tweets candidates" + default_availability: 10000 + +enable_user_video_graph_traffic: + comment: "Enable the traffic to user video graph to fetch similar tweets candidates" + default_availability: 10000 + +enable_user_ad_graph_traffic: + comment: "Enable the traffic to user ad graph to fetch similar tweets candidates" + default_availability: 10000 + +enable_qig_similar_tweets_traffic: + comment: "Enable the traffic to QIG to fetch similar tweet candidates" + default_availability: 0 + +enable_frs_traffic: + comment: "Enable the traffic to FRS to fetch user follow recommendations" + default_availability: 0 + +enable_hydra_dark_traffic: + comment: "Enable dark traffic to hydra" + default_availability: 0 + +enable_real_graph_mh_store: + comment: "Enable traffic for the real graph manhattan based store" + default_availability: 0 + +enable_simclusters_ann_experimental_dark_traffic: + comment: "Enable dark traffic to simclusters-ann-experimental" + default_availability: 0 + +enable_simclusters_ann_2_dark_traffic: + comment: "Enable dark traffic to prod SimClustersANN2" + default_availability: 0 + +enable_user_state_store: + comment: "Enable traffic user state store to hydrate user state" + default_availability: 0 + +upper_funnel_per_step_scribe_rate: + comment: "Enable Upper Funnel Event Scribe Sampling (fetch, pre-rank, interleave etc.) for getTweetsRecommendations() endpoint" + default_availability: 0 + +kafka_message_scribe_sample_rate: + comment: "Gates the production of forked scribe messages to kafka for the async feature hydrator" + default_availability: 0 + +top_level_api_ddg_metrics_scribe_rate: + comment: "Enable Top Level API DDG Metrics Scribe Sampling for getTweetsRecommendations() endpoint" + default_availability: 0 + +ads_recommendations_per_experiment_scribe_rate: + comment: "Percentage of DDG traffic to Scribe for getAdsRecommendations() endpoint" + default_availability: 0 + +enable_loadshedding_getTweetRecommendations: + comment: "Enable loadshedding (from 0% to 100%). Requests that have been shed will return an empty response" + default_availability: 0 + +enable_loadshedding_getTweetRecommendations_Home: + comment: "Enable loadshedding (from 0% to 100%). Requests that have been shed will return an empty response" + default_availability: 0 + +enable_loadshedding_getTweetRecommendations_Notifications: + comment: "Enable loadshedding (from 0% to 100%). Requests that have been shed will return an empty response" + default_availability: 0 + +enable_loadshedding_getTweetRecommendations_Email: + comment: "Enable loadshedding (from 0% to 100%). Requests that have been shed will return an empty response" + default_availability: 0 + +enable_loadshedding_getRelatedTweetsForQueryTweet: + comment: "Enable loadshedding (from 0% to 100%). Requests that have been shed will return an empty response" + default_availability: 0 + +enable_loadshedding_getRelatedTweetsForQueryTweet_Home: + comment: "Enable loadshedding (from 0% to 100%). Requests that have been shed will return an empty response" + default_availability: 0 + +enable_loadshedding_getRelatedTweetsForQueryTweet_MoreTweetsModule: + comment: "Enable loadshedding (from 0% to 100%). Requests that have been shed will return an empty response" + default_availability: 0 + +enable_loadshedding_getRelatedTweetsForQueryAuthor: + comment: "Enable loadshedding (from 0% to 100%). Requests that have been shed will return an empty response" + default_availability: 0 + +enable_loadshedding_getRelatedTweetsForQueryAuthor_MoreTweetsModule: + comment: "Enable loadshedding (from 0% to 100%). Requests that have been shed will return an empty response" + default_availability: 0 + +enable_loadshedding_getFrsBasedTweetRecommendations_Home: + comment: "Enable loadshedding (from 0% to 100%). Requests that have been shed will return an empty response" + default_availability: 0 + +enable_loadshedding_getFrsBasedTweetRecommendations_Notifications: + comment: "Enable loadshedding (from 0% to 100%). Requests that have been shed will return an empty response" + default_availability: 0 + +enable_user_media_representation_store: + comment: "Enable fetching user nudity rate signal from Media Understanding" + default_availability: 0 + +enable_magic_recs_real_time_aggregates_store: + comment: "Enable fetching real time aggregates features from Magic Recs memcache" + default_availability: 0 + +enable_utg_realtime_tweet_engagement_score: + comment: "Enable fetching real time tweet engagement score from utg-plus" + default_availability: 0 + +get_tweet_recommendations_cache_rate: + comment: "Proportion of users where getTweetRecommendations() request and responses will be cached" + default_availability: 1000 + +enable_earlybird_traffic: + comment: "Enable fetching tweet candidates from Earlybird" + default_availability: 0 + +enable_scribe_for_blue_verified_tweet_candidates: + comment: "Enable scribing for tweet candidates from Blue Verified users" + default_availability: 0 diff --git a/cr-mixer/server/src/main/resources/logback.xml b/cr-mixer/server/src/main/resources/logback.xml new file mode 100644 index 0000000000..24e7fd57e1 --- /dev/null +++ b/cr-mixer/server/src/main/resources/logback.xml @@ -0,0 +1,168 @@ + + + + + + + + + + + + + + + + + true + + + + + + + + + + + ${log.service.output} + + + ${log.service.output}.%d.gz + + 21 + true + + + %date %.-3level ${DEFAULT_SERVICE_PATTERN}%n + + + + + + ${log.access.output} + + + ${log.access.output}.%d.gz + + 21 + true + + + ${DEFAULT_ACCESS_PATTERN}%n + + + + + + true + ${log.lens.category} + ${log.lens.index} + ${log.lens.tag}/service + + %msg + + + + + + true + ${log.lens.category} + ${log.lens.index} + ${log.lens.tag}/access + + %msg + + + + + + allow_listed_pipeline_executions.log + + + allow_listed_pipeline_executions.log.%d.gz + + 7 + true + + + %date %.-3level ${DEFAULT_SERVICE_PATTERN}%n + + + + + + + + + + + + ${async_queue_size} + ${async_max_flush_time} + + + + + ${async_queue_size} + ${async_max_flush_time} + + + + + ${async_queue_size} + ${async_max_flush_time} + + + + + ${async_queue_size} + ${async_max_flush_time} + + + + + ${async_queue_size} + ${async_max_flush_time} + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/BUILD.bazel b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/BUILD.bazel new file mode 100644 index 0000000000..533a86c1fc --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/BUILD.bazel @@ -0,0 +1,48 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + strict_deps = True, + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/com/google/inject:guice", + "3rdparty/jvm/javax/inject:javax.inject", + "3rdparty/jvm/net/codingwell:scala-guice", + "3rdparty/jvm/org/slf4j:slf4j-api", + "cr-mixer/server/src/main/resources", + "cr-mixer/server/src/main/scala/com/twitter/cr_mixer/candidate_generation", + "cr-mixer/server/src/main/scala/com/twitter/cr_mixer/controller", + "cr-mixer/server/src/main/scala/com/twitter/cr_mixer/featureswitch", + "cr-mixer/server/src/main/scala/com/twitter/cr_mixer/logging", + "cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module", + "cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param", + "cr-mixer/server/src/main/scala/com/twitter/cr_mixer/scribe", + "cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine", + "cr-mixer/thrift/src/main/thrift:thrift-scala", + "decider/src/main/scala", + "finagle/finagle-core/src/main", + "finagle/finagle-http/src/main/scala", + "finagle/finagle-thriftmux/src/main/scala", + "finatra-internal/mtls-http/src/main/scala", + "finatra-internal/mtls-thriftmux/src/main/scala", + "finatra/http-core/src/main/java/com/twitter/finatra/http", + "finatra/inject/inject-app/src/main/scala", + "finatra/inject/inject-core/src/main/scala", + "finatra/inject/inject-server/src/main/scala", + "finatra/inject/inject-utils/src/main/scala", + "finatra/utils/src/main/java/com/twitter/finatra/annotations", + "hydra/common/libraries/src/main/scala/com/twitter/hydra/common/model_config", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/controllers", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/module", + "product-mixer/core/src/main/thrift/com/twitter/product_mixer/core:thrift-scala", + "relevance-platform/src/main/scala/com/twitter/relevance_platform/common/filters", + "src/thrift/com/twitter/timelines/render:thrift-scala", + "thrift-web-forms/src/main/scala/com/twitter/thriftwebforms", + "thrift-web-forms/src/main/scala/com/twitter/thriftwebforms/view", + "timelines/src/main/scala/com/twitter/timelines/features/app", + "twitter-server-internal", + "twitter-server/server/src/main/scala", + "util/util-app/src/main/scala", + "util/util-core:scala", + "util/util-slf4j-api/src/main/scala", + ], +) diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/CrMixerHttpServerWarmupHandler.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/CrMixerHttpServerWarmupHandler.scala new file mode 100644 index 0000000000..85e302d2a3 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/CrMixerHttpServerWarmupHandler.scala @@ -0,0 +1,18 @@ +package com.twitter.cr_mixer + +import com.twitter.finatra.http.routing.HttpWarmup +import com.twitter.finatra.httpclient.RequestBuilder._ +import com.twitter.inject.Logging +import com.twitter.inject.utils.Handler +import com.twitter.util.Try +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class CrMixerHttpServerWarmupHandler @Inject() (warmup: HttpWarmup) extends Handler with Logging { + + override def handle(): Unit = { + Try(warmup.send(get("/admin/cr-mixer/product-pipelines"), admin = true)()) + .onFailure(e => error(e.getMessage, e)) + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/CrMixerServer.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/CrMixerServer.scala new file mode 100644 index 0000000000..887aab83f6 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/CrMixerServer.scala @@ -0,0 +1,229 @@ +package com.twitter.cr_mixer + +import com.google.inject.Module +import com.twitter.cr_mixer.controller.CrMixerThriftController +import com.twitter.cr_mixer.featureswitch.SetImpressedBucketsLocalContextFilter +import com.twitter.cr_mixer.module.ActivePromotedTweetStoreModule +import com.twitter.cr_mixer.module.CertoStratoStoreModule +import com.twitter.cr_mixer.module.CrMixerParamConfigModule +import com.twitter.cr_mixer.module.EmbeddingStoreModule +import com.twitter.cr_mixer.module.FrsStoreModule +import com.twitter.cr_mixer.module.MHMtlsParamsModule +import com.twitter.cr_mixer.module.OfflineCandidateStoreModule +import com.twitter.cr_mixer.module.RealGraphStoreMhModule +import com.twitter.cr_mixer.module.RealGraphOonStoreModule +import com.twitter.cr_mixer.module.RepresentationManagerModule +import com.twitter.cr_mixer.module.RepresentationScorerModule +import com.twitter.cr_mixer.module.TweetInfoStoreModule +import com.twitter.cr_mixer.module.TweetRecentEngagedUserStoreModule +import com.twitter.cr_mixer.module.TweetRecommendationResultsStoreModule +import com.twitter.cr_mixer.module.TripCandidateStoreModule +import com.twitter.cr_mixer.module.TwhinCollabFilterStratoStoreModule +import com.twitter.cr_mixer.module.UserSignalServiceColumnModule +import com.twitter.cr_mixer.module.UserSignalServiceStoreModule +import com.twitter.cr_mixer.module.UserStateStoreModule +import com.twitter.cr_mixer.module.core.ABDeciderModule +import com.twitter.cr_mixer.module.core.CrMixerFlagModule +import com.twitter.cr_mixer.module.core.CrMixerLoggingABDeciderModule +import com.twitter.cr_mixer.module.core.FeatureContextBuilderModule +import com.twitter.cr_mixer.module.core.FeatureSwitchesModule +import com.twitter.cr_mixer.module.core.KafkaProducerModule +import com.twitter.cr_mixer.module.core.LoggerFactoryModule +import com.twitter.cr_mixer.module.similarity_engine.ConsumerEmbeddingBasedTripSimilarityEngineModule +import com.twitter.cr_mixer.module.similarity_engine.ConsumerEmbeddingBasedTwHINSimilarityEngineModule +import com.twitter.cr_mixer.module.similarity_engine.ConsumerEmbeddingBasedTwoTowerSimilarityEngineModule +import com.twitter.cr_mixer.module.similarity_engine.ConsumersBasedUserAdGraphSimilarityEngineModule +import com.twitter.cr_mixer.module.similarity_engine.ConsumersBasedUserVideoGraphSimilarityEngineModule +import com.twitter.cr_mixer.module.similarity_engine.ProducerBasedUserAdGraphSimilarityEngineModule +import com.twitter.cr_mixer.module.similarity_engine.ProducerBasedUserTweetGraphSimilarityEngineModule +import com.twitter.cr_mixer.module.similarity_engine.ProducerBasedUnifiedSimilarityEngineModule +import com.twitter.cr_mixer.module.similarity_engine.SimClustersANNSimilarityEngineModule +import com.twitter.cr_mixer.module.similarity_engine.TweetBasedUnifiedSimilarityEngineModule +import com.twitter.cr_mixer.module.similarity_engine.TweetBasedQigSimilarityEngineModule +import com.twitter.cr_mixer.module.similarity_engine.TweetBasedTwHINSimlarityEngineModule +import com.twitter.cr_mixer.module.similarity_engine.TweetBasedUserAdGraphSimilarityEngineModule +import com.twitter.cr_mixer.module.similarity_engine.TweetBasedUserTweetGraphSimilarityEngineModule +import com.twitter.cr_mixer.module.similarity_engine.TweetBasedUserVideoGraphSimilarityEngineModule +import com.twitter.cr_mixer.module.similarity_engine.TwhinCollabFilterLookupSimilarityEngineModule +import com.twitter.cr_mixer.module.ConsumersBasedUserAdGraphStoreModule +import com.twitter.cr_mixer.module.ConsumersBasedUserTweetGraphStoreModule +import com.twitter.cr_mixer.module.ConsumersBasedUserVideoGraphStoreModule +import com.twitter.cr_mixer.module.DiffusionStoreModule +import com.twitter.cr_mixer.module.EarlybirdRecencyBasedCandidateStoreModule +import com.twitter.cr_mixer.module.TwiceClustersMembersStoreModule +import com.twitter.cr_mixer.module.StrongTiePredictionStoreModule +import com.twitter.cr_mixer.module.thrift_client.AnnQueryServiceClientModule +import com.twitter.cr_mixer.module.thrift_client.EarlybirdSearchClientModule +import com.twitter.cr_mixer.module.thrift_client.FrsClientModule +import com.twitter.cr_mixer.module.thrift_client.QigServiceClientModule +import com.twitter.cr_mixer.module.thrift_client.SimClustersAnnServiceClientModule +import com.twitter.cr_mixer.module.thrift_client.TweetyPieClientModule +import com.twitter.cr_mixer.module.thrift_client.UserTweetGraphClientModule +import com.twitter.cr_mixer.module.thrift_client.UserTweetGraphPlusClientModule +import com.twitter.cr_mixer.module.thrift_client.UserVideoGraphClientModule +import com.twitter.cr_mixer.{thriftscala => st} +import com.twitter.finagle.Filter +import com.twitter.finatra.annotations.DarkTrafficFilterType +import com.twitter.finatra.decider.modules.DeciderModule +import com.twitter.finatra.http.HttpServer +import com.twitter.finatra.http.routing.HttpRouter +import com.twitter.finatra.jackson.modules.ScalaObjectMapperModule +import com.twitter.finatra.mtls.http.{Mtls => HttpMtls} +import com.twitter.finatra.mtls.thriftmux.Mtls +import com.twitter.finatra.mtls.thriftmux.modules.MtlsThriftWebFormsModule +import com.twitter.finatra.thrift.ThriftServer +import com.twitter.finatra.thrift.filters._ +import com.twitter.finatra.thrift.routing.ThriftRouter +import com.twitter.hydra.common.model_config.{ConfigModule => HydraConfigModule} +import com.twitter.inject.thrift.modules.ThriftClientIdModule +import com.twitter.product_mixer.core.module.LoggingThrowableExceptionMapper +import com.twitter.product_mixer.core.module.StratoClientModule +import com.twitter.product_mixer.core.module.product_mixer_flags.ProductMixerFlagModule +import com.twitter.relevance_platform.common.filters.ClientStatsFilter +import com.twitter.relevance_platform.common.filters.DarkTrafficFilterModule +import com.twitter.cr_mixer.module.SimClustersANNServiceNameToClientMapper +import com.twitter.cr_mixer.module.SkitStratoStoreModule +import com.twitter.cr_mixer.module.BlueVerifiedAnnotationStoreModule +import com.twitter.cr_mixer.module.core.TimeoutConfigModule +import com.twitter.cr_mixer.module.grpc_client.NaviGRPCClientModule +import com.twitter.cr_mixer.module.similarity_engine.CertoTopicTweetSimilarityEngineModule +import com.twitter.cr_mixer.module.similarity_engine.ConsumerBasedWalsSimilarityEngineModule +import com.twitter.cr_mixer.module.similarity_engine.DiffusionBasedSimilarityEngineModule +import com.twitter.cr_mixer.module.similarity_engine.EarlybirdSimilarityEngineModule +import com.twitter.cr_mixer.module.similarity_engine.SkitTopicTweetSimilarityEngineModule +import com.twitter.cr_mixer.module.similarity_engine.UserTweetEntityGraphSimilarityEngineModule +import com.twitter.cr_mixer.module.thrift_client.HydraPartitionClientModule +import com.twitter.cr_mixer.module.thrift_client.HydraRootClientModule +import com.twitter.cr_mixer.module.thrift_client.UserAdGraphClientModule +import com.twitter.cr_mixer.module.thrift_client.UserTweetEntityGraphClientModule +import com.twitter.thriftwebforms.MethodOptions + +object CrMixerServerMain extends CrMixerServer + +class CrMixerServer extends ThriftServer with Mtls with HttpServer with HttpMtls { + override val name = "cr-mixer-server" + + private val coreModules = Seq( + ABDeciderModule, + CrMixerFlagModule, + CrMixerLoggingABDeciderModule, + CrMixerParamConfigModule, + new DarkTrafficFilterModule[st.CrMixer.ReqRepServicePerEndpoint](), + DeciderModule, + FeatureContextBuilderModule, + FeatureSwitchesModule, + KafkaProducerModule, + LoggerFactoryModule, + MHMtlsParamsModule, + ProductMixerFlagModule, + ScalaObjectMapperModule, + ThriftClientIdModule + ) + + private val thriftClientModules = Seq( + AnnQueryServiceClientModule, + EarlybirdSearchClientModule, + FrsClientModule, + HydraPartitionClientModule, + HydraRootClientModule, + QigServiceClientModule, + SimClustersAnnServiceClientModule, + TweetyPieClientModule, + UserAdGraphClientModule, + UserTweetEntityGraphClientModule, + UserTweetGraphClientModule, + UserTweetGraphPlusClientModule, + UserVideoGraphClientModule, + ) + + private val grpcClientModules = Seq( + NaviGRPCClientModule + ) + + // Modules sorted alphabetically, please keep the order when adding a new module + override val modules: Seq[Module] = + coreModules ++ thriftClientModules ++ grpcClientModules ++ + Seq( + ActivePromotedTweetStoreModule, + CertoStratoStoreModule, + CertoTopicTweetSimilarityEngineModule, + ConsumersBasedUserAdGraphSimilarityEngineModule, + ConsumersBasedUserTweetGraphStoreModule, + ConsumersBasedUserVideoGraphSimilarityEngineModule, + ConsumersBasedUserVideoGraphStoreModule, + ConsumerEmbeddingBasedTripSimilarityEngineModule, + ConsumerEmbeddingBasedTwHINSimilarityEngineModule, + ConsumerEmbeddingBasedTwoTowerSimilarityEngineModule, + ConsumersBasedUserAdGraphStoreModule, + ConsumerBasedWalsSimilarityEngineModule, + DiffusionStoreModule, + EmbeddingStoreModule, + EarlybirdSimilarityEngineModule, + EarlybirdRecencyBasedCandidateStoreModule, + FrsStoreModule, + HydraConfigModule, + OfflineCandidateStoreModule, + ProducerBasedUnifiedSimilarityEngineModule, + ProducerBasedUserAdGraphSimilarityEngineModule, + ProducerBasedUserTweetGraphSimilarityEngineModule, + RealGraphOonStoreModule, + RealGraphStoreMhModule, + RepresentationManagerModule, + RepresentationScorerModule, + SimClustersANNServiceNameToClientMapper, + SimClustersANNSimilarityEngineModule, + SkitStratoStoreModule, + SkitTopicTweetSimilarityEngineModule, + StratoClientModule, + StrongTiePredictionStoreModule, + TimeoutConfigModule, + TripCandidateStoreModule, + TwiceClustersMembersStoreModule, + TweetBasedQigSimilarityEngineModule, + TweetBasedTwHINSimlarityEngineModule, + TweetBasedUnifiedSimilarityEngineModule, + TweetBasedUserAdGraphSimilarityEngineModule, + TweetBasedUserTweetGraphSimilarityEngineModule, + TweetBasedUserVideoGraphSimilarityEngineModule, + TweetInfoStoreModule, + TweetRecentEngagedUserStoreModule, + TweetRecommendationResultsStoreModule, + TwhinCollabFilterStratoStoreModule, + TwhinCollabFilterLookupSimilarityEngineModule, + UserSignalServiceColumnModule, + UserSignalServiceStoreModule, + UserStateStoreModule, + UserTweetEntityGraphSimilarityEngineModule, + DiffusionBasedSimilarityEngineModule, + BlueVerifiedAnnotationStoreModule, + new MtlsThriftWebFormsModule[st.CrMixer.MethodPerEndpoint](this) { + override protected def defaultMethodAccess: MethodOptions.Access = { + MethodOptions.Access.ByLdapGroup( + Seq( + "cr-mixer-admins", + "recosplat-sensitive-data-medium", + "recos-platform-admins", + )) + } + } + ) + + def configureThrift(router: ThriftRouter): Unit = { + router + .filter[LoggingMDCFilter] + .filter[TraceIdMDCFilter] + .filter[ThriftMDCFilter] + .filter[ClientStatsFilter] + .filter[AccessLoggingFilter] + .filter[SetImpressedBucketsLocalContextFilter] + .filter[ExceptionMappingFilter] + .filter[Filter.TypeAgnostic, DarkTrafficFilterType] + .exceptionMapper[LoggingThrowableExceptionMapper] + .add[CrMixerThriftController] + } + + override protected def warmup(): Unit = { + handle[CrMixerThriftServerWarmupHandler]() + handle[CrMixerHttpServerWarmupHandler]() + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/CrMixerThriftServerWarmupHandler.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/CrMixerThriftServerWarmupHandler.scala new file mode 100644 index 0000000000..46c46c92b2 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/CrMixerThriftServerWarmupHandler.scala @@ -0,0 +1,75 @@ +package com.twitter.cr_mixer + +import com.twitter.finagle.thrift.ClientId +import com.twitter.finatra.thrift.routing.ThriftWarmup +import com.twitter.inject.Logging +import com.twitter.inject.utils.Handler +import com.twitter.product_mixer.core.{thriftscala => pt} +import com.twitter.cr_mixer.{thriftscala => st} +import com.twitter.scrooge.Request +import com.twitter.scrooge.Response +import com.twitter.util.Return +import com.twitter.util.Throw +import com.twitter.util.Try +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class CrMixerThriftServerWarmupHandler @Inject() (warmup: ThriftWarmup) + extends Handler + with Logging { + + private val clientId = ClientId("thrift-warmup-client") + + def handle(): Unit = { + val testIds = Seq(1, 2, 3) + try { + clientId.asCurrent { + testIds.foreach { id => + val warmupReq = warmupQuery(id) + info(s"Sending warm-up request to service with query: $warmupReq") + warmup.sendRequest( + method = st.CrMixer.GetTweetRecommendations, + req = Request(st.CrMixer.GetTweetRecommendations.Args(warmupReq)))(assertWarmupResponse) + } + } + } catch { + case e: Throwable => + // we don't want a warmup failure to prevent start-up + error(e.getMessage, e) + } + info("Warm-up done.") + } + + private def warmupQuery(userId: Long): st.CrMixerTweetRequest = { + val clientContext = pt.ClientContext( + userId = Some(userId), + guestId = None, + appId = Some(258901L), + ipAddress = Some("0.0.0.0"), + userAgent = Some("FAKE_USER_AGENT_FOR_WARMUPS"), + countryCode = Some("US"), + languageCode = Some("en"), + isTwoffice = None, + userRoles = None, + deviceId = Some("FAKE_DEVICE_ID_FOR_WARMUPS") + ) + st.CrMixerTweetRequest( + clientContext = clientContext, + product = st.Product.Home, + productContext = Some(st.ProductContext.HomeContext(st.HomeContext())), + ) + } + + private def assertWarmupResponse( + result: Try[Response[st.CrMixer.GetTweetRecommendations.SuccessType]] + ): Unit = { + // we collect and log any exceptions from the result. + result match { + case Return(_) => // ok + case Throw(exception) => + warn("Error performing warm-up request.") + error(exception.getMessage, exception) + } + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/blender/AdsBlender.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/blender/AdsBlender.scala new file mode 100644 index 0000000000..4e8f0a41de --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/blender/AdsBlender.scala @@ -0,0 +1,77 @@ +package com.twitter.cr_mixer.blender + +import com.twitter.cr_mixer.model.BlendedAdsCandidate +import com.twitter.cr_mixer.model.CandidateGenerationInfo +import com.twitter.cr_mixer.model.InitialAdsCandidate +import com.twitter.cr_mixer.util.InterleaveUtil +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.simclusters_v2.common.TweetId +import com.twitter.util.Future +import javax.inject.Inject +import javax.inject.Singleton +import scala.collection.mutable + +@Singleton +case class AdsBlender @Inject() (globalStats: StatsReceiver) { + + private val name: String = this.getClass.getCanonicalName + private val stats: StatsReceiver = globalStats.scope(name) + + /** + * Interleaves candidates by iteratively choosing InterestedIn candidates and TWISTLY candidates + * in turn. InterestedIn candidates have no source signal, whereas TWISTLY candidates do. TWISTLY + * candidates themselves are interleaved by source before equal blending with InterestedIn + * candidates. + */ + def blend( + inputCandidates: Seq[Seq[InitialAdsCandidate]], + ): Future[Seq[BlendedAdsCandidate]] = { + + // Filter out empty candidate sequence + val candidates = inputCandidates.filter(_.nonEmpty) + val (interestedInCandidates, twistlyCandidates) = + candidates.partition(_.head.candidateGenerationInfo.sourceInfoOpt.isEmpty) + // First interleave twistly candidates + val interleavedTwistlyCandidates = InterleaveUtil.interleave(twistlyCandidates) + + val twistlyAndInterestedInCandidates = + Seq(interestedInCandidates.flatten, interleavedTwistlyCandidates) + + // then interleave twistly candidates with interested in to make them even + val interleavedCandidates = InterleaveUtil.interleave(twistlyAndInterestedInCandidates) + + stats.stat("candidates").add(interleavedCandidates.size) + + val blendedCandidates = buildBlendedAdsCandidate(inputCandidates, interleavedCandidates) + Future.value(blendedCandidates) + } + private def buildBlendedAdsCandidate( + inputCandidates: Seq[Seq[InitialAdsCandidate]], + interleavedCandidates: Seq[InitialAdsCandidate] + ): Seq[BlendedAdsCandidate] = { + val cgInfoLookupMap = buildCandidateToCGInfosMap(inputCandidates) + interleavedCandidates.map { interleavedCandidate => + interleavedCandidate.toBlendedAdsCandidate(cgInfoLookupMap(interleavedCandidate.tweetId)) + } + } + + private def buildCandidateToCGInfosMap( + candidateSeq: Seq[Seq[InitialAdsCandidate]], + ): Map[TweetId, Seq[CandidateGenerationInfo]] = { + val tweetIdMap = mutable.HashMap[TweetId, Seq[CandidateGenerationInfo]]() + + candidateSeq.foreach { candidates => + candidates.foreach { candidate => + val candidateGenerationInfoSeq = { + tweetIdMap.getOrElse(candidate.tweetId, Seq.empty) + } + val candidateGenerationInfo = candidate.candidateGenerationInfo + tweetIdMap.put( + candidate.tweetId, + candidateGenerationInfoSeq ++ Seq(candidateGenerationInfo)) + } + } + tweetIdMap.toMap + } + +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/blender/BUILD b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/blender/BUILD new file mode 100644 index 0000000000..604e35f991 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/blender/BUILD @@ -0,0 +1,20 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + strict_deps = True, + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/com/twitter/storehaus:core", + "3rdparty/jvm/javax/inject:javax.inject", + "3rdparty/src/jvm/com/twitter/storehaus:core", + "configapi/configapi-core", + "content-recommender/thrift/src/main/thrift:thrift-scala", + "cr-mixer/server/src/main/scala/com/twitter/cr_mixer/model", + "cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param", + "cr-mixer/server/src/main/scala/com/twitter/cr_mixer/util", + "cr-mixer/thrift/src/main/thrift:thrift-scala", + "snowflake/src/main/scala/com/twitter/snowflake/id", + "src/scala/com/twitter/simclusters_v2/common", + "src/thrift/com/twitter/core_workflows/user_model:user_model-scala", + ], +) diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/blender/BlendedCandidatesBuilder.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/blender/BlendedCandidatesBuilder.scala new file mode 100644 index 0000000000..1a864a6c2d --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/blender/BlendedCandidatesBuilder.scala @@ -0,0 +1,48 @@ +package com.twitter.cr_mixer.blender + +import com.twitter.cr_mixer.model.BlendedCandidate +import com.twitter.cr_mixer.model.CandidateGenerationInfo +import com.twitter.cr_mixer.model.InitialCandidate +import com.twitter.simclusters_v2.common.TweetId +import scala.collection.mutable + +object BlendedCandidatesBuilder { + + /** + * @param inputCandidates input candidate prior to interleaving + * @param interleavedCandidates after interleaving. These tweets are de-duplicated. + */ + def build( + inputCandidates: Seq[Seq[InitialCandidate]], + interleavedCandidates: Seq[InitialCandidate] + ): Seq[BlendedCandidate] = { + val cgInfoLookupMap = buildCandidateToCGInfosMap(inputCandidates) + interleavedCandidates.map { interleavedCandidate => + interleavedCandidate.toBlendedCandidate(cgInfoLookupMap(interleavedCandidate.tweetId)) + } + } + + /** + * The same tweet can be generated by different sources. + * This function tells you which CandidateGenerationInfo generated a given tweet + */ + private def buildCandidateToCGInfosMap( + candidateSeq: Seq[Seq[InitialCandidate]], + ): Map[TweetId, Seq[CandidateGenerationInfo]] = { + val tweetIdMap = mutable.HashMap[TweetId, Seq[CandidateGenerationInfo]]() + + candidateSeq.foreach { candidates => + candidates.foreach { candidate => + val candidateGenerationInfoSeq = { + tweetIdMap.getOrElse(candidate.tweetId, Seq.empty) + } + val candidateGenerationInfo = candidate.candidateGenerationInfo + tweetIdMap.put( + candidate.tweetId, + candidateGenerationInfoSeq ++ Seq(candidateGenerationInfo)) + } + } + tweetIdMap.toMap + } + +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/blender/ContentSignalBlender.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/blender/ContentSignalBlender.scala new file mode 100644 index 0000000000..9ef81009b4 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/blender/ContentSignalBlender.scala @@ -0,0 +1,121 @@ +package com.twitter.cr_mixer.blender + +import com.twitter.cr_mixer.model.BlendedCandidate +import com.twitter.cr_mixer.model.InitialCandidate +import com.twitter.cr_mixer.param.BlenderParams +import com.twitter.cr_mixer.thriftscala.SimilarityEngineType +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.snowflake.id.SnowflakeId +import com.twitter.timelines.configapi.Params +import com.twitter.util.Future +import com.twitter.util.Time +import javax.inject.Inject + +case class ContentSignalBlender @Inject() (globalStats: StatsReceiver) { + + private val name: String = this.getClass.getCanonicalName + private val stats: StatsReceiver = globalStats.scope(name) + + /** + * Exposes multiple types of sorting relying only on Content Based signals + * Candidate Recency, Random, FavoriteCount and finally Standardized, which standardizes the scores + * that come from the active SimilarityEngine and then sort on the standardized scores. + */ + def blend( + params: Params, + inputCandidates: Seq[Seq[InitialCandidate]], + ): Future[Seq[BlendedCandidate]] = { + // Filter out empty candidate sequence + val candidates = inputCandidates.filter(_.nonEmpty) + val sortedCandidates = params(BlenderParams.ContentBlenderTypeSortingAlgorithmParam) match { + case BlenderParams.ContentBasedSortingAlgorithmEnum.CandidateRecency => + candidates.flatten.sortBy(c => getSnowflakeTimeStamp(c.tweetId)).reverse + case BlenderParams.ContentBasedSortingAlgorithmEnum.RandomSorting => + candidates.flatten.sortBy(_ => scala.util.Random.nextDouble()) + case BlenderParams.ContentBasedSortingAlgorithmEnum.FavoriteCount => + candidates.flatten.sortBy(-_.tweetInfo.favCount) + case BlenderParams.ContentBasedSortingAlgorithmEnum.SimilarityToSignalSorting => + standardizeAndSortByScore(flattenAndGroupByEngineTypeOrFirstContribEngine(candidates)) + case _ => + candidates.flatten.sortBy(-_.tweetInfo.favCount) + } + + stats.stat("candidates").add(sortedCandidates.size) + + val blendedCandidates = + BlendedCandidatesBuilder.build(inputCandidates, removeDuplicates(sortedCandidates)) + Future.value(blendedCandidates) + } + + private def removeDuplicates(candidates: Seq[InitialCandidate]): Seq[InitialCandidate] = { + val seen = collection.mutable.Set.empty[Long] + candidates.filter { c => + if (seen.contains(c.tweetId)) { + false + } else { + seen += c.tweetId + true + } + } + } + + private def groupByEngineTypeOrFirstContribEngine( + candidates: Seq[InitialCandidate] + ): Map[SimilarityEngineType, Seq[InitialCandidate]] = { + val grouped = candidates.groupBy { candidate => + val contrib = candidate.candidateGenerationInfo.contributingSimilarityEngines + if (contrib.nonEmpty) { + contrib.head.similarityEngineType + } else { + candidate.candidateGenerationInfo.similarityEngineInfo.similarityEngineType + } + } + grouped + } + + private def flattenAndGroupByEngineTypeOrFirstContribEngine( + candidates: Seq[Seq[InitialCandidate]] + ): Seq[Seq[InitialCandidate]] = { + val flat = candidates.flatten + val grouped = groupByEngineTypeOrFirstContribEngine(flat) + grouped.values.toSeq + } + + private def standardizeAndSortByScore( + candidates: Seq[Seq[InitialCandidate]] + ): Seq[InitialCandidate] = { + candidates + .map { innerSeq => + val meanScore = innerSeq + .map(c => c.candidateGenerationInfo.similarityEngineInfo.score.getOrElse(0.0)) + .sum / innerSeq.length + val stdDev = scala.math + .sqrt( + innerSeq + .map(c => c.candidateGenerationInfo.similarityEngineInfo.score.getOrElse(0.0)) + .map(a => a - meanScore) + .map(a => a * a) + .sum / innerSeq.length) + innerSeq + .map(c => + ( + c, + c.candidateGenerationInfo.similarityEngineInfo.score + .map { score => + if (stdDev != 0) (score - meanScore) / stdDev + else 0.0 + } + .getOrElse(0.0))) + }.flatten.sortBy { case (_, standardizedScore) => -standardizedScore } + .map { case (candidate, _) => candidate } + } + + private def getSnowflakeTimeStamp(tweetId: Long): Time = { + val isSnowflake = SnowflakeId.isSnowflakeId(tweetId) + if (isSnowflake) { + SnowflakeId(tweetId).time + } else { + Time.fromMilliseconds(0L) + } + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/blender/CountWeightedInterleaveBlender.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/blender/CountWeightedInterleaveBlender.scala new file mode 100644 index 0000000000..4c5dd07c38 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/blender/CountWeightedInterleaveBlender.scala @@ -0,0 +1,90 @@ +package com.twitter.cr_mixer.blender + +import com.twitter.cr_mixer.model.BlendedCandidate +import com.twitter.cr_mixer.model.CrCandidateGeneratorQuery +import com.twitter.cr_mixer.model.InitialCandidate +import com.twitter.cr_mixer.param.BlenderParams +import com.twitter.cr_mixer.util.CountWeightedInterleaveUtil +import com.twitter.cr_mixer.util.InterleaveUtil +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.timelines.configapi.Params +import com.twitter.util.Future +import javax.inject.Inject +import javax.inject.Singleton + +/** + * A weighted round robin interleaving algorithm. + * The weight of each blending group based on the count of candidates in each blending group. + * The more candidates under a blending group, the more candidates are selected from it during round + * robin, which in effect prioritizes this group. + * + * Weights sum up to 1. For example: + * total candidates = 8 + * Group Weight + * [A1, A2, A3, A4] 4/8 = 0.5 // select 50% of results from group A + * [B1, B2] 2/8 = 0.25 // 25% from group B + * [C1, C2] 2/8 = 0.25 // 25% from group C + * + * Blended results = [A1, A2, B1, C1, A3, A4, B2, C2] + * See @linht's go/weighted-interleave + */ +@Singleton +case class CountWeightedInterleaveBlender @Inject() (globalStats: StatsReceiver) { + import CountWeightedInterleaveBlender._ + + private val name: String = this.getClass.getCanonicalName + private val stats: StatsReceiver = globalStats.scope(name) + + def blend( + query: CrCandidateGeneratorQuery, + inputCandidates: Seq[Seq[InitialCandidate]] + ): Future[Seq[BlendedCandidate]] = { + val weightedBlenderQuery = CountWeightedInterleaveBlender.paramToQuery(query.params) + countWeightedInterleave(weightedBlenderQuery, inputCandidates) + } + + private[blender] def countWeightedInterleave( + query: WeightedBlenderQuery, + inputCandidates: Seq[Seq[InitialCandidate]], + ): Future[Seq[BlendedCandidate]] = { + + val candidatesAndWeightKeyByIndexId: Seq[(Seq[InitialCandidate], Double)] = { + CountWeightedInterleaveUtil.buildInitialCandidatesWithWeightKeyByFeature( + inputCandidates, + query.rankerWeightShrinkage) + } + + val interleavedCandidates = + InterleaveUtil.weightedInterleave(candidatesAndWeightKeyByIndexId, query.maxWeightAdjustments) + + stats.stat("candidates").add(interleavedCandidates.size) + + val blendedCandidates = BlendedCandidatesBuilder.build(inputCandidates, interleavedCandidates) + Future.value(blendedCandidates) + } +} + +object CountWeightedInterleaveBlender { + + /** + * We pass two parameters to the weighted interleaver: + * @param rankerWeightShrinkage shrinkage parameter between [0, 1] that determines how close we + * stay to uniform sampling. The bigger the shrinkage the + * closer we are to uniform round robin + * @param maxWeightAdjustments max number of weighted sampling to do prior to defaulting to + * uniform. Set so that we avoid infinite loops (e.g. if weights are + * 0) + */ + case class WeightedBlenderQuery( + rankerWeightShrinkage: Double, + maxWeightAdjustments: Int) + + def paramToQuery(params: Params): WeightedBlenderQuery = { + val rankerWeightShrinkage: Double = + params(BlenderParams.RankingInterleaveWeightShrinkageParam) + val maxWeightAdjustments: Int = + params(BlenderParams.RankingInterleaveMaxWeightAdjustments) + + WeightedBlenderQuery(rankerWeightShrinkage, maxWeightAdjustments) + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/blender/InterleaveBlender.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/blender/InterleaveBlender.scala new file mode 100644 index 0000000000..92cdfe0922 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/blender/InterleaveBlender.scala @@ -0,0 +1,33 @@ +package com.twitter.cr_mixer.blender + +import com.twitter.cr_mixer.model.BlendedCandidate +import com.twitter.cr_mixer.model.InitialCandidate +import com.twitter.cr_mixer.util.InterleaveUtil +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.util.Future +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +case class InterleaveBlender @Inject() (globalStats: StatsReceiver) { + + private val name: String = this.getClass.getCanonicalName + private val stats: StatsReceiver = globalStats.scope(name) + + /** + * Interleaves candidates, by taking 1 candidate from each Seq[Seq[InitialCandidate]] in sequence, + * until we run out of candidates. + */ + def blend( + inputCandidates: Seq[Seq[InitialCandidate]], + ): Future[Seq[BlendedCandidate]] = { + + val interleavedCandidates = InterleaveUtil.interleave(inputCandidates) + + stats.stat("candidates").add(interleavedCandidates.size) + + val blendedCandidates = BlendedCandidatesBuilder.build(inputCandidates, interleavedCandidates) + Future.value(blendedCandidates) + } + +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/blender/SourceTypeBackFillBlender.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/blender/SourceTypeBackFillBlender.scala new file mode 100644 index 0000000000..14e93d53d9 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/blender/SourceTypeBackFillBlender.scala @@ -0,0 +1,64 @@ +package com.twitter.cr_mixer.blender + +import com.twitter.cr_mixer.blender.ImplicitSignalBackFillBlender.BackFillSourceTypes +import com.twitter.cr_mixer.blender.ImplicitSignalBackFillBlender.BackFillSourceTypesWithVideo +import com.twitter.cr_mixer.model.BlendedCandidate +import com.twitter.cr_mixer.model.InitialCandidate +import com.twitter.cr_mixer.param.BlenderParams +import com.twitter.cr_mixer.thriftscala.SourceType +import com.twitter.cr_mixer.util.InterleaveUtil +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.timelines.configapi.Params +import com.twitter.util.Future +import javax.inject.Inject + +case class SourceTypeBackFillBlender @Inject() (globalStats: StatsReceiver) { + + private val name: String = this.getClass.getCanonicalName + private val stats: StatsReceiver = globalStats.scope(name) + + /** + * Partition the candidates based on source type + * Interleave the two partitions of candidates separately + * Then append the back fill candidates to the end + */ + def blend( + params: Params, + inputCandidates: Seq[Seq[InitialCandidate]], + ): Future[Seq[BlendedCandidate]] = { + + // Filter out empty candidate sequence + val candidates = inputCandidates.filter(_.nonEmpty) + + val backFillSourceTypes = + if (params(BlenderParams.SourceTypeBackFillEnableVideoBackFill)) BackFillSourceTypesWithVideo + else BackFillSourceTypes + // partition candidates based on their source types + val (backFillCandidates, regularCandidates) = + candidates.partition( + _.head.candidateGenerationInfo.sourceInfoOpt + .exists(sourceInfo => backFillSourceTypes.contains(sourceInfo.sourceType))) + + val interleavedRegularCandidates = InterleaveUtil.interleave(regularCandidates) + val interleavedBackFillCandidates = + InterleaveUtil.interleave(backFillCandidates) + stats.stat("backFillCandidates").add(interleavedBackFillCandidates.size) + // Append interleaved backfill candidates to the end + val interleavedCandidates = interleavedRegularCandidates ++ interleavedBackFillCandidates + + stats.stat("candidates").add(interleavedCandidates.size) + + val blendedCandidates = BlendedCandidatesBuilder.build(inputCandidates, interleavedCandidates) + Future.value(blendedCandidates) + } + +} + +object ImplicitSignalBackFillBlender { + final val BackFillSourceTypesWithVideo: Set[SourceType] = Set( + SourceType.UserRepeatedProfileVisit, + SourceType.VideoTweetPlayback50, + SourceType.VideoTweetQualityView) + + final val BackFillSourceTypes: Set[SourceType] = Set(SourceType.UserRepeatedProfileVisit) +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/blender/SwitchBlender.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/blender/SwitchBlender.scala new file mode 100644 index 0000000000..7052a71a57 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/blender/SwitchBlender.scala @@ -0,0 +1,81 @@ +package com.twitter.cr_mixer.blender + +import com.twitter.core_workflows.user_model.thriftscala.UserState +import com.twitter.cr_mixer.model.BlendedCandidate +import com.twitter.cr_mixer.model.InitialCandidate +import com.twitter.cr_mixer.param.BlenderParams +import com.twitter.cr_mixer.param.BlenderParams.BlendingAlgorithmEnum +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.timelines.configapi.Params +import com.twitter.util.Future +import com.twitter.util.Time +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +case class SwitchBlender @Inject() ( + defaultBlender: InterleaveBlender, + sourceTypeBackFillBlender: SourceTypeBackFillBlender, + adsBlender: AdsBlender, + contentSignalBlender: ContentSignalBlender, + globalStats: StatsReceiver) { + + private val stats = globalStats.scope(this.getClass.getCanonicalName) + + def blend( + params: Params, + userState: UserState, + inputCandidates: Seq[Seq[InitialCandidate]], + ): Future[Seq[BlendedCandidate]] = { + // Take out empty seq + val nonEmptyCandidates = inputCandidates.collect { + case candidates if candidates.nonEmpty => + candidates + } + stats.stat("num_of_sequences").add(inputCandidates.size) + + // Sort the seqs in an order + val innerSignalSorting = params(BlenderParams.SignalTypeSortingAlgorithmParam) match { + case BlenderParams.ContentBasedSortingAlgorithmEnum.SourceSignalRecency => + SwitchBlender.TimestampOrder + case BlenderParams.ContentBasedSortingAlgorithmEnum.RandomSorting => SwitchBlender.RandomOrder + case _ => SwitchBlender.TimestampOrder + } + + val candidatesToBlend = nonEmptyCandidates.sortBy(_.head)(innerSignalSorting) + // Blend based on specified blender rules + params(BlenderParams.BlendingAlgorithmParam) match { + case BlendingAlgorithmEnum.RoundRobin => + defaultBlender.blend(candidatesToBlend) + case BlendingAlgorithmEnum.SourceTypeBackFill => + sourceTypeBackFillBlender.blend(params, candidatesToBlend) + case BlendingAlgorithmEnum.SourceSignalSorting => + contentSignalBlender.blend(params, candidatesToBlend) + case _ => defaultBlender.blend(candidatesToBlend) + } + } +} + +object SwitchBlender { + + /** + * Prefers candidates generated from sources with the latest timestamps. + * The newer the source signal, the higher a candidate ranks. + * This ordering biases against consumer-based candidates because their timestamp defaults to 0 + * + * Within a Seq[Seq[Candidate]], all candidates within a inner Seq + * are guaranteed to have the same sourceInfo because they are grouped by (sourceInfo, SE model). + * Hence, we can pick .headOption to represent the whole list when filtering by the internalId of the sourceInfoOpt. + * But of course the similarityEngine score in a CGInfo could be different. + */ + val TimestampOrder: Ordering[InitialCandidate] = + math.Ordering + .by[InitialCandidate, Time]( + _.candidateGenerationInfo.sourceInfoOpt + .flatMap(_.sourceEventTime) + .getOrElse(Time.fromMilliseconds(0L))) + .reverse + + private val RandomOrder: Ordering[InitialCandidate] = + Ordering.by[InitialCandidate, Double](_ => scala.util.Random.nextDouble()) +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/candidate_generation/AdsCandidateGenerator.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/candidate_generation/AdsCandidateGenerator.scala new file mode 100644 index 0000000000..e240ebf2dd --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/candidate_generation/AdsCandidateGenerator.scala @@ -0,0 +1,140 @@ +package com.twitter.cr_mixer.candidate_generation + +import com.twitter.cr_mixer.blender.AdsBlender +import com.twitter.cr_mixer.logging.AdsRecommendationsScribeLogger +import com.twitter.cr_mixer.model.AdsCandidateGeneratorQuery +import com.twitter.cr_mixer.model.BlendedAdsCandidate +import com.twitter.cr_mixer.model.InitialAdsCandidate +import com.twitter.cr_mixer.model.RankedAdsCandidate +import com.twitter.cr_mixer.model.SourceInfo +import com.twitter.cr_mixer.param.AdsParams +import com.twitter.cr_mixer.param.ConsumersBasedUserAdGraphParams +import com.twitter.cr_mixer.source_signal.RealGraphInSourceGraphFetcher +import com.twitter.cr_mixer.source_signal.SourceFetcher.FetcherQuery +import com.twitter.cr_mixer.source_signal.UssSourceSignalFetcher +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.frigate.common.util.StatsUtil +import com.twitter.simclusters_v2.common.UserId +import com.twitter.util.Future + +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class AdsCandidateGenerator @Inject() ( + ussSourceSignalFetcher: UssSourceSignalFetcher, + realGraphInSourceGraphFetcher: RealGraphInSourceGraphFetcher, + adsCandidateSourceRouter: AdsCandidateSourcesRouter, + adsBlender: AdsBlender, + scribeLogger: AdsRecommendationsScribeLogger, + globalStats: StatsReceiver) { + + private val stats: StatsReceiver = globalStats.scope(this.getClass.getCanonicalName) + private val fetchSourcesStats = stats.scope("fetchSources") + private val fetchRealGraphSeedsStats = stats.scope("fetchRealGraphSeeds") + private val fetchCandidatesStats = stats.scope("fetchCandidates") + private val interleaveStats = stats.scope("interleave") + private val rankStats = stats.scope("rank") + + def get(query: AdsCandidateGeneratorQuery): Future[Seq[RankedAdsCandidate]] = { + val allStats = stats.scope("all") + val perProductStats = stats.scope("perProduct", query.product.toString) + + StatsUtil.trackItemsStats(allStats) { + StatsUtil.trackItemsStats(perProductStats) { + for { + // fetch source signals + sourceSignals <- StatsUtil.trackBlockStats(fetchSourcesStats) { + fetchSources(query) + } + realGraphSeeds <- StatsUtil.trackItemMapStats(fetchRealGraphSeedsStats) { + fetchSeeds(query) + } + // get initial candidates from similarity engines + // hydrate lineItemInfo and filter out non active ads + initialCandidates <- StatsUtil.trackBlockStats(fetchCandidatesStats) { + fetchCandidates(query, sourceSignals, realGraphSeeds) + } + + // blend candidates + blendedCandidates <- StatsUtil.trackItemsStats(interleaveStats) { + interleave(initialCandidates) + } + + rankedCandidates <- StatsUtil.trackItemsStats(rankStats) { + rank( + blendedCandidates, + query.params(AdsParams.EnableScoreBoost), + query.params(AdsParams.AdsCandidateGenerationScoreBoostFactor), + rankStats) + } + } yield { + rankedCandidates.take(query.maxNumResults) + } + } + } + + } + + def fetchSources( + query: AdsCandidateGeneratorQuery + ): Future[Set[SourceInfo]] = { + val fetcherQuery = + FetcherQuery(query.userId, query.product, query.userState, query.params) + ussSourceSignalFetcher.get(fetcherQuery).map(_.getOrElse(Seq.empty).toSet) + } + + private def fetchCandidates( + query: AdsCandidateGeneratorQuery, + sourceSignals: Set[SourceInfo], + realGraphSeeds: Map[UserId, Double] + ): Future[Seq[Seq[InitialAdsCandidate]]] = { + scribeLogger.scribeInitialAdsCandidates( + query, + adsCandidateSourceRouter + .fetchCandidates(query.userId, sourceSignals, realGraphSeeds, query.params), + query.params(AdsParams.EnableScribe) + ) + + } + + private def fetchSeeds( + query: AdsCandidateGeneratorQuery + ): Future[Map[UserId, Double]] = { + if (query.params(ConsumersBasedUserAdGraphParams.EnableSourceParam)) { + realGraphInSourceGraphFetcher + .get(FetcherQuery(query.userId, query.product, query.userState, query.params)) + .map(_.map(_.seedWithScores).getOrElse(Map.empty)) + } else Future.value(Map.empty[UserId, Double]) + } + + private def interleave( + candidates: Seq[Seq[InitialAdsCandidate]] + ): Future[Seq[BlendedAdsCandidate]] = { + adsBlender + .blend(candidates) + } + + private def rank( + candidates: Seq[BlendedAdsCandidate], + enableScoreBoost: Boolean, + scoreBoostFactor: Double, + statsReceiver: StatsReceiver, + ): Future[Seq[RankedAdsCandidate]] = { + + val candidateSize = candidates.size + val rankedCandidates = candidates.zipWithIndex.map { + case (candidate, index) => + val score = 0.5 + 0.5 * ((candidateSize - index).toDouble / candidateSize) + val boostedScore = if (enableScoreBoost) { + statsReceiver.stat("boostedScore").add((100.0 * score * scoreBoostFactor).toFloat) + score * scoreBoostFactor + } else { + statsReceiver.stat("score").add((100.0 * score).toFloat) + score + } + candidate.toRankedAdsCandidate(boostedScore) + } + Future.value(rankedCandidates) + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/candidate_generation/AdsCandidateSourcesRouter.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/candidate_generation/AdsCandidateSourcesRouter.scala new file mode 100644 index 0000000000..69ef31b74c --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/candidate_generation/AdsCandidateSourcesRouter.scala @@ -0,0 +1,516 @@ + package com.twitter.cr_mixer.candidate_generation + +import com.twitter.cr_mixer.model.CandidateGenerationInfo +import com.twitter.cr_mixer.model.InitialAdsCandidate +import com.twitter.cr_mixer.model.ModelConfig +import com.twitter.cr_mixer.model.ModuleNames +import com.twitter.cr_mixer.model.SimilarityEngineInfo +import com.twitter.cr_mixer.model.SourceInfo +import com.twitter.cr_mixer.model.TweetWithCandidateGenerationInfo +import com.twitter.cr_mixer.model.TweetWithScore +import com.twitter.cr_mixer.param.ConsumersBasedUserAdGraphParams +import com.twitter.cr_mixer.param.ConsumerBasedWalsParams +import com.twitter.cr_mixer.param.ConsumerEmbeddingBasedCandidateGenerationParams +import com.twitter.cr_mixer.param.GlobalParams +import com.twitter.cr_mixer.param.InterestedInParams +import com.twitter.cr_mixer.param.ProducerBasedCandidateGenerationParams +import com.twitter.cr_mixer.param.SimClustersANNParams +import com.twitter.cr_mixer.param.TweetBasedCandidateGenerationParams +import com.twitter.cr_mixer.param.decider.CrMixerDecider +import com.twitter.cr_mixer.param.decider.DeciderConstants +import com.twitter.cr_mixer.similarity_engine.ConsumerBasedWalsSimilarityEngine +import com.twitter.cr_mixer.similarity_engine.ConsumersBasedUserAdGraphSimilarityEngine +import com.twitter.cr_mixer.similarity_engine.FilterUtil +import com.twitter.cr_mixer.similarity_engine.HnswANNEngineQuery +import com.twitter.cr_mixer.similarity_engine.HnswANNSimilarityEngine +import com.twitter.cr_mixer.similarity_engine.ProducerBasedUserAdGraphSimilarityEngine +import com.twitter.cr_mixer.similarity_engine.SimClustersANNSimilarityEngine +import com.twitter.cr_mixer.similarity_engine.SimClustersANNSimilarityEngine.Query +import com.twitter.cr_mixer.similarity_engine.StandardSimilarityEngine +import com.twitter.cr_mixer.similarity_engine.TweetBasedUserAdGraphSimilarityEngine +import com.twitter.cr_mixer.thriftscala.LineItemInfo +import com.twitter.cr_mixer.thriftscala.SimilarityEngineType +import com.twitter.cr_mixer.thriftscala.SourceType +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.simclusters_v2.common.ModelVersions +import com.twitter.simclusters_v2.common.TweetId +import com.twitter.simclusters_v2.common.UserId +import com.twitter.simclusters_v2.thriftscala.EmbeddingType +import com.twitter.simclusters_v2.thriftscala.InternalId +import com.twitter.storehaus.ReadableStore +import com.twitter.timelines.configapi +import com.twitter.timelines.configapi.Params +import com.twitter.util.Future + +import javax.inject.Inject +import javax.inject.Named +import javax.inject.Singleton + +@Singleton +case class AdsCandidateSourcesRouter @Inject() ( + activePromotedTweetStore: ReadableStore[TweetId, Seq[LineItemInfo]], + decider: CrMixerDecider, + @Named(ModuleNames.SimClustersANNSimilarityEngine) simClustersANNSimilarityEngine: StandardSimilarityEngine[ + Query, + TweetWithScore + ], + @Named(ModuleNames.TweetBasedUserAdGraphSimilarityEngine) + tweetBasedUserAdGraphSimilarityEngine: StandardSimilarityEngine[ + TweetBasedUserAdGraphSimilarityEngine.Query, + TweetWithScore + ], + @Named(ModuleNames.ConsumersBasedUserAdGraphSimilarityEngine) + consumersBasedUserAdGraphSimilarityEngine: StandardSimilarityEngine[ + ConsumersBasedUserAdGraphSimilarityEngine.Query, + TweetWithScore + ], + @Named(ModuleNames.ProducerBasedUserAdGraphSimilarityEngine) + producerBasedUserAdGraphSimilarityEngine: StandardSimilarityEngine[ + ProducerBasedUserAdGraphSimilarityEngine.Query, + TweetWithScore + ], + @Named(ModuleNames.TweetBasedTwHINANNSimilarityEngine) + tweetBasedTwHINANNSimilarityEngine: HnswANNSimilarityEngine, + @Named(ModuleNames.ConsumerEmbeddingBasedTwHINANNSimilarityEngine) consumerTwHINANNSimilarityEngine: HnswANNSimilarityEngine, + @Named(ModuleNames.ConsumerBasedWalsSimilarityEngine) + consumerBasedWalsSimilarityEngine: StandardSimilarityEngine[ + ConsumerBasedWalsSimilarityEngine.Query, + TweetWithScore + ], + globalStats: StatsReceiver, +) { + + import AdsCandidateSourcesRouter._ + + val stats: StatsReceiver = globalStats.scope(this.getClass.getSimpleName) + + def fetchCandidates( + requestUserId: UserId, + sourceSignals: Set[SourceInfo], + realGraphSeeds: Map[UserId, Double], + params: configapi.Params + ): Future[Seq[Seq[InitialAdsCandidate]]] = { + + val simClustersANN1ConfigId = params(SimClustersANNParams.SimClustersANN1ConfigId) + + val tweetBasedSANNMinScore = params( + TweetBasedCandidateGenerationParams.SimClustersMinScoreParam) + val tweetBasedSANN1Candidates = + if (params(TweetBasedCandidateGenerationParams.EnableSimClustersANN1Param)) { + Future.collect( + CandidateSourcesRouter.getTweetBasedSourceInfo(sourceSignals).toSeq.map { sourceInfo => + getSimClustersANNCandidates( + requestUserId, + Some(sourceInfo), + params, + simClustersANN1ConfigId, + tweetBasedSANNMinScore) + }) + } else Future.value(Seq.empty) + + val simClustersANN2ConfigId = params(SimClustersANNParams.SimClustersANN2ConfigId) + val tweetBasedSANN2Candidates = + if (params(TweetBasedCandidateGenerationParams.EnableSimClustersANN2Param)) { + Future.collect( + CandidateSourcesRouter.getTweetBasedSourceInfo(sourceSignals).toSeq.map { sourceInfo => + getSimClustersANNCandidates( + requestUserId, + Some(sourceInfo), + params, + simClustersANN2ConfigId, + tweetBasedSANNMinScore) + }) + } else Future.value(Seq.empty) + + val tweetBasedUagCandidates = + if (params(TweetBasedCandidateGenerationParams.EnableUAGParam)) { + Future.collect( + CandidateSourcesRouter.getTweetBasedSourceInfo(sourceSignals).toSeq.map { sourceInfo => + getTweetBasedUserAdGraphCandidates(Some(sourceInfo), params) + }) + } else Future.value(Seq.empty) + + val realGraphInNetworkBasedUagCandidates = + if (params(ConsumersBasedUserAdGraphParams.EnableSourceParam)) { + getRealGraphConsumersBasedUserAdGraphCandidates(realGraphSeeds, params).map(Seq(_)) + } else Future.value(Seq.empty) + + val producerBasedUagCandidates = + if (params(ProducerBasedCandidateGenerationParams.EnableUAGParam)) { + Future.collect( + CandidateSourcesRouter.getProducerBasedSourceInfo(sourceSignals).toSeq.map { sourceInfo => + getProducerBasedUserAdGraphCandidates(Some(sourceInfo), params) + }) + } else Future.value(Seq.empty) + + val tweetBasedTwhinAdsCandidates = + if (params(TweetBasedCandidateGenerationParams.EnableTwHINParam)) { + Future.collect( + CandidateSourcesRouter.getTweetBasedSourceInfo(sourceSignals).toSeq.map { sourceInfo => + getTwHINAdsCandidates( + tweetBasedTwHINANNSimilarityEngine, + SimilarityEngineType.TweetBasedTwHINANN, + requestUserId, + Some(sourceInfo), + ModelConfig.DebuggerDemo) + }) + } else Future.value(Seq.empty) + + val producerBasedSANNMinScore = params( + ProducerBasedCandidateGenerationParams.SimClustersMinScoreParam) + val producerBasedSANN1Candidates = + if (params(ProducerBasedCandidateGenerationParams.EnableSimClustersANN1Param)) { + Future.collect( + CandidateSourcesRouter.getProducerBasedSourceInfo(sourceSignals).toSeq.map { sourceInfo => + getSimClustersANNCandidates( + requestUserId, + Some(sourceInfo), + params, + simClustersANN1ConfigId, + producerBasedSANNMinScore) + }) + } else Future.value(Seq.empty) + val producerBasedSANN2Candidates = + if (params(ProducerBasedCandidateGenerationParams.EnableSimClustersANN2Param)) { + Future.collect( + CandidateSourcesRouter.getProducerBasedSourceInfo(sourceSignals).toSeq.map { sourceInfo => + getSimClustersANNCandidates( + requestUserId, + Some(sourceInfo), + params, + simClustersANN2ConfigId, + producerBasedSANNMinScore) + }) + } else Future.value(Seq.empty) + + val interestedInMinScore = params(InterestedInParams.MinScoreParam) + val interestedInSANN1Candidates = if (params(InterestedInParams.EnableSimClustersANN1Param)) { + getSimClustersANNCandidates( + requestUserId, + None, + params, + simClustersANN1ConfigId, + interestedInMinScore).map(Seq(_)) + } else Future.value(Seq.empty) + + val interestedInSANN2Candidates = if (params(InterestedInParams.EnableSimClustersANN2Param)) { + getSimClustersANNCandidates( + requestUserId, + None, + params, + simClustersANN2ConfigId, + interestedInMinScore).map(Seq(_)) + } else Future.value(Seq.empty) + + val consumerTwHINAdsCandidates = + if (params(ConsumerEmbeddingBasedCandidateGenerationParams.EnableTwHINParam)) { + getTwHINAdsCandidates( + consumerTwHINANNSimilarityEngine, + SimilarityEngineType.ConsumerEmbeddingBasedTwHINANN, + requestUserId, + None, + ModelConfig.DebuggerDemo).map(Seq(_)) + } else Future.value(Seq.empty) + + val consumerBasedWalsCandidates = + if (params( + ConsumerBasedWalsParams.EnableSourceParam + )) { + getConsumerBasedWalsCandidates(sourceSignals, params) + }.map { + Seq(_) + } + else Future.value(Seq.empty) + + Future + .collect(Seq( + tweetBasedSANN1Candidates, + tweetBasedSANN2Candidates, + tweetBasedUagCandidates, + tweetBasedTwhinAdsCandidates, + producerBasedUagCandidates, + producerBasedSANN1Candidates, + producerBasedSANN2Candidates, + realGraphInNetworkBasedUagCandidates, + interestedInSANN1Candidates, + interestedInSANN2Candidates, + consumerTwHINAdsCandidates, + consumerBasedWalsCandidates, + )).map(_.flatten).map { tweetsWithCGInfoSeq => + Future.collect( + tweetsWithCGInfoSeq.map(candidates => convertToInitialCandidates(candidates, stats))) + }.flatten.map { candidatesLists => + val result = candidatesLists.filter(_.nonEmpty) + stats.stat("numOfSequences").add(result.size) + stats.stat("flattenCandidatesWithDup").add(result.flatten.size) + result + } + } + + private[candidate_generation] def convertToInitialCandidates( + candidates: Seq[TweetWithCandidateGenerationInfo], + stats: StatsReceiver + ): Future[Seq[InitialAdsCandidate]] = { + val tweetIds = candidates.map(_.tweetId).toSet + stats.stat("initialCandidateSizeBeforeLineItemFilter").add(tweetIds.size) + Future.collect(activePromotedTweetStore.multiGet(tweetIds)).map { lineItemInfos => + /** * + * If lineItemInfo does not exist, we will filter out the promoted tweet as it cannot be targeted and ranked in admixer + */ + val filteredCandidates = candidates.collect { + case candidate if lineItemInfos.getOrElse(candidate.tweetId, None).isDefined => + val lineItemInfo = lineItemInfos(candidate.tweetId) + .getOrElse(throw new IllegalStateException("Check previous line's condition")) + + InitialAdsCandidate( + tweetId = candidate.tweetId, + lineItemInfo = lineItemInfo, + candidate.candidateGenerationInfo + ) + } + stats.stat("initialCandidateSizeAfterLineItemFilter").add(filteredCandidates.size) + filteredCandidates + } + } + + private[candidate_generation] def getSimClustersANNCandidates( + requestUserId: UserId, + sourceInfo: Option[SourceInfo], + params: configapi.Params, + configId: String, + minScore: Double + ) = { + + val simClustersModelVersion = + ModelVersions.Enum.enumToSimClustersModelVersionMap(params(GlobalParams.ModelVersionParam)) + + val embeddingType = + if (sourceInfo.isEmpty) { + params(InterestedInParams.InterestedInEmbeddingIdParam).embeddingType + } else getSimClustersANNEmbeddingType(sourceInfo.get) + val query = SimClustersANNSimilarityEngine.fromParams( + if (sourceInfo.isEmpty) InternalId.UserId(requestUserId) else sourceInfo.get.internalId, + embeddingType, + simClustersModelVersion, + configId, + params + ) + + // dark traffic to simclusters-ann-2 + if (decider.isAvailable(DeciderConstants.enableSimClustersANN2DarkTrafficDeciderKey)) { + val simClustersANN2ConfigId = params(SimClustersANNParams.SimClustersANN2ConfigId) + val sann2Query = SimClustersANNSimilarityEngine.fromParams( + if (sourceInfo.isEmpty) InternalId.UserId(requestUserId) else sourceInfo.get.internalId, + embeddingType, + simClustersModelVersion, + simClustersANN2ConfigId, + params + ) + simClustersANNSimilarityEngine + .getCandidates(sann2Query) + } + + simClustersANNSimilarityEngine + .getCandidates(query).map(_.getOrElse(Seq.empty)).map(_.filter(_.score > minScore).map { + tweetWithScore => + val similarityEngineInfo = SimClustersANNSimilarityEngine + .toSimilarityEngineInfo(query, tweetWithScore.score) + TweetWithCandidateGenerationInfo( + tweetWithScore.tweetId, + CandidateGenerationInfo( + sourceInfo, + similarityEngineInfo, + Seq(similarityEngineInfo) + )) + }) + } + + private[candidate_generation] def getProducerBasedUserAdGraphCandidates( + sourceInfo: Option[SourceInfo], + params: configapi.Params + ) = { + + val query = ProducerBasedUserAdGraphSimilarityEngine.fromParams( + sourceInfo.get.internalId, + params + ) + producerBasedUserAdGraphSimilarityEngine + .getCandidates(query).map(_.getOrElse(Seq.empty)).map(_.map { tweetWithScore => + val similarityEngineInfo = ProducerBasedUserAdGraphSimilarityEngine + .toSimilarityEngineInfo(tweetWithScore.score) + TweetWithCandidateGenerationInfo( + tweetWithScore.tweetId, + CandidateGenerationInfo( + sourceInfo, + similarityEngineInfo, + Seq(similarityEngineInfo) + )) + }) + } + + private[candidate_generation] def getTweetBasedUserAdGraphCandidates( + sourceInfo: Option[SourceInfo], + params: configapi.Params + ) = { + + val query = TweetBasedUserAdGraphSimilarityEngine.fromParams( + sourceInfo.get.internalId, + params + ) + tweetBasedUserAdGraphSimilarityEngine + .getCandidates(query).map(_.getOrElse(Seq.empty)).map(_.map { tweetWithScore => + val similarityEngineInfo = TweetBasedUserAdGraphSimilarityEngine + .toSimilarityEngineInfo(tweetWithScore.score) + TweetWithCandidateGenerationInfo( + tweetWithScore.tweetId, + CandidateGenerationInfo( + sourceInfo, + similarityEngineInfo, + Seq(similarityEngineInfo) + )) + }) + } + + private[candidate_generation] def getRealGraphConsumersBasedUserAdGraphCandidates( + realGraphSeeds: Map[UserId, Double], + params: configapi.Params + ) = { + + val query = ConsumersBasedUserAdGraphSimilarityEngine + .fromParams(realGraphSeeds, params) + + // The internalId is a placeholder value. We do not plan to store the full seedUserId set. + val sourceInfo = SourceInfo( + sourceType = SourceType.RealGraphIn, + internalId = InternalId.UserId(0L), + sourceEventTime = None + ) + consumersBasedUserAdGraphSimilarityEngine + .getCandidates(query).map(_.getOrElse(Seq.empty)).map(_.map { tweetWithScore => + val similarityEngineInfo = ConsumersBasedUserAdGraphSimilarityEngine + .toSimilarityEngineInfo(tweetWithScore.score) + TweetWithCandidateGenerationInfo( + tweetWithScore.tweetId, + CandidateGenerationInfo( + Some(sourceInfo), + similarityEngineInfo, + Seq.empty // Atomic Similarity Engine. Hence it has no contributing SEs + ) + ) + }) + } + + private[candidate_generation] def getTwHINAdsCandidates( + similarityEngine: HnswANNSimilarityEngine, + similarityEngineType: SimilarityEngineType, + requestUserId: UserId, + sourceInfo: Option[SourceInfo], // if none, then it's consumer-based similarity engine + model: String + ): Future[Seq[TweetWithCandidateGenerationInfo]] = { + val internalId = + if (sourceInfo.nonEmpty) sourceInfo.get.internalId else InternalId.UserId(requestUserId) + similarityEngine + .getCandidates(buildHnswANNQuery(internalId, model)).map(_.getOrElse(Seq.empty)).map(_.map { + tweetWithScore => + val similarityEngineInfo = SimilarityEngineInfo( + similarityEngineType = similarityEngineType, + modelId = Some(model), + score = Some(tweetWithScore.score)) + TweetWithCandidateGenerationInfo( + tweetWithScore.tweetId, + CandidateGenerationInfo( + None, + similarityEngineInfo, + Seq(similarityEngineInfo) + )) + }) + } + + private[candidate_generation] def getConsumerBasedWalsCandidates( + sourceSignals: Set[SourceInfo], + params: configapi.Params + ): Future[Seq[TweetWithCandidateGenerationInfo]] = { + // Fetch source signals and filter them based on age. + val signals = FilterUtil.tweetSourceAgeFilter( + getConsumerBasedWalsSourceInfo(sourceSignals).toSeq, + params(ConsumerBasedWalsParams.MaxTweetSignalAgeHoursParam)) + + val candidatesOptFut = consumerBasedWalsSimilarityEngine.getCandidates( + ConsumerBasedWalsSimilarityEngine.fromParams(signals, params) + ) + val tweetsWithCandidateGenerationInfoOptFut = candidatesOptFut.map { + _.map { tweetsWithScores => + val sortedCandidates = tweetsWithScores.sortBy(-_.score) + val filteredCandidates = + FilterUtil.tweetAgeFilter(sortedCandidates, params(GlobalParams.MaxTweetAgeHoursParam)) + consumerBasedWalsSimilarityEngine.getScopedStats + .stat("filteredCandidates_size").add(filteredCandidates.size) + + val tweetsWithCandidateGenerationInfo = filteredCandidates.map { tweetWithScore => + { + val similarityEngineInfo = + ConsumerBasedWalsSimilarityEngine.toSimilarityEngineInfo(tweetWithScore.score) + TweetWithCandidateGenerationInfo( + tweetWithScore.tweetId, + CandidateGenerationInfo( + None, + similarityEngineInfo, + Seq.empty // Atomic Similarity Engine. Hence it has no contributing SEs + ) + ) + } + } + val maxCandidateNum = params(GlobalParams.MaxCandidateNumPerSourceKeyParam) + tweetsWithCandidateGenerationInfo.take(maxCandidateNum) + } + } + for { + tweetsWithCandidateGenerationInfoOpt <- tweetsWithCandidateGenerationInfoOptFut + } yield tweetsWithCandidateGenerationInfoOpt.toSeq.flatten + } +} + +object AdsCandidateSourcesRouter { + def getSimClustersANNEmbeddingType( + sourceInfo: SourceInfo + ): EmbeddingType = { + sourceInfo.sourceType match { + case SourceType.TweetFavorite | SourceType.Retweet | SourceType.OriginalTweet | + SourceType.Reply | SourceType.TweetShare | SourceType.NotificationClick | + SourceType.GoodTweetClick | SourceType.VideoTweetQualityView | + SourceType.VideoTweetPlayback50 => + EmbeddingType.LogFavLongestL2EmbeddingTweet + case SourceType.UserFollow | SourceType.UserRepeatedProfileVisit | SourceType.RealGraphOon | + SourceType.FollowRecommendation | SourceType.UserTrafficAttributionProfileVisit | + SourceType.GoodProfileClick | SourceType.TwiceUserId => + EmbeddingType.FavBasedProducer + case _ => throw new IllegalArgumentException("sourceInfo.sourceType not supported") + } + } + + def buildHnswANNQuery(internalId: InternalId, modelId: String): HnswANNEngineQuery = { + HnswANNEngineQuery( + sourceId = internalId, + modelId = modelId, + params = Params.Empty + ) + } + + def getConsumerBasedWalsSourceInfo( + sourceSignals: Set[SourceInfo] + ): Set[SourceInfo] = { + val AllowedSourceTypesForConsumerBasedWalsSE = Set( + SourceType.TweetFavorite.value, + SourceType.Retweet.value, + SourceType.TweetDontLike.value, //currently no-op + SourceType.TweetReport.value, //currently no-op + SourceType.AccountMute.value, //currently no-op + SourceType.AccountBlock.value //currently no-op + ) + sourceSignals.collect { + case sourceInfo + if AllowedSourceTypesForConsumerBasedWalsSE.contains(sourceInfo.sourceType.value) => + sourceInfo + } + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/candidate_generation/BUILD b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/candidate_generation/BUILD new file mode 100644 index 0000000000..f1b6e69804 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/candidate_generation/BUILD @@ -0,0 +1,51 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + strict_deps = True, + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/com/twitter/storehaus:core", + "3rdparty/jvm/javax/inject:javax.inject", + "3rdparty/src/jvm/com/twitter/storehaus:core", + "ann/src/main/scala/com/twitter/ann/hnsw", + "ann/src/main/thrift/com/twitter/ann/common:ann-common-scala", + "configapi/configapi-core", + "content-recommender/thrift/src/main/thrift:thrift-scala", + "cr-mixer/server/src/main/scala/com/twitter/cr_mixer/blender", + "cr-mixer/server/src/main/scala/com/twitter/cr_mixer/config", + "cr-mixer/server/src/main/scala/com/twitter/cr_mixer/filter", + "cr-mixer/server/src/main/scala/com/twitter/cr_mixer/logging", + "cr-mixer/server/src/main/scala/com/twitter/cr_mixer/model", + "cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param", + "cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/decider", + "cr-mixer/server/src/main/scala/com/twitter/cr_mixer/ranker", + "cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine", + "cr-mixer/server/src/main/scala/com/twitter/cr_mixer/source_signal", + "cr-mixer/server/src/main/scala/com/twitter/cr_mixer/util", + "cr-mixer/thrift/src/main/thrift:thrift-scala", + "cuad/projects/hashspace/thrift:thrift-scala", + "decider/src/main/scala", + "finatra/inject/inject-core/src/main/scala", + "follow-recommendations-service/thrift/src/main/thrift:thrift-scala", + "frigate/frigate-common:base", + "frigate/frigate-common/src/main/scala/com/twitter/frigate/common/base", + "frigate/frigate-common/src/main/scala/com/twitter/frigate/common/candidate", + "frigate/frigate-common/src/main/scala/com/twitter/frigate/common/util:stats_util", + "hermit/hermit-core/src/main/scala/com/twitter/hermit/constants", + "hermit/hermit-core/src/main/scala/com/twitter/hermit/model", + "simclusters-ann/thrift/src/main/thrift:thrift-scala", + "snowflake/src/main/scala/com/twitter/snowflake/id", + "src/scala/com/twitter/cortex/ml/embeddings/common:Helpers", + "src/scala/com/twitter/ml/featurestore/lib", + "src/scala/com/twitter/simclusters_v2/common", + "src/thrift/com/twitter/frigate/data_pipeline/scalding:blue_verified_annotations-scala", + "src/thrift/com/twitter/ml/api:embedding-scala", + "src/thrift/com/twitter/recos/user_tweet_graph:user_tweet_graph-scala", + "src/thrift/com/twitter/recos/user_tweet_graph_plus:user_tweet_graph_plus-scala", + "src/thrift/com/twitter/search:earlybird-scala", + "src/thrift/com/twitter/search/query_interaction_graph/service:qig-service-scala", + "src/thrift/com/twitter/simclusters_v2:simclusters_v2-thrift-scala", + "src/thrift/com/twitter/wtf/candidate:wtf-candidate-scala", + "strato/config/columns/cuad/hashspace:hashspace-strato-client", + ], +) diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/candidate_generation/CandidateSourcesRouter.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/candidate_generation/CandidateSourcesRouter.scala new file mode 100644 index 0000000000..49cc37bded --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/candidate_generation/CandidateSourcesRouter.scala @@ -0,0 +1,536 @@ +package com.twitter.cr_mixer.candidate_generation + +import com.twitter.contentrecommender.thriftscala.TweetInfo +import com.twitter.cr_mixer.model.CandidateGenerationInfo +import com.twitter.cr_mixer.model.GraphSourceInfo +import com.twitter.cr_mixer.model.InitialCandidate +import com.twitter.cr_mixer.model.ModelConfig +import com.twitter.cr_mixer.model.ModuleNames +import com.twitter.cr_mixer.model.SimilarityEngineInfo +import com.twitter.cr_mixer.model.SourceInfo +import com.twitter.cr_mixer.model.TripTweetWithScore +import com.twitter.cr_mixer.model.TweetWithCandidateGenerationInfo +import com.twitter.cr_mixer.model.TweetWithScore +import com.twitter.cr_mixer.model.TweetWithScoreAndSocialProof +import com.twitter.cr_mixer.param.ConsumerBasedWalsParams +import com.twitter.cr_mixer.param.ConsumerEmbeddingBasedCandidateGenerationParams +import com.twitter.cr_mixer.param.ConsumersBasedUserVideoGraphParams +import com.twitter.cr_mixer.param.GlobalParams +import com.twitter.cr_mixer.similarity_engine.ConsumersBasedUserVideoGraphSimilarityEngine +import com.twitter.cr_mixer.similarity_engine.ConsumerBasedWalsSimilarityEngine +import com.twitter.cr_mixer.similarity_engine.ConsumerEmbeddingBasedTripSimilarityEngine +import com.twitter.cr_mixer.similarity_engine.ConsumerEmbeddingBasedTwHINSimilarityEngine +import com.twitter.cr_mixer.similarity_engine.ConsumerEmbeddingBasedTwoTowerSimilarityEngine +import com.twitter.cr_mixer.similarity_engine.EngineQuery +import com.twitter.cr_mixer.similarity_engine.FilterUtil +import com.twitter.cr_mixer.similarity_engine.HnswANNEngineQuery +import com.twitter.cr_mixer.similarity_engine.HnswANNSimilarityEngine +import com.twitter.cr_mixer.similarity_engine.ProducerBasedUnifiedSimilarityEngine +import com.twitter.cr_mixer.similarity_engine.StandardSimilarityEngine +import com.twitter.cr_mixer.similarity_engine.TripEngineQuery +import com.twitter.cr_mixer.similarity_engine.TweetBasedUnifiedSimilarityEngine +import com.twitter.cr_mixer.similarity_engine.UserTweetEntityGraphSimilarityEngine +import com.twitter.cr_mixer.thriftscala.SimilarityEngineType +import com.twitter.cr_mixer.thriftscala.SourceType +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.simclusters_v2.common.TweetId +import com.twitter.simclusters_v2.common.UserId +import com.twitter.simclusters_v2.thriftscala.InternalId +import com.twitter.storehaus.ReadableStore +import com.twitter.timelines.configapi +import com.twitter.util.Future +import javax.inject.Inject +import javax.inject.Named +import javax.inject.Singleton + +/** + * Route the SourceInfo to the associated Candidate Engines. + */ +@Singleton +case class CandidateSourcesRouter @Inject() ( + customizedRetrievalCandidateGeneration: CustomizedRetrievalCandidateGeneration, + simClustersInterestedInCandidateGeneration: SimClustersInterestedInCandidateGeneration, + @Named(ModuleNames.TweetBasedUnifiedSimilarityEngine) + tweetBasedUnifiedSimilarityEngine: StandardSimilarityEngine[ + TweetBasedUnifiedSimilarityEngine.Query, + TweetWithCandidateGenerationInfo + ], + @Named(ModuleNames.ProducerBasedUnifiedSimilarityEngine) + producerBasedUnifiedSimilarityEngine: StandardSimilarityEngine[ + ProducerBasedUnifiedSimilarityEngine.Query, + TweetWithCandidateGenerationInfo + ], + @Named(ModuleNames.ConsumerEmbeddingBasedTripSimilarityEngine) + consumerEmbeddingBasedTripSimilarityEngine: StandardSimilarityEngine[ + TripEngineQuery, + TripTweetWithScore + ], + @Named(ModuleNames.ConsumerEmbeddingBasedTwHINANNSimilarityEngine) + consumerBasedTwHINANNSimilarityEngine: HnswANNSimilarityEngine, + @Named(ModuleNames.ConsumerEmbeddingBasedTwoTowerANNSimilarityEngine) + consumerBasedTwoTowerSimilarityEngine: HnswANNSimilarityEngine, + @Named(ModuleNames.ConsumersBasedUserVideoGraphSimilarityEngine) + consumersBasedUserVideoGraphSimilarityEngine: StandardSimilarityEngine[ + ConsumersBasedUserVideoGraphSimilarityEngine.Query, + TweetWithScore + ], + @Named(ModuleNames.UserTweetEntityGraphSimilarityEngine) userTweetEntityGraphSimilarityEngine: StandardSimilarityEngine[ + UserTweetEntityGraphSimilarityEngine.Query, + TweetWithScoreAndSocialProof + ], + @Named(ModuleNames.ConsumerBasedWalsSimilarityEngine) + consumerBasedWalsSimilarityEngine: StandardSimilarityEngine[ + ConsumerBasedWalsSimilarityEngine.Query, + TweetWithScore + ], + tweetInfoStore: ReadableStore[TweetId, TweetInfo], + globalStats: StatsReceiver, +) { + + import CandidateSourcesRouter._ + val stats: StatsReceiver = globalStats.scope(this.getClass.getSimpleName) + + def fetchCandidates( + requestUserId: UserId, + sourceSignals: Set[SourceInfo], + sourceGraphs: Map[String, Option[GraphSourceInfo]], + params: configapi.Params, + ): Future[Seq[Seq[InitialCandidate]]] = { + + val tweetBasedCandidatesFuture = getCandidates( + getTweetBasedSourceInfo(sourceSignals), + params, + TweetBasedUnifiedSimilarityEngine.fromParams, + tweetBasedUnifiedSimilarityEngine.getCandidates) + + val producerBasedCandidatesFuture = + getCandidates( + getProducerBasedSourceInfo(sourceSignals), + params, + ProducerBasedUnifiedSimilarityEngine.fromParams(_, _), + producerBasedUnifiedSimilarityEngine.getCandidates + ) + + val simClustersInterestedInBasedCandidatesFuture = + getCandidatesPerSimilarityEngineModel( + requestUserId, + params, + SimClustersInterestedInCandidateGeneration.fromParams, + simClustersInterestedInCandidateGeneration.get) + + val consumerEmbeddingBasedLogFavBasedTripCandidatesFuture = + if (params( + ConsumerEmbeddingBasedCandidateGenerationParams.EnableLogFavBasedSimClustersTripParam)) { + getSimClustersTripCandidates( + params, + ConsumerEmbeddingBasedTripSimilarityEngine.fromParams( + ModelConfig.ConsumerLogFavBasedInterestedInEmbedding, + InternalId.UserId(requestUserId), + params + ), + consumerEmbeddingBasedTripSimilarityEngine + ).map { + Seq(_) + } + } else + Future.Nil + + val consumersBasedUvgRealGraphInCandidatesFuture = + if (params(ConsumersBasedUserVideoGraphParams.EnableSourceParam)) { + val realGraphInGraphSourceInfoOpt = + getGraphSourceInfoBySourceType(SourceType.RealGraphIn.name, sourceGraphs) + + getGraphBasedCandidates( + params, + ConsumersBasedUserVideoGraphSimilarityEngine + .fromParamsForRealGraphIn( + realGraphInGraphSourceInfoOpt + .map { graphSourceInfo => graphSourceInfo.seedWithScores }.getOrElse(Map.empty), + params), + consumersBasedUserVideoGraphSimilarityEngine, + ConsumersBasedUserVideoGraphSimilarityEngine.toSimilarityEngineInfo, + realGraphInGraphSourceInfoOpt + ).map { + Seq(_) + } + } else Future.Nil + + val consumerEmbeddingBasedFollowBasedTripCandidatesFuture = + if (params( + ConsumerEmbeddingBasedCandidateGenerationParams.EnableFollowBasedSimClustersTripParam)) { + getSimClustersTripCandidates( + params, + ConsumerEmbeddingBasedTripSimilarityEngine.fromParams( + ModelConfig.ConsumerFollowBasedInterestedInEmbedding, + InternalId.UserId(requestUserId), + params + ), + consumerEmbeddingBasedTripSimilarityEngine + ).map { + Seq(_) + } + } else + Future.Nil + + val consumerBasedWalsCandidatesFuture = + if (params( + ConsumerBasedWalsParams.EnableSourceParam + )) { + getConsumerBasedWalsCandidates(sourceSignals, params) + }.map { Seq(_) } + else Future.Nil + + val consumerEmbeddingBasedTwHINCandidatesFuture = + if (params(ConsumerEmbeddingBasedCandidateGenerationParams.EnableTwHINParam)) { + getHnswCandidates( + params, + ConsumerEmbeddingBasedTwHINSimilarityEngine.fromParams( + InternalId.UserId(requestUserId), + params), + consumerBasedTwHINANNSimilarityEngine + ).map { Seq(_) } + } else Future.Nil + + val consumerEmbeddingBasedTwoTowerCandidatesFuture = + if (params(ConsumerEmbeddingBasedCandidateGenerationParams.EnableTwoTowerParam)) { + getHnswCandidates( + params, + ConsumerEmbeddingBasedTwoTowerSimilarityEngine.fromParams( + InternalId.UserId(requestUserId), + params), + consumerBasedTwoTowerSimilarityEngine + ).map { + Seq(_) + } + } else Future.Nil + + val customizedRetrievalBasedCandidatesFuture = + getCandidatesPerSimilarityEngineModel( + requestUserId, + params, + CustomizedRetrievalCandidateGeneration.fromParams, + customizedRetrievalCandidateGeneration.get) + + Future + .collect( + Seq( + tweetBasedCandidatesFuture, + producerBasedCandidatesFuture, + simClustersInterestedInBasedCandidatesFuture, + consumerBasedWalsCandidatesFuture, + consumerEmbeddingBasedLogFavBasedTripCandidatesFuture, + consumerEmbeddingBasedFollowBasedTripCandidatesFuture, + consumerEmbeddingBasedTwHINCandidatesFuture, + consumerEmbeddingBasedTwoTowerCandidatesFuture, + consumersBasedUvgRealGraphInCandidatesFuture, + customizedRetrievalBasedCandidatesFuture + )).map { candidatesList => + // remove empty innerSeq + val result = candidatesList.flatten.filter(_.nonEmpty) + stats.stat("numOfSequences").add(result.size) + stats.stat("flattenCandidatesWithDup").add(result.flatten.size) + + result + } + } + + private def getGraphBasedCandidates[QueryType]( + params: configapi.Params, + query: EngineQuery[QueryType], + engine: StandardSimilarityEngine[QueryType, TweetWithScore], + toSimilarityEngineInfo: Double => SimilarityEngineInfo, + graphSourceInfoOpt: Option[GraphSourceInfo] = None + ): Future[Seq[InitialCandidate]] = { + val candidatesOptFut = engine.getCandidates(query) + val tweetsWithCandidateGenerationInfoOptFut = candidatesOptFut.map { + _.map { tweetsWithScores => + val sortedCandidates = tweetsWithScores.sortBy(-_.score) + engine.getScopedStats.stat("sortedCandidates_size").add(sortedCandidates.size) + val tweetsWithCandidateGenerationInfo = sortedCandidates.map { tweetWithScore => + { + val similarityEngineInfo = toSimilarityEngineInfo(tweetWithScore.score) + val sourceInfo = graphSourceInfoOpt.map { graphSourceInfo => + // The internalId is a placeholder value. We do not plan to store the full seedUserId set. + SourceInfo( + sourceType = graphSourceInfo.sourceType, + internalId = InternalId.UserId(0L), + sourceEventTime = None + ) + } + TweetWithCandidateGenerationInfo( + tweetWithScore.tweetId, + CandidateGenerationInfo( + sourceInfo, + similarityEngineInfo, + Seq.empty // Atomic Similarity Engine. Hence it has no contributing SEs + ) + ) + } + } + val maxCandidateNum = params(GlobalParams.MaxCandidateNumPerSourceKeyParam) + tweetsWithCandidateGenerationInfo.take(maxCandidateNum) + } + } + for { + tweetsWithCandidateGenerationInfoOpt <- tweetsWithCandidateGenerationInfoOptFut + initialCandidates <- convertToInitialCandidates( + tweetsWithCandidateGenerationInfoOpt.toSeq.flatten) + } yield initialCandidates + } + + private def getCandidates[QueryType]( + sourceSignals: Set[SourceInfo], + params: configapi.Params, + fromParams: (SourceInfo, configapi.Params) => QueryType, + getFunc: QueryType => Future[Option[Seq[TweetWithCandidateGenerationInfo]]] + ): Future[Seq[Seq[InitialCandidate]]] = { + val queries = sourceSignals.map { sourceInfo => + fromParams(sourceInfo, params) + }.toSeq + + Future + .collect { + queries.map { query => + for { + candidates <- getFunc(query) + prefilterCandidates <- convertToInitialCandidates(candidates.toSeq.flatten) + } yield { + prefilterCandidates + } + } + } + } + + private def getConsumerBasedWalsCandidates( + sourceSignals: Set[SourceInfo], + params: configapi.Params + ): Future[Seq[InitialCandidate]] = { + // Fetch source signals and filter them based on age. + val signals = FilterUtil.tweetSourceAgeFilter( + getConsumerBasedWalsSourceInfo(sourceSignals).toSeq, + params(ConsumerBasedWalsParams.MaxTweetSignalAgeHoursParam)) + + val candidatesOptFut = consumerBasedWalsSimilarityEngine.getCandidates( + ConsumerBasedWalsSimilarityEngine.fromParams(signals, params) + ) + val tweetsWithCandidateGenerationInfoOptFut = candidatesOptFut.map { + _.map { tweetsWithScores => + val sortedCandidates = tweetsWithScores.sortBy(-_.score) + val filteredCandidates = + FilterUtil.tweetAgeFilter(sortedCandidates, params(GlobalParams.MaxTweetAgeHoursParam)) + consumerBasedWalsSimilarityEngine.getScopedStats + .stat("filteredCandidates_size").add(filteredCandidates.size) + + val tweetsWithCandidateGenerationInfo = filteredCandidates.map { tweetWithScore => + { + val similarityEngineInfo = + ConsumerBasedWalsSimilarityEngine.toSimilarityEngineInfo(tweetWithScore.score) + TweetWithCandidateGenerationInfo( + tweetWithScore.tweetId, + CandidateGenerationInfo( + None, + similarityEngineInfo, + Seq.empty // Atomic Similarity Engine. Hence it has no contributing SEs + ) + ) + } + } + val maxCandidateNum = params(GlobalParams.MaxCandidateNumPerSourceKeyParam) + tweetsWithCandidateGenerationInfo.take(maxCandidateNum) + } + } + for { + tweetsWithCandidateGenerationInfoOpt <- tweetsWithCandidateGenerationInfoOptFut + initialCandidates <- convertToInitialCandidates( + tweetsWithCandidateGenerationInfoOpt.toSeq.flatten) + } yield initialCandidates + } + + private def getSimClustersTripCandidates( + params: configapi.Params, + query: TripEngineQuery, + engine: StandardSimilarityEngine[ + TripEngineQuery, + TripTweetWithScore + ], + ): Future[Seq[InitialCandidate]] = { + val tweetsWithCandidatesGenerationInfoOptFut = + engine.getCandidates(EngineQuery(query, params)).map { + _.map { + _.map { tweetWithScore => + // define filters + TweetWithCandidateGenerationInfo( + tweetWithScore.tweetId, + CandidateGenerationInfo( + None, + SimilarityEngineInfo( + SimilarityEngineType.ExploreTripOfflineSimClustersTweets, + None, + Some(tweetWithScore.score)), + Seq.empty + ) + ) + } + } + } + for { + tweetsWithCandidateGenerationInfoOpt <- tweetsWithCandidatesGenerationInfoOptFut + initialCandidates <- convertToInitialCandidates( + tweetsWithCandidateGenerationInfoOpt.toSeq.flatten) + } yield initialCandidates + } + + private def getHnswCandidates( + params: configapi.Params, + query: HnswANNEngineQuery, + engine: HnswANNSimilarityEngine, + ): Future[Seq[InitialCandidate]] = { + val candidatesOptFut = engine.getCandidates(query) + val tweetsWithCandidateGenerationInfoOptFut = candidatesOptFut.map { + _.map { tweetsWithScores => + val sortedCandidates = tweetsWithScores.sortBy(-_.score) + val filteredCandidates = + FilterUtil.tweetAgeFilter(sortedCandidates, params(GlobalParams.MaxTweetAgeHoursParam)) + engine.getScopedStats.stat("filteredCandidates_size").add(filteredCandidates.size) + val tweetsWithCandidateGenerationInfo = filteredCandidates.map { tweetWithScore => + { + val similarityEngineInfo = + engine.toSimilarityEngineInfo(query, tweetWithScore.score) + TweetWithCandidateGenerationInfo( + tweetWithScore.tweetId, + CandidateGenerationInfo( + None, + similarityEngineInfo, + Seq.empty // Atomic Similarity Engine. Hence it has no contributing SEs + ) + ) + } + } + val maxCandidateNum = params(GlobalParams.MaxCandidateNumPerSourceKeyParam) + tweetsWithCandidateGenerationInfo.take(maxCandidateNum) + } + } + for { + tweetsWithCandidateGenerationInfoOpt <- tweetsWithCandidateGenerationInfoOptFut + initialCandidates <- convertToInitialCandidates( + tweetsWithCandidateGenerationInfoOpt.toSeq.flatten) + } yield initialCandidates + } + + /** + * Returns candidates from each similarity engine separately. + * For 1 requestUserId, it will fetch results from each similarity engine e_i, + * and returns Seq[Seq[TweetCandidate]]. + */ + private def getCandidatesPerSimilarityEngineModel[QueryType]( + requestUserId: UserId, + params: configapi.Params, + fromParams: (InternalId, configapi.Params) => QueryType, + getFunc: QueryType => Future[ + Option[Seq[Seq[TweetWithCandidateGenerationInfo]]] + ] + ): Future[Seq[Seq[InitialCandidate]]] = { + val query = fromParams(InternalId.UserId(requestUserId), params) + getFunc(query).flatMap { candidatesPerSimilarityEngineModelOpt => + val candidatesPerSimilarityEngineModel = candidatesPerSimilarityEngineModelOpt.toSeq.flatten + Future.collect { + candidatesPerSimilarityEngineModel.map(convertToInitialCandidates) + } + } + } + + private[candidate_generation] def convertToInitialCandidates( + candidates: Seq[TweetWithCandidateGenerationInfo], + ): Future[Seq[InitialCandidate]] = { + val tweetIds = candidates.map(_.tweetId).toSet + Future.collect(tweetInfoStore.multiGet(tweetIds)).map { tweetInfos => + /*** + * If tweetInfo does not exist, we will filter out this tweet candidate. + */ + candidates.collect { + case candidate if tweetInfos.getOrElse(candidate.tweetId, None).isDefined => + val tweetInfo = tweetInfos(candidate.tweetId) + .getOrElse(throw new IllegalStateException("Check previous line's condition")) + + InitialCandidate( + tweetId = candidate.tweetId, + tweetInfo = tweetInfo, + candidate.candidateGenerationInfo + ) + } + } + } +} + +object CandidateSourcesRouter { + def getGraphSourceInfoBySourceType( + sourceTypeStr: String, + sourceGraphs: Map[String, Option[GraphSourceInfo]] + ): Option[GraphSourceInfo] = { + sourceGraphs.getOrElse(sourceTypeStr, None) + } + + def getTweetBasedSourceInfo( + sourceSignals: Set[SourceInfo] + ): Set[SourceInfo] = { + sourceSignals.collect { + case sourceInfo + if AllowedSourceTypesForTweetBasedUnifiedSE.contains(sourceInfo.sourceType.value) => + sourceInfo + } + } + + def getProducerBasedSourceInfo( + sourceSignals: Set[SourceInfo] + ): Set[SourceInfo] = { + sourceSignals.collect { + case sourceInfo + if AllowedSourceTypesForProducerBasedUnifiedSE.contains(sourceInfo.sourceType.value) => + sourceInfo + } + } + + def getConsumerBasedWalsSourceInfo( + sourceSignals: Set[SourceInfo] + ): Set[SourceInfo] = { + sourceSignals.collect { + case sourceInfo + if AllowedSourceTypesForConsumerBasedWalsSE.contains(sourceInfo.sourceType.value) => + sourceInfo + } + } + + /*** + * Signal funneling should not exist in CG or even in any SimilarityEngine. + * They will be in Router, or eventually, in CrCandidateGenerator. + */ + val AllowedSourceTypesForConsumerBasedWalsSE = Set( + SourceType.TweetFavorite.value, + SourceType.Retweet.value, + SourceType.TweetDontLike.value, //currently no-op + SourceType.TweetReport.value, //currently no-op + SourceType.AccountMute.value, //currently no-op + SourceType.AccountBlock.value //currently no-op + ) + val AllowedSourceTypesForTweetBasedUnifiedSE = Set( + SourceType.TweetFavorite.value, + SourceType.Retweet.value, + SourceType.OriginalTweet.value, + SourceType.Reply.value, + SourceType.TweetShare.value, + SourceType.NotificationClick.value, + SourceType.GoodTweetClick.value, + SourceType.VideoTweetQualityView.value, + SourceType.VideoTweetPlayback50.value, + SourceType.TweetAggregation.value, + ) + val AllowedSourceTypesForProducerBasedUnifiedSE = Set( + SourceType.UserFollow.value, + SourceType.UserRepeatedProfileVisit.value, + SourceType.RealGraphOon.value, + SourceType.FollowRecommendation.value, + SourceType.UserTrafficAttributionProfileVisit.value, + SourceType.GoodProfileClick.value, + SourceType.ProducerAggregation.value, + ) +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/candidate_generation/CrCandidateGenerator.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/candidate_generation/CrCandidateGenerator.scala new file mode 100644 index 0000000000..c69d0c4f28 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/candidate_generation/CrCandidateGenerator.scala @@ -0,0 +1,350 @@ +package com.twitter.cr_mixer.candidate_generation + +import com.twitter.cr_mixer.blender.SwitchBlender +import com.twitter.cr_mixer.config.TimeoutConfig +import com.twitter.cr_mixer.filter.PostRankFilterRunner +import com.twitter.cr_mixer.filter.PreRankFilterRunner +import com.twitter.cr_mixer.logging.CrMixerScribeLogger +import com.twitter.cr_mixer.model.BlendedCandidate +import com.twitter.cr_mixer.model.CrCandidateGeneratorQuery +import com.twitter.cr_mixer.model.GraphSourceInfo +import com.twitter.cr_mixer.model.InitialCandidate +import com.twitter.cr_mixer.model.RankedCandidate +import com.twitter.cr_mixer.model.SourceInfo +import com.twitter.cr_mixer.param.RankerParams +import com.twitter.cr_mixer.param.RecentNegativeSignalParams +import com.twitter.cr_mixer.ranker.SwitchRanker +import com.twitter.cr_mixer.source_signal.SourceInfoRouter +import com.twitter.cr_mixer.source_signal.UssStore.EnabledNegativeSourceTypes +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.frigate.common.util.StatsUtil +import com.twitter.simclusters_v2.thriftscala.InternalId +import com.twitter.util.Future +import com.twitter.util.JavaTimer +import com.twitter.util.Timer + +import javax.inject.Inject +import javax.inject.Singleton + +/** + * For now it performs the main steps as follows: + * 1. Source signal (via USS, FRS) fetch + * 2. Candidate generation + * 3. Filtering + * 4. Interleave blender + * 5. Ranker + * 6. Post-ranker filter + * 7. Truncation + */ +@Singleton +class CrCandidateGenerator @Inject() ( + sourceInfoRouter: SourceInfoRouter, + candidateSourceRouter: CandidateSourcesRouter, + switchBlender: SwitchBlender, + preRankFilterRunner: PreRankFilterRunner, + postRankFilterRunner: PostRankFilterRunner, + switchRanker: SwitchRanker, + crMixerScribeLogger: CrMixerScribeLogger, + timeoutConfig: TimeoutConfig, + globalStats: StatsReceiver) { + private val timer: Timer = new JavaTimer(true) + + private val stats: StatsReceiver = globalStats.scope(this.getClass.getCanonicalName) + + private val fetchSourcesStats = stats.scope("fetchSources") + private val fetchPositiveSourcesStats = stats.scope("fetchPositiveSources") + private val fetchNegativeSourcesStats = stats.scope("fetchNegativeSources") + private val fetchCandidatesStats = stats.scope("fetchCandidates") + private val fetchCandidatesAfterFilterStats = stats.scope("fetchCandidatesAfterFilter") + private val preRankFilterStats = stats.scope("preRankFilter") + private val interleaveStats = stats.scope("interleave") + private val rankStats = stats.scope("rank") + private val postRankFilterStats = stats.scope("postRankFilter") + private val blueVerifiedTweetStats = stats.scope("blueVerifiedTweetStats") + private val blueVerifiedTweetStatsPerSimilarityEngine = + stats.scope("blueVerifiedTweetStatsPerSimilarityEngine") + + def get(query: CrCandidateGeneratorQuery): Future[Seq[RankedCandidate]] = { + val allStats = stats.scope("all") + val perProductStats = stats.scope("perProduct", query.product.toString) + val perProductBlueVerifiedStats = + blueVerifiedTweetStats.scope("perProduct", query.product.toString) + + StatsUtil.trackItemsStats(allStats) { + trackResultStats(perProductStats) { + StatsUtil.trackItemsStats(perProductStats) { + val result = for { + (sourceSignals, sourceGraphsMap) <- StatsUtil.trackBlockStats(fetchSourcesStats) { + fetchSources(query) + } + initialCandidates <- StatsUtil.trackBlockStats(fetchCandidatesAfterFilterStats) { + // find the positive and negative signals + val (positiveSignals, negativeSignals) = sourceSignals.partition { signal => + !EnabledNegativeSourceTypes.contains(signal.sourceType) + } + fetchPositiveSourcesStats.stat("size").add(positiveSignals.size) + fetchNegativeSourcesStats.stat("size").add(negativeSignals.size) + + // find the positive signals to keep, removing block and muted users + val filteredSourceInfo = + if (negativeSignals.nonEmpty && query.params( + RecentNegativeSignalParams.EnableSourceParam)) { + filterSourceInfo(positiveSignals, negativeSignals) + } else { + positiveSignals + } + + // fetch candidates from the positive signals + StatsUtil.trackBlockStats(fetchCandidatesStats) { + fetchCandidates(query, filteredSourceInfo, sourceGraphsMap) + } + } + filteredCandidates <- StatsUtil.trackBlockStats(preRankFilterStats) { + preRankFilter(query, initialCandidates) + } + interleavedCandidates <- StatsUtil.trackItemsStats(interleaveStats) { + interleave(query, filteredCandidates) + } + rankedCandidates <- StatsUtil.trackItemsStats(rankStats) { + val candidatesToRank = + interleavedCandidates.take(query.params(RankerParams.MaxCandidatesToRank)) + rank(query, candidatesToRank) + } + postRankFilterCandidates <- StatsUtil.trackItemsStats(postRankFilterStats) { + postRankFilter(query, rankedCandidates) + } + } yield { + trackTopKStats( + 800, + postRankFilterCandidates, + isQueryK = false, + perProductBlueVerifiedStats) + trackTopKStats( + 400, + postRankFilterCandidates, + isQueryK = false, + perProductBlueVerifiedStats) + trackTopKStats( + query.maxNumResults, + postRankFilterCandidates, + isQueryK = true, + perProductBlueVerifiedStats) + + val (blueVerifiedTweets, remainingTweets) = + postRankFilterCandidates.partition( + _.tweetInfo.hasBlueVerifiedAnnotation.contains(true)) + val topKBlueVerified = blueVerifiedTweets.take(query.maxNumResults) + val topKRemaining = remainingTweets.take(query.maxNumResults - topKBlueVerified.size) + + trackBlueVerifiedTweetStats(topKBlueVerified, perProductBlueVerifiedStats) + + if (topKBlueVerified.nonEmpty && query.params(RankerParams.EnableBlueVerifiedTopK)) { + topKBlueVerified ++ topKRemaining + } else { + postRankFilterCandidates + } + } + result.raiseWithin(timeoutConfig.serviceTimeout)(timer) + } + } + } + } + + private def fetchSources( + query: CrCandidateGeneratorQuery + ): Future[(Set[SourceInfo], Map[String, Option[GraphSourceInfo]])] = { + crMixerScribeLogger.scribeSignalSources( + query, + sourceInfoRouter + .get(query.userId, query.product, query.userState, query.params)) + } + + private def filterSourceInfo( + positiveSignals: Set[SourceInfo], + negativeSignals: Set[SourceInfo] + ): Set[SourceInfo] = { + val filterUsers: Set[Long] = negativeSignals.flatMap { + case SourceInfo(_, InternalId.UserId(userId), _) => Some(userId) + case _ => None + } + + positiveSignals.filter { + case SourceInfo(_, InternalId.UserId(userId), _) => !filterUsers.contains(userId) + case _ => true + } + } + + def fetchCandidates( + query: CrCandidateGeneratorQuery, + sourceSignals: Set[SourceInfo], + sourceGraphs: Map[String, Option[GraphSourceInfo]] + ): Future[Seq[Seq[InitialCandidate]]] = { + val initialCandidates = candidateSourceRouter + .fetchCandidates( + query.userId, + sourceSignals, + sourceGraphs, + query.params + ) + + initialCandidates.map(_.flatten.map { candidate => + if (candidate.tweetInfo.hasBlueVerifiedAnnotation.contains(true)) { + blueVerifiedTweetStatsPerSimilarityEngine + .scope(query.product.toString).scope( + candidate.candidateGenerationInfo.contributingSimilarityEngines.head.similarityEngineType.toString).counter( + candidate.tweetInfo.authorId.toString).incr() + } + }) + + crMixerScribeLogger.scribeInitialCandidates( + query, + initialCandidates + ) + } + + private def preRankFilter( + query: CrCandidateGeneratorQuery, + candidates: Seq[Seq[InitialCandidate]] + ): Future[Seq[Seq[InitialCandidate]]] = { + crMixerScribeLogger.scribePreRankFilterCandidates( + query, + preRankFilterRunner + .runSequentialFilters(query, candidates)) + } + + private def postRankFilter( + query: CrCandidateGeneratorQuery, + candidates: Seq[RankedCandidate] + ): Future[Seq[RankedCandidate]] = { + postRankFilterRunner.run(query, candidates) + } + + private def interleave( + query: CrCandidateGeneratorQuery, + candidates: Seq[Seq[InitialCandidate]] + ): Future[Seq[BlendedCandidate]] = { + crMixerScribeLogger.scribeInterleaveCandidates( + query, + switchBlender + .blend(query.params, query.userState, candidates)) + } + + private def rank( + query: CrCandidateGeneratorQuery, + candidates: Seq[BlendedCandidate], + ): Future[Seq[RankedCandidate]] = { + crMixerScribeLogger.scribeRankedCandidates( + query, + switchRanker.rank(query, candidates) + ) + } + + private def trackResultStats( + stats: StatsReceiver + )( + fn: => Future[Seq[RankedCandidate]] + ): Future[Seq[RankedCandidate]] = { + fn.onSuccess { candidates => + trackReasonChosenSourceTypeStats(candidates, stats) + trackReasonChosenSimilarityEngineStats(candidates, stats) + trackPotentialReasonsSourceTypeStats(candidates, stats) + trackPotentialReasonsSimilarityEngineStats(candidates, stats) + } + } + + private def trackReasonChosenSourceTypeStats( + candidates: Seq[RankedCandidate], + stats: StatsReceiver + ): Unit = { + candidates + .groupBy(_.reasonChosen.sourceInfoOpt.map(_.sourceType)) + .foreach { + case (sourceTypeOpt, rankedCands) => + val sourceType = sourceTypeOpt.map(_.toString).getOrElse("RequesterId") // default + stats.stat("reasonChosen", "sourceType", sourceType, "size").add(rankedCands.size) + } + } + + private def trackReasonChosenSimilarityEngineStats( + candidates: Seq[RankedCandidate], + stats: StatsReceiver + ): Unit = { + candidates + .groupBy(_.reasonChosen.similarityEngineInfo.similarityEngineType) + .foreach { + case (seInfoType, rankedCands) => + stats + .stat("reasonChosen", "similarityEngine", seInfoType.toString, "size").add( + rankedCands.size) + } + } + + private def trackPotentialReasonsSourceTypeStats( + candidates: Seq[RankedCandidate], + stats: StatsReceiver + ): Unit = { + candidates + .flatMap(_.potentialReasons.map(_.sourceInfoOpt.map(_.sourceType))) + .groupBy(source => source) + .foreach { + case (sourceInfoOpt, seq) => + val sourceType = sourceInfoOpt.map(_.toString).getOrElse("RequesterId") // default + stats.stat("potentialReasons", "sourceType", sourceType, "size").add(seq.size) + } + } + + private def trackPotentialReasonsSimilarityEngineStats( + candidates: Seq[RankedCandidate], + stats: StatsReceiver + ): Unit = { + candidates + .flatMap(_.potentialReasons.map(_.similarityEngineInfo.similarityEngineType)) + .groupBy(se => se) + .foreach { + case (seType, seq) => + stats.stat("potentialReasons", "similarityEngine", seType.toString, "size").add(seq.size) + } + } + + private def trackBlueVerifiedTweetStats( + candidates: Seq[RankedCandidate], + statsReceiver: StatsReceiver + ): Unit = { + candidates.foreach { candidate => + if (candidate.tweetInfo.hasBlueVerifiedAnnotation.contains(true)) { + statsReceiver.counter(candidate.tweetInfo.authorId.toString).incr() + statsReceiver + .scope(candidate.tweetInfo.authorId.toString).counter(candidate.tweetId.toString).incr() + } + } + } + + private def trackTopKStats( + k: Int, + tweetCandidates: Seq[RankedCandidate], + isQueryK: Boolean, + statsReceiver: StatsReceiver + ): Unit = { + val (topK, beyondK) = tweetCandidates.splitAt(k) + + val blueVerifiedIds = tweetCandidates.collect { + case candidate if candidate.tweetInfo.hasBlueVerifiedAnnotation.contains(true) => + candidate.tweetInfo.authorId + }.toSet + + blueVerifiedIds.foreach { blueVerifiedId => + val numTweetsTopK = topK.count(_.tweetInfo.authorId == blueVerifiedId) + val numTweetsBeyondK = beyondK.count(_.tweetInfo.authorId == blueVerifiedId) + + if (isQueryK) { + statsReceiver.scope(blueVerifiedId.toString).stat(s"topK").add(numTweetsTopK) + statsReceiver + .scope(blueVerifiedId.toString).stat(s"beyondK").add(numTweetsBeyondK) + } else { + statsReceiver.scope(blueVerifiedId.toString).stat(s"top$k").add(numTweetsTopK) + statsReceiver + .scope(blueVerifiedId.toString).stat(s"beyond$k").add(numTweetsBeyondK) + } + } + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/candidate_generation/CustomizedRetrievalCandidateGeneration.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/candidate_generation/CustomizedRetrievalCandidateGeneration.scala new file mode 100644 index 0000000000..427dd9b746 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/candidate_generation/CustomizedRetrievalCandidateGeneration.scala @@ -0,0 +1,345 @@ +package com.twitter.cr_mixer.candidate_generation + +import com.twitter.cr_mixer.candidate_generation.CustomizedRetrievalCandidateGeneration.Query +import com.twitter.cr_mixer.model.CandidateGenerationInfo +import com.twitter.cr_mixer.model.ModuleNames +import com.twitter.cr_mixer.model.TweetWithCandidateGenerationInfo +import com.twitter.cr_mixer.model.TweetWithScore +import com.twitter.cr_mixer.param.CustomizedRetrievalBasedCandidateGenerationParams._ +import com.twitter.cr_mixer.param.CustomizedRetrievalBasedTwhinParams._ +import com.twitter.cr_mixer.param.GlobalParams +import com.twitter.cr_mixer.similarity_engine.DiffusionBasedSimilarityEngine +import com.twitter.cr_mixer.similarity_engine.LookupEngineQuery +import com.twitter.cr_mixer.similarity_engine.LookupSimilarityEngine +import com.twitter.cr_mixer.similarity_engine.TwhinCollabFilterSimilarityEngine +import com.twitter.cr_mixer.util.InterleaveUtil +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.frigate.common.base.CandidateSource +import com.twitter.frigate.common.base.Stats +import com.twitter.simclusters_v2.thriftscala.InternalId +import com.twitter.snowflake.id.SnowflakeId +import com.twitter.timelines.configapi +import com.twitter.util.Duration +import com.twitter.util.Future +import com.twitter.util.Time +import javax.inject.Inject +import javax.inject.Named +import javax.inject.Singleton +import scala.collection.mutable.ArrayBuffer + +/** + * A candidate generator that fetches similar tweets from multiple customized retrieval based candidate sources + * + * Different from [[TweetBasedCandidateGeneration]], this store returns candidates from different + * similarity engines without blending. In other words, this class shall not be thought of as a + * Unified Similarity Engine. It is a CG that calls multiple singular Similarity Engines. + */ +@Singleton +case class CustomizedRetrievalCandidateGeneration @Inject() ( + @Named(ModuleNames.TwhinCollabFilterSimilarityEngine) + twhinCollabFilterSimilarityEngine: LookupSimilarityEngine[ + TwhinCollabFilterSimilarityEngine.Query, + TweetWithScore + ], + @Named(ModuleNames.DiffusionBasedSimilarityEngine) + diffusionBasedSimilarityEngine: LookupSimilarityEngine[ + DiffusionBasedSimilarityEngine.Query, + TweetWithScore + ], + statsReceiver: StatsReceiver) + extends CandidateSource[ + Query, + Seq[TweetWithCandidateGenerationInfo] + ] { + + override def name: String = this.getClass.getSimpleName + + private val stats = statsReceiver.scope(name) + private val fetchCandidatesStat = stats.scope("fetchCandidates") + + /** + * For each Similarity Engine Model, return a list of tweet candidates + */ + override def get( + query: Query + ): Future[Option[Seq[Seq[TweetWithCandidateGenerationInfo]]]] = { + query.internalId match { + case InternalId.UserId(_) => + Stats.trackOption(fetchCandidatesStat) { + val twhinCollabFilterForFollowCandidatesFut = if (query.enableTwhinCollabFilter) { + twhinCollabFilterSimilarityEngine.getCandidates(query.twhinCollabFilterFollowQuery) + } else Future.None + + val twhinCollabFilterForEngagementCandidatesFut = + if (query.enableTwhinCollabFilter) { + twhinCollabFilterSimilarityEngine.getCandidates( + query.twhinCollabFilterEngagementQuery) + } else Future.None + + val twhinMultiClusterForFollowCandidatesFut = if (query.enableTwhinMultiCluster) { + twhinCollabFilterSimilarityEngine.getCandidates(query.twhinMultiClusterFollowQuery) + } else Future.None + + val twhinMultiClusterForEngagementCandidatesFut = + if (query.enableTwhinMultiCluster) { + twhinCollabFilterSimilarityEngine.getCandidates( + query.twhinMultiClusterEngagementQuery) + } else Future.None + + val diffusionBasedSimilarityEngineCandidatesFut = if (query.enableRetweetBasedDiffusion) { + diffusionBasedSimilarityEngine.getCandidates(query.diffusionBasedSimilarityEngineQuery) + } else Future.None + + Future + .join( + twhinCollabFilterForFollowCandidatesFut, + twhinCollabFilterForEngagementCandidatesFut, + twhinMultiClusterForFollowCandidatesFut, + twhinMultiClusterForEngagementCandidatesFut, + diffusionBasedSimilarityEngineCandidatesFut + ).map { + case ( + twhinCollabFilterForFollowCandidates, + twhinCollabFilterForEngagementCandidates, + twhinMultiClusterForFollowCandidates, + twhinMultiClusterForEngagementCandidates, + diffusionBasedSimilarityEngineCandidates) => + val maxCandidateNumPerSourceKey = 200 + val twhinCollabFilterForFollowWithCGInfo = + getTwhinCollabCandidatesWithCGInfo( + twhinCollabFilterForFollowCandidates, + maxCandidateNumPerSourceKey, + query.twhinCollabFilterFollowQuery, + ) + val twhinCollabFilterForEngagementWithCGInfo = + getTwhinCollabCandidatesWithCGInfo( + twhinCollabFilterForEngagementCandidates, + maxCandidateNumPerSourceKey, + query.twhinCollabFilterEngagementQuery, + ) + val twhinMultiClusterForFollowWithCGInfo = + getTwhinCollabCandidatesWithCGInfo( + twhinMultiClusterForFollowCandidates, + maxCandidateNumPerSourceKey, + query.twhinMultiClusterFollowQuery, + ) + val twhinMultiClusterForEngagementWithCGInfo = + getTwhinCollabCandidatesWithCGInfo( + twhinMultiClusterForEngagementCandidates, + maxCandidateNumPerSourceKey, + query.twhinMultiClusterEngagementQuery, + ) + val retweetBasedDiffusionWithCGInfo = + getDiffusionBasedCandidatesWithCGInfo( + diffusionBasedSimilarityEngineCandidates, + maxCandidateNumPerSourceKey, + query.diffusionBasedSimilarityEngineQuery, + ) + + val twhinCollabCandidateSourcesToBeInterleaved = + ArrayBuffer[Seq[TweetWithCandidateGenerationInfo]]( + twhinCollabFilterForFollowWithCGInfo, + twhinCollabFilterForEngagementWithCGInfo, + ) + + val twhinMultiClusterCandidateSourcesToBeInterleaved = + ArrayBuffer[Seq[TweetWithCandidateGenerationInfo]]( + twhinMultiClusterForFollowWithCGInfo, + twhinMultiClusterForEngagementWithCGInfo, + ) + + val interleavedTwhinCollabCandidates = + InterleaveUtil.interleave(twhinCollabCandidateSourcesToBeInterleaved) + + val interleavedTwhinMultiClusterCandidates = + InterleaveUtil.interleave(twhinMultiClusterCandidateSourcesToBeInterleaved) + + val twhinCollabFilterResults = + if (interleavedTwhinCollabCandidates.nonEmpty) { + Some(interleavedTwhinCollabCandidates.take(maxCandidateNumPerSourceKey)) + } else None + + val twhinMultiClusterResults = + if (interleavedTwhinMultiClusterCandidates.nonEmpty) { + Some(interleavedTwhinMultiClusterCandidates.take(maxCandidateNumPerSourceKey)) + } else None + + val diffusionResults = + if (retweetBasedDiffusionWithCGInfo.nonEmpty) { + Some(retweetBasedDiffusionWithCGInfo.take(maxCandidateNumPerSourceKey)) + } else None + + Some( + Seq( + twhinCollabFilterResults, + twhinMultiClusterResults, + diffusionResults + ).flatten) + } + } + case _ => + throw new IllegalArgumentException("sourceId_is_not_userId_cnt") + } + } + + /** Returns a list of tweets that are generated less than `maxTweetAgeHours` hours ago */ + private def tweetAgeFilter( + candidates: Seq[TweetWithScore], + maxTweetAgeHours: Duration + ): Seq[TweetWithScore] = { + // Tweet IDs are approximately chronological (see http://go/snowflake), + // so we are building the earliest tweet id once + // The per-candidate logic here then be candidate.tweetId > earliestPermittedTweetId, which is far cheaper. + val earliestTweetId = SnowflakeId.firstIdFor(Time.now - maxTweetAgeHours) + candidates.filter { candidate => candidate.tweetId >= earliestTweetId } + } + + /** + * AgeFilters tweetCandidates with stats + * Only age filter logic is effective here (through tweetAgeFilter). This function acts mostly for metric logging. + */ + private def ageFilterWithStats( + offlineInterestedInCandidates: Seq[TweetWithScore], + maxTweetAgeHours: Duration, + scopedStatsReceiver: StatsReceiver + ): Seq[TweetWithScore] = { + scopedStatsReceiver.stat("size").add(offlineInterestedInCandidates.size) + val candidates = offlineInterestedInCandidates.map { candidate => + TweetWithScore(candidate.tweetId, candidate.score) + } + val filteredCandidates = tweetAgeFilter(candidates, maxTweetAgeHours) + scopedStatsReceiver.stat(f"filtered_size").add(filteredCandidates.size) + if (filteredCandidates.isEmpty) scopedStatsReceiver.counter(f"empty").incr() + + filteredCandidates + } + + private def getTwhinCollabCandidatesWithCGInfo( + tweetCandidates: Option[Seq[TweetWithScore]], + maxCandidateNumPerSourceKey: Int, + twhinCollabFilterQuery: LookupEngineQuery[ + TwhinCollabFilterSimilarityEngine.Query + ], + ): Seq[TweetWithCandidateGenerationInfo] = { + val twhinTweets = tweetCandidates match { + case Some(tweetsWithScores) => + tweetsWithScores.map { tweetWithScore => + TweetWithCandidateGenerationInfo( + tweetWithScore.tweetId, + CandidateGenerationInfo( + None, + TwhinCollabFilterSimilarityEngine + .toSimilarityEngineInfo(twhinCollabFilterQuery, tweetWithScore.score), + Seq.empty + ) + ) + } + case _ => Seq.empty + } + twhinTweets.take(maxCandidateNumPerSourceKey) + } + + private def getDiffusionBasedCandidatesWithCGInfo( + tweetCandidates: Option[Seq[TweetWithScore]], + maxCandidateNumPerSourceKey: Int, + diffusionBasedSimilarityEngineQuery: LookupEngineQuery[ + DiffusionBasedSimilarityEngine.Query + ], + ): Seq[TweetWithCandidateGenerationInfo] = { + val diffusionTweets = tweetCandidates match { + case Some(tweetsWithScores) => + tweetsWithScores.map { tweetWithScore => + TweetWithCandidateGenerationInfo( + tweetWithScore.tweetId, + CandidateGenerationInfo( + None, + DiffusionBasedSimilarityEngine + .toSimilarityEngineInfo(diffusionBasedSimilarityEngineQuery, tweetWithScore.score), + Seq.empty + ) + ) + } + case _ => Seq.empty + } + diffusionTweets.take(maxCandidateNumPerSourceKey) + } +} + +object CustomizedRetrievalCandidateGeneration { + + case class Query( + internalId: InternalId, + maxCandidateNumPerSourceKey: Int, + maxTweetAgeHours: Duration, + // twhinCollabFilter + enableTwhinCollabFilter: Boolean, + twhinCollabFilterFollowQuery: LookupEngineQuery[ + TwhinCollabFilterSimilarityEngine.Query + ], + twhinCollabFilterEngagementQuery: LookupEngineQuery[ + TwhinCollabFilterSimilarityEngine.Query + ], + // twhinMultiCluster + enableTwhinMultiCluster: Boolean, + twhinMultiClusterFollowQuery: LookupEngineQuery[ + TwhinCollabFilterSimilarityEngine.Query + ], + twhinMultiClusterEngagementQuery: LookupEngineQuery[ + TwhinCollabFilterSimilarityEngine.Query + ], + enableRetweetBasedDiffusion: Boolean, + diffusionBasedSimilarityEngineQuery: LookupEngineQuery[ + DiffusionBasedSimilarityEngine.Query + ], + ) + + def fromParams( + internalId: InternalId, + params: configapi.Params + ): Query = { + val twhinCollabFilterFollowQuery = + TwhinCollabFilterSimilarityEngine.fromParams( + internalId, + params(CustomizedRetrievalBasedTwhinCollabFilterFollowSource), + params) + + val twhinCollabFilterEngagementQuery = + TwhinCollabFilterSimilarityEngine.fromParams( + internalId, + params(CustomizedRetrievalBasedTwhinCollabFilterEngagementSource), + params) + + val twhinMultiClusterFollowQuery = + TwhinCollabFilterSimilarityEngine.fromParams( + internalId, + params(CustomizedRetrievalBasedTwhinMultiClusterFollowSource), + params) + + val twhinMultiClusterEngagementQuery = + TwhinCollabFilterSimilarityEngine.fromParams( + internalId, + params(CustomizedRetrievalBasedTwhinMultiClusterEngagementSource), + params) + + val diffusionBasedSimilarityEngineQuery = + DiffusionBasedSimilarityEngine.fromParams( + internalId, + params(CustomizedRetrievalBasedRetweetDiffusionSource), + params) + + Query( + internalId = internalId, + maxCandidateNumPerSourceKey = params(GlobalParams.MaxCandidateNumPerSourceKeyParam), + maxTweetAgeHours = params(GlobalParams.MaxTweetAgeHoursParam), + // twhinCollabFilter + enableTwhinCollabFilter = params(EnableTwhinCollabFilterClusterParam), + twhinCollabFilterFollowQuery = twhinCollabFilterFollowQuery, + twhinCollabFilterEngagementQuery = twhinCollabFilterEngagementQuery, + enableTwhinMultiCluster = params(EnableTwhinMultiClusterParam), + twhinMultiClusterFollowQuery = twhinMultiClusterFollowQuery, + twhinMultiClusterEngagementQuery = twhinMultiClusterEngagementQuery, + enableRetweetBasedDiffusion = params(EnableRetweetBasedDiffusionParam), + diffusionBasedSimilarityEngineQuery = diffusionBasedSimilarityEngineQuery + ) + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/candidate_generation/FrsTweetCandidateGenerator.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/candidate_generation/FrsTweetCandidateGenerator.scala new file mode 100644 index 0000000000..0c5334c28f --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/candidate_generation/FrsTweetCandidateGenerator.scala @@ -0,0 +1,220 @@ +package com.twitter.cr_mixer.candidate_generation + +import com.twitter.contentrecommender.thriftscala.TweetInfo +import com.twitter.cr_mixer.config.TimeoutConfig +import com.twitter.cr_mixer.model.FrsTweetCandidateGeneratorQuery +import com.twitter.cr_mixer.model.ModuleNames +import com.twitter.cr_mixer.model.TweetWithAuthor +import com.twitter.cr_mixer.param.FrsParams +import com.twitter.cr_mixer.similarity_engine.EarlybirdSimilarityEngineRouter +import com.twitter.cr_mixer.source_signal.FrsStore +import com.twitter.cr_mixer.source_signal.FrsStore.FrsQueryResult +import com.twitter.cr_mixer.thriftscala.FrsTweet +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.finagle.util.DefaultTimer +import com.twitter.frigate.common.util.StatsUtil +import com.twitter.hermit.constants.AlgorithmFeedbackTokens +import com.twitter.hermit.constants.AlgorithmFeedbackTokens.AlgorithmToFeedbackTokenMap +import com.twitter.hermit.model.Algorithm +import com.twitter.simclusters_v2.common.TweetId +import com.twitter.simclusters_v2.common.UserId +import com.twitter.storehaus.ReadableStore +import com.twitter.timelines.configapi.Params +import com.twitter.util.Future +import javax.inject.Inject +import javax.inject.Named +import javax.inject.Singleton + +/** + * TweetCandidateGenerator based on FRS seed users. For now this candidate generator fetches seed + * users from FRS, and retrieves the seed users' past tweets from Earlybird with Earlybird light + * ranking models. + */ +@Singleton +class FrsTweetCandidateGenerator @Inject() ( + @Named(ModuleNames.FrsStore) frsStore: ReadableStore[FrsStore.Query, Seq[FrsQueryResult]], + frsBasedSimilarityEngine: EarlybirdSimilarityEngineRouter, + tweetInfoStore: ReadableStore[TweetId, TweetInfo], + timeoutConfig: TimeoutConfig, + globalStats: StatsReceiver) { + import FrsTweetCandidateGenerator._ + + private val timer = DefaultTimer + private val stats: StatsReceiver = globalStats.scope(this.getClass.getCanonicalName) + private val fetchSeedsStats = stats.scope("fetchSeeds") + private val fetchCandidatesStats = stats.scope("fetchCandidates") + private val filterCandidatesStats = stats.scope("filterCandidates") + private val hydrateCandidatesStats = stats.scope("hydrateCandidates") + private val getCandidatesStats = stats.scope("getCandidates") + + /** + * The function retrieves the candidate for the given user as follows: + * 1. Seed user fetch from FRS. + * 2. Candidate fetch from Earlybird. + * 3. Filtering. + * 4. Candidate hydration. + * 5. Truncation. + */ + def get( + frsTweetCandidateGeneratorQuery: FrsTweetCandidateGeneratorQuery + ): Future[Seq[FrsTweet]] = { + val userId = frsTweetCandidateGeneratorQuery.userId + val product = frsTweetCandidateGeneratorQuery.product + val allStats = stats.scope("all") + val perProductStats = stats.scope("perProduct", product.name) + StatsUtil.trackItemsStats(allStats) { + StatsUtil.trackItemsStats(perProductStats) { + val result = for { + seedAuthorWithScores <- StatsUtil.trackOptionItemMapStats(fetchSeedsStats) { + fetchSeeds( + userId, + frsTweetCandidateGeneratorQuery.impressedUserList, + frsTweetCandidateGeneratorQuery.languageCodeOpt, + frsTweetCandidateGeneratorQuery.countryCodeOpt, + frsTweetCandidateGeneratorQuery.params, + ) + } + tweetCandidates <- StatsUtil.trackOptionItemsStats(fetchCandidatesStats) { + fetchCandidates( + userId, + seedAuthorWithScores.map(_.keys.toSeq).getOrElse(Seq.empty), + frsTweetCandidateGeneratorQuery.impressedTweetList, + seedAuthorWithScores.map(_.mapValues(_.score)).getOrElse(Map.empty), + frsTweetCandidateGeneratorQuery.params + ) + } + filteredTweetCandidates <- StatsUtil.trackOptionItemsStats(filterCandidatesStats) { + filterCandidates( + tweetCandidates, + frsTweetCandidateGeneratorQuery.params + ) + } + hydratedTweetCandidates <- StatsUtil.trackOptionItemsStats(hydrateCandidatesStats) { + hydrateCandidates( + seedAuthorWithScores, + filteredTweetCandidates + ) + } + } yield { + hydratedTweetCandidates + .map(_.take(frsTweetCandidateGeneratorQuery.maxNumResults)).getOrElse(Seq.empty) + } + result.raiseWithin(timeoutConfig.frsBasedTweetEndpointTimeout)(timer) + } + } + } + + /** + * Fetch recommended seed users from FRS + */ + private def fetchSeeds( + userId: UserId, + userDenyList: Set[UserId], + languageCodeOpt: Option[String], + countryCodeOpt: Option[String], + params: Params + ): Future[Option[Map[UserId, FrsQueryResult]]] = { + frsStore + .get( + FrsStore.Query( + userId, + params(FrsParams.FrsBasedCandidateGenerationMaxSeedsNumParam), + params(FrsParams.FrsBasedCandidateGenerationDisplayLocationParam).displayLocation, + userDenyList.toSeq, + languageCodeOpt, + countryCodeOpt + )).map { + _.map { seedAuthors => + seedAuthors.map(user => user.userId -> user).toMap + } + } + } + + /** + * Fetch tweet candidates from Earlybird + */ + private def fetchCandidates( + searcherUserId: UserId, + seedAuthors: Seq[UserId], + impressedTweetList: Set[TweetId], + frsUserToScores: Map[UserId, Double], + params: Params + ): Future[Option[Seq[TweetWithAuthor]]] = { + if (seedAuthors.nonEmpty) { + // call earlybird + val query = EarlybirdSimilarityEngineRouter.queryFromParams( + Some(searcherUserId), + seedAuthors, + impressedTweetList, + frsUserToScoresForScoreAdjustment = Some(frsUserToScores), + params + ) + frsBasedSimilarityEngine.get(query) + } else Future.None + } + + /** + * Filter candidates that do not pass visibility filter policy + */ + private def filterCandidates( + candidates: Option[Seq[TweetWithAuthor]], + params: Params + ): Future[Option[Seq[TweetWithAuthor]]] = { + val tweetIds = candidates.map(_.map(_.tweetId).toSet).getOrElse(Set.empty) + if (params(FrsParams.FrsBasedCandidateGenerationEnableVisibilityFilteringParam)) + Future + .collect(tweetInfoStore.multiGet(tweetIds)).map { tweetInfos => + candidates.map { + // If tweetInfo does not exist, we will filter out this tweet candidate. + _.filter(candidate => tweetInfos.getOrElse(candidate.tweetId, None).isDefined) + } + } + else { + Future.value(candidates) + } + } + + /** + * Hydrate the candidates with the FRS candidate sources and scores + */ + private def hydrateCandidates( + frsAuthorWithScores: Option[Map[UserId, FrsQueryResult]], + candidates: Option[Seq[TweetWithAuthor]] + ): Future[Option[Seq[FrsTweet]]] = { + Future.value { + candidates.map { + _.map { tweetWithAuthor => + val frsQueryResult = frsAuthorWithScores.flatMap(_.get(tweetWithAuthor.authorId)) + FrsTweet( + tweetId = tweetWithAuthor.tweetId, + authorId = tweetWithAuthor.authorId, + frsPrimarySource = frsQueryResult.flatMap(_.primarySource), + frsAuthorScore = frsQueryResult.map(_.score), + frsCandidateSourceScores = frsQueryResult.flatMap { result => + result.sourceWithScores.map { + _.collect { + // see TokenStrToAlgorithmMap @ https://sourcegraph.twitter.biz/git.twitter.biz/source/-/blob/hermit/hermit-core/src/main/scala/com/twitter/hermit/constants/AlgorithmFeedbackTokens.scala + // see Algorithm @ https://sourcegraph.twitter.biz/git.twitter.biz/source/-/blob/hermit/hermit-core/src/main/scala/com/twitter/hermit/model/Algorithm.scala + case (candidateSourceAlgoStr, score) + if AlgorithmFeedbackTokens.TokenStrToAlgorithmMap.contains( + candidateSourceAlgoStr) => + AlgorithmToFeedbackTokenMap.getOrElse( + AlgorithmFeedbackTokens.TokenStrToAlgorithmMap + .getOrElse(candidateSourceAlgoStr, DefaultAlgo), + DefaultAlgoToken) -> score + } + } + } + ) + } + } + } + } + +} + +object FrsTweetCandidateGenerator { + val DefaultAlgo: Algorithm.Value = Algorithm.Other + // 9999 is the token for Algorithm.Other + val DefaultAlgoToken: Int = AlgorithmToFeedbackTokenMap.getOrElse(DefaultAlgo, 9999) +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/candidate_generation/RelatedTweetCandidateGenerator.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/candidate_generation/RelatedTweetCandidateGenerator.scala new file mode 100644 index 0000000000..45a919a57e --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/candidate_generation/RelatedTweetCandidateGenerator.scala @@ -0,0 +1,156 @@ +package com.twitter.cr_mixer.candidate_generation + +import com.twitter.contentrecommender.thriftscala.TweetInfo +import com.twitter.cr_mixer.filter.PreRankFilterRunner +import com.twitter.cr_mixer.logging.RelatedTweetScribeLogger +import com.twitter.cr_mixer.model.InitialCandidate +import com.twitter.cr_mixer.model.RelatedTweetCandidateGeneratorQuery +import com.twitter.cr_mixer.model.TweetWithCandidateGenerationInfo +import com.twitter.cr_mixer.model.ModuleNames +import com.twitter.cr_mixer.similarity_engine.ProducerBasedUnifiedSimilarityEngine +import com.twitter.cr_mixer.similarity_engine.StandardSimilarityEngine +import com.twitter.cr_mixer.similarity_engine.TweetBasedUnifiedSimilarityEngine +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.frigate.common.util.StatsUtil +import com.twitter.simclusters_v2.common.TweetId +import com.twitter.simclusters_v2.thriftscala.InternalId +import com.twitter.storehaus.ReadableStore +import com.twitter.timelines.configapi +import com.twitter.util.Future +import javax.inject.Inject +import javax.inject.Named +import javax.inject.Singleton + +@Singleton +class RelatedTweetCandidateGenerator @Inject() ( + @Named(ModuleNames.TweetBasedUnifiedSimilarityEngine) tweetBasedUnifiedSimilarityEngine: StandardSimilarityEngine[ + TweetBasedUnifiedSimilarityEngine.Query, + TweetWithCandidateGenerationInfo + ], + @Named(ModuleNames.ProducerBasedUnifiedSimilarityEngine) producerBasedUnifiedSimilarityEngine: StandardSimilarityEngine[ + ProducerBasedUnifiedSimilarityEngine.Query, + TweetWithCandidateGenerationInfo + ], + preRankFilterRunner: PreRankFilterRunner, + relatedTweetScribeLogger: RelatedTweetScribeLogger, + tweetInfoStore: ReadableStore[TweetId, TweetInfo], + globalStats: StatsReceiver) { + + private val stats: StatsReceiver = globalStats.scope(this.getClass.getCanonicalName) + private val fetchCandidatesStats = stats.scope("fetchCandidates") + private val preRankFilterStats = stats.scope("preRankFilter") + + def get( + query: RelatedTweetCandidateGeneratorQuery + ): Future[Seq[InitialCandidate]] = { + + val allStats = stats.scope("all") + val perProductStats = stats.scope("perProduct", query.product.toString) + StatsUtil.trackItemsStats(allStats) { + StatsUtil.trackItemsStats(perProductStats) { + for { + initialCandidates <- StatsUtil.trackBlockStats(fetchCandidatesStats) { + fetchCandidates(query) + } + filteredCandidates <- StatsUtil.trackBlockStats(preRankFilterStats) { + preRankFilter(query, initialCandidates) + } + } yield { + filteredCandidates.headOption + .getOrElse( + throw new UnsupportedOperationException( + "RelatedTweetCandidateGenerator results invalid") + ).take(query.maxNumResults) + } + } + } + } + + def fetchCandidates( + query: RelatedTweetCandidateGeneratorQuery + ): Future[Seq[Seq[InitialCandidate]]] = { + relatedTweetScribeLogger.scribeInitialCandidates( + query, + query.internalId match { + case InternalId.TweetId(_) => + getCandidatesFromSimilarityEngine( + query, + TweetBasedUnifiedSimilarityEngine.fromParamsForRelatedTweet, + tweetBasedUnifiedSimilarityEngine.getCandidates) + case InternalId.UserId(_) => + getCandidatesFromSimilarityEngine( + query, + ProducerBasedUnifiedSimilarityEngine.fromParamsForRelatedTweet, + producerBasedUnifiedSimilarityEngine.getCandidates) + case _ => + throw new UnsupportedOperationException( + "RelatedTweetCandidateGenerator gets invalid InternalId") + } + ) + } + + /*** + * fetch Candidates from TweetBased/ProducerBased Unified Similarity Engine, + * and apply VF filter based on TweetInfoStore + * To align with the downstream processing (filter, rank), we tend to return a Seq[Seq[InitialCandidate]] + * instead of a Seq[Candidate] even though we only have a Seq in it. + */ + private def getCandidatesFromSimilarityEngine[QueryType]( + query: RelatedTweetCandidateGeneratorQuery, + fromParamsForRelatedTweet: (InternalId, configapi.Params) => QueryType, + getFunc: QueryType => Future[Option[Seq[TweetWithCandidateGenerationInfo]]] + ): Future[Seq[Seq[InitialCandidate]]] = { + + /*** + * We wrap the query to be a Seq of queries for the Sim Engine to ensure evolvability of candidate generation + * and as a result, it will return Seq[Seq[InitialCandidate]] + */ + val engineQueries = + Seq(fromParamsForRelatedTweet(query.internalId, query.params)) + + Future + .collect { + engineQueries.map { query => + for { + candidates <- getFunc(query) + prefilterCandidates <- convertToInitialCandidates( + candidates.toSeq.flatten + ) + } yield prefilterCandidates + } + } + } + + private def preRankFilter( + query: RelatedTweetCandidateGeneratorQuery, + candidates: Seq[Seq[InitialCandidate]] + ): Future[Seq[Seq[InitialCandidate]]] = { + relatedTweetScribeLogger.scribePreRankFilterCandidates( + query, + preRankFilterRunner + .runSequentialFilters(query, candidates)) + } + + private[candidate_generation] def convertToInitialCandidates( + candidates: Seq[TweetWithCandidateGenerationInfo], + ): Future[Seq[InitialCandidate]] = { + val tweetIds = candidates.map(_.tweetId).toSet + Future.collect(tweetInfoStore.multiGet(tweetIds)).map { tweetInfos => + /*** + * If tweetInfo does not exist, we will filter out this tweet candidate. + * This tweetInfo filter also acts as the VF filter + */ + candidates.collect { + case candidate if tweetInfos.getOrElse(candidate.tweetId, None).isDefined => + val tweetInfo = tweetInfos(candidate.tweetId) + .getOrElse(throw new IllegalStateException("Check previous line's condition")) + + InitialCandidate( + tweetId = candidate.tweetId, + tweetInfo = tweetInfo, + candidate.candidateGenerationInfo + ) + } + } + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/candidate_generation/RelatedVideoTweetCandidateGenerator.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/candidate_generation/RelatedVideoTweetCandidateGenerator.scala new file mode 100644 index 0000000000..cc7f558590 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/candidate_generation/RelatedVideoTweetCandidateGenerator.scala @@ -0,0 +1,139 @@ +package com.twitter.cr_mixer.candidate_generation + +import com.twitter.contentrecommender.thriftscala.TweetInfo +import com.twitter.cr_mixer.filter.PreRankFilterRunner +import com.twitter.cr_mixer.model.InitialCandidate +import com.twitter.cr_mixer.model.RelatedVideoTweetCandidateGeneratorQuery +import com.twitter.cr_mixer.model.TweetWithCandidateGenerationInfo +import com.twitter.cr_mixer.model.ModuleNames +import com.twitter.cr_mixer.similarity_engine.StandardSimilarityEngine +import com.twitter.cr_mixer.similarity_engine.TweetBasedUnifiedSimilarityEngine +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.frigate.common.util.StatsUtil +import com.twitter.simclusters_v2.common.TweetId +import com.twitter.simclusters_v2.thriftscala.InternalId +import com.twitter.storehaus.ReadableStore +import com.twitter.timelines.configapi +import com.twitter.util.Future +import javax.inject.Inject +import javax.inject.Named +import javax.inject.Singleton + +@Singleton +class RelatedVideoTweetCandidateGenerator @Inject() ( + @Named(ModuleNames.TweetBasedUnifiedSimilarityEngine) tweetBasedUnifiedSimilarityEngine: StandardSimilarityEngine[ + TweetBasedUnifiedSimilarityEngine.Query, + TweetWithCandidateGenerationInfo + ], + preRankFilterRunner: PreRankFilterRunner, + tweetInfoStore: ReadableStore[TweetId, TweetInfo], + globalStats: StatsReceiver) { + + private val stats: StatsReceiver = globalStats.scope(this.getClass.getCanonicalName) + private val fetchCandidatesStats = stats.scope("fetchCandidates") + private val preRankFilterStats = stats.scope("preRankFilter") + + def get( + query: RelatedVideoTweetCandidateGeneratorQuery + ): Future[Seq[InitialCandidate]] = { + + val allStats = stats.scope("all") + val perProductStats = stats.scope("perProduct", query.product.toString) + StatsUtil.trackItemsStats(allStats) { + StatsUtil.trackItemsStats(perProductStats) { + for { + initialCandidates <- StatsUtil.trackBlockStats(fetchCandidatesStats) { + fetchCandidates(query) + } + filteredCandidates <- StatsUtil.trackBlockStats(preRankFilterStats) { + preRankFilter(query, initialCandidates) + } + } yield { + filteredCandidates.headOption + .getOrElse( + throw new UnsupportedOperationException( + "RelatedVideoTweetCandidateGenerator results invalid") + ).take(query.maxNumResults) + } + } + } + } + + def fetchCandidates( + query: RelatedVideoTweetCandidateGeneratorQuery + ): Future[Seq[Seq[InitialCandidate]]] = { + query.internalId match { + case InternalId.TweetId(_) => + getCandidatesFromSimilarityEngine( + query, + TweetBasedUnifiedSimilarityEngine.fromParamsForRelatedVideoTweet, + tweetBasedUnifiedSimilarityEngine.getCandidates) + case _ => + throw new UnsupportedOperationException( + "RelatedVideoTweetCandidateGenerator gets invalid InternalId") + } + } + + /*** + * fetch Candidates from TweetBased/ProducerBased Unified Similarity Engine, + * and apply VF filter based on TweetInfoStore + * To align with the downstream processing (filter, rank), we tend to return a Seq[Seq[InitialCandidate]] + * instead of a Seq[Candidate] even though we only have a Seq in it. + */ + private def getCandidatesFromSimilarityEngine[QueryType]( + query: RelatedVideoTweetCandidateGeneratorQuery, + fromParamsForRelatedVideoTweet: (InternalId, configapi.Params) => QueryType, + getFunc: QueryType => Future[Option[Seq[TweetWithCandidateGenerationInfo]]] + ): Future[Seq[Seq[InitialCandidate]]] = { + + /*** + * We wrap the query to be a Seq of queries for the Sim Engine to ensure evolvability of candidate generation + * and as a result, it will return Seq[Seq[InitialCandidate]] + */ + val engineQueries = + Seq(fromParamsForRelatedVideoTweet(query.internalId, query.params)) + + Future + .collect { + engineQueries.map { query => + for { + candidates <- getFunc(query) + prefilterCandidates <- convertToInitialCandidates( + candidates.toSeq.flatten + ) + } yield prefilterCandidates + } + } + } + + private def preRankFilter( + query: RelatedVideoTweetCandidateGeneratorQuery, + candidates: Seq[Seq[InitialCandidate]] + ): Future[Seq[Seq[InitialCandidate]]] = { + preRankFilterRunner + .runSequentialFilters(query, candidates) + } + + private[candidate_generation] def convertToInitialCandidates( + candidates: Seq[TweetWithCandidateGenerationInfo], + ): Future[Seq[InitialCandidate]] = { + val tweetIds = candidates.map(_.tweetId).toSet + Future.collect(tweetInfoStore.multiGet(tweetIds)).map { tweetInfos => + /*** + * If tweetInfo does not exist, we will filter out this tweet candidate. + * This tweetInfo filter also acts as the VF filter + */ + candidates.collect { + case candidate if tweetInfos.getOrElse(candidate.tweetId, None).isDefined => + val tweetInfo = tweetInfos(candidate.tweetId) + .getOrElse(throw new IllegalStateException("Check previous line's condition")) + + InitialCandidate( + tweetId = candidate.tweetId, + tweetInfo = tweetInfo, + candidate.candidateGenerationInfo + ) + } + } + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/candidate_generation/SimClustersInterestedInCandidateGeneration.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/candidate_generation/SimClustersInterestedInCandidateGeneration.scala new file mode 100644 index 0000000000..a40901a585 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/candidate_generation/SimClustersInterestedInCandidateGeneration.scala @@ -0,0 +1,640 @@ +package com.twitter.cr_mixer.candidate_generation + +import com.twitter.cr_mixer.model.CandidateGenerationInfo +import com.twitter.cr_mixer.model.TweetWithCandidateGenerationInfo +import com.twitter.cr_mixer.model.TweetWithScore +import com.twitter.cr_mixer.param.GlobalParams +import com.twitter.cr_mixer.param.InterestedInParams +import com.twitter.cr_mixer.param.SimClustersANNParams +import com.twitter.cr_mixer.similarity_engine.EngineQuery +import com.twitter.cr_mixer.similarity_engine.SimClustersANNSimilarityEngine +import com.twitter.cr_mixer.similarity_engine.StandardSimilarityEngine +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.frigate.common.base.CandidateSource +import com.twitter.frigate.common.util.StatsUtil +import com.twitter.simclusters_v2.common.ModelVersions +import com.twitter.simclusters_v2.thriftscala.InternalId +import com.twitter.timelines.configapi +import com.twitter.util.Future +import javax.inject.Inject +import javax.inject.Singleton +import javax.inject.Named +import com.twitter.cr_mixer.model.ModuleNames + +/** + * This store looks for similar tweets for a given UserId that generates UserInterestedIn + * from SimClustersANN. It will be a standalone CandidateGeneration class moving forward. + * + * After the abstraction improvement (apply SimilarityEngine trait) + * these CG will be subjected to change. + */ +@Singleton +case class SimClustersInterestedInCandidateGeneration @Inject() ( + @Named(ModuleNames.SimClustersANNSimilarityEngine) + simClustersANNSimilarityEngine: StandardSimilarityEngine[ + SimClustersANNSimilarityEngine.Query, + TweetWithScore + ], + statsReceiver: StatsReceiver) + extends CandidateSource[ + SimClustersInterestedInCandidateGeneration.Query, + Seq[TweetWithCandidateGenerationInfo] + ] { + + override def name: String = this.getClass.getSimpleName + private val stats = statsReceiver.scope(name) + private val fetchCandidatesStat = stats.scope("fetchCandidates") + + override def get( + query: SimClustersInterestedInCandidateGeneration.Query + ): Future[Option[Seq[Seq[TweetWithCandidateGenerationInfo]]]] = { + + query.internalId match { + case _: InternalId.UserId => + StatsUtil.trackOptionItemsStats(fetchCandidatesStat) { + // UserInterestedIn Queries + val userInterestedInCandidateResultFut = + if (query.enableUserInterestedIn && query.enableProdSimClustersANNSimilarityEngine) + getInterestedInCandidateResult( + simClustersANNSimilarityEngine, + query.interestedInSimClustersANNQuery, + query.simClustersInterestedInMinScore) + else + Future.None + + val userInterestedInExperimentalSANNCandidateResultFut = + if (query.enableUserInterestedIn && query.enableExperimentalSimClustersANNSimilarityEngine) + getInterestedInCandidateResult( + simClustersANNSimilarityEngine, + query.interestedInExperimentalSimClustersANNQuery, + query.simClustersInterestedInMinScore) + else + Future.None + + val userInterestedInSANN1CandidateResultFut = + if (query.enableUserInterestedIn && query.enableSimClustersANN1SimilarityEngine) + getInterestedInCandidateResult( + simClustersANNSimilarityEngine, + query.interestedInSimClustersANN1Query, + query.simClustersInterestedInMinScore) + else + Future.None + + val userInterestedInSANN2CandidateResultFut = + if (query.enableUserInterestedIn && query.enableSimClustersANN2SimilarityEngine) + getInterestedInCandidateResult( + simClustersANNSimilarityEngine, + query.interestedInSimClustersANN2Query, + query.simClustersInterestedInMinScore) + else + Future.None + + val userInterestedInSANN3CandidateResultFut = + if (query.enableUserInterestedIn && query.enableSimClustersANN3SimilarityEngine) + getInterestedInCandidateResult( + simClustersANNSimilarityEngine, + query.interestedInSimClustersANN3Query, + query.simClustersInterestedInMinScore) + else + Future.None + + val userInterestedInSANN5CandidateResultFut = + if (query.enableUserInterestedIn && query.enableSimClustersANN5SimilarityEngine) + getInterestedInCandidateResult( + simClustersANNSimilarityEngine, + query.interestedInSimClustersANN5Query, + query.simClustersInterestedInMinScore) + else + Future.None + + val userInterestedInSANN4CandidateResultFut = + if (query.enableUserInterestedIn && query.enableSimClustersANN4SimilarityEngine) + getInterestedInCandidateResult( + simClustersANNSimilarityEngine, + query.interestedInSimClustersANN4Query, + query.simClustersInterestedInMinScore) + else + Future.None + // UserNextInterestedIn Queries + val userNextInterestedInCandidateResultFut = + if (query.enableUserNextInterestedIn && query.enableProdSimClustersANNSimilarityEngine) + getInterestedInCandidateResult( + simClustersANNSimilarityEngine, + query.nextInterestedInSimClustersANNQuery, + query.simClustersInterestedInMinScore) + else + Future.None + + val userNextInterestedInExperimentalSANNCandidateResultFut = + if (query.enableUserNextInterestedIn && query.enableExperimentalSimClustersANNSimilarityEngine) + getInterestedInCandidateResult( + simClustersANNSimilarityEngine, + query.nextInterestedInExperimentalSimClustersANNQuery, + query.simClustersInterestedInMinScore) + else + Future.None + + val userNextInterestedInSANN1CandidateResultFut = + if (query.enableUserNextInterestedIn && query.enableSimClustersANN1SimilarityEngine) + getInterestedInCandidateResult( + simClustersANNSimilarityEngine, + query.nextInterestedInSimClustersANN1Query, + query.simClustersInterestedInMinScore) + else + Future.None + + val userNextInterestedInSANN2CandidateResultFut = + if (query.enableUserNextInterestedIn && query.enableSimClustersANN2SimilarityEngine) + getInterestedInCandidateResult( + simClustersANNSimilarityEngine, + query.nextInterestedInSimClustersANN2Query, + query.simClustersInterestedInMinScore) + else + Future.None + + val userNextInterestedInSANN3CandidateResultFut = + if (query.enableUserNextInterestedIn && query.enableSimClustersANN3SimilarityEngine) + getInterestedInCandidateResult( + simClustersANNSimilarityEngine, + query.nextInterestedInSimClustersANN3Query, + query.simClustersInterestedInMinScore) + else + Future.None + + val userNextInterestedInSANN5CandidateResultFut = + if (query.enableUserNextInterestedIn && query.enableSimClustersANN5SimilarityEngine) + getInterestedInCandidateResult( + simClustersANNSimilarityEngine, + query.nextInterestedInSimClustersANN5Query, + query.simClustersInterestedInMinScore) + else + Future.None + + val userNextInterestedInSANN4CandidateResultFut = + if (query.enableUserNextInterestedIn && query.enableSimClustersANN4SimilarityEngine) + getInterestedInCandidateResult( + simClustersANNSimilarityEngine, + query.nextInterestedInSimClustersANN4Query, + query.simClustersInterestedInMinScore) + else + Future.None + + // AddressBookInterestedIn Queries + val userAddressBookInterestedInCandidateResultFut = + if (query.enableAddressBookNextInterestedIn && query.enableProdSimClustersANNSimilarityEngine) + getInterestedInCandidateResult( + simClustersANNSimilarityEngine, + query.addressbookInterestedInSimClustersANNQuery, + query.simClustersInterestedInMinScore) + else + Future.None + + val userAddressBookExperimentalSANNCandidateResultFut = + if (query.enableAddressBookNextInterestedIn && query.enableExperimentalSimClustersANNSimilarityEngine) + getInterestedInCandidateResult( + simClustersANNSimilarityEngine, + query.addressbookInterestedInExperimentalSimClustersANNQuery, + query.simClustersInterestedInMinScore) + else + Future.None + + val userAddressBookSANN1CandidateResultFut = + if (query.enableAddressBookNextInterestedIn && query.enableSimClustersANN1SimilarityEngine) + getInterestedInCandidateResult( + simClustersANNSimilarityEngine, + query.addressbookInterestedInSimClustersANN1Query, + query.simClustersInterestedInMinScore) + else + Future.None + + val userAddressBookSANN2CandidateResultFut = + if (query.enableAddressBookNextInterestedIn && query.enableSimClustersANN2SimilarityEngine) + getInterestedInCandidateResult( + simClustersANNSimilarityEngine, + query.addressbookInterestedInSimClustersANN2Query, + query.simClustersInterestedInMinScore) + else + Future.None + + val userAddressBookSANN3CandidateResultFut = + if (query.enableAddressBookNextInterestedIn && query.enableSimClustersANN3SimilarityEngine) + getInterestedInCandidateResult( + simClustersANNSimilarityEngine, + query.addressbookInterestedInSimClustersANN3Query, + query.simClustersInterestedInMinScore) + else + Future.None + + val userAddressBookSANN5CandidateResultFut = + if (query.enableAddressBookNextInterestedIn && query.enableSimClustersANN5SimilarityEngine) + getInterestedInCandidateResult( + simClustersANNSimilarityEngine, + query.addressbookInterestedInSimClustersANN5Query, + query.simClustersInterestedInMinScore) + else + Future.None + + val userAddressBookSANN4CandidateResultFut = + if (query.enableAddressBookNextInterestedIn && query.enableSimClustersANN4SimilarityEngine) + getInterestedInCandidateResult( + simClustersANNSimilarityEngine, + query.addressbookInterestedInSimClustersANN4Query, + query.simClustersInterestedInMinScore) + else + Future.None + + Future + .collect( + Seq( + userInterestedInCandidateResultFut, + userNextInterestedInCandidateResultFut, + userAddressBookInterestedInCandidateResultFut, + userInterestedInExperimentalSANNCandidateResultFut, + userNextInterestedInExperimentalSANNCandidateResultFut, + userAddressBookExperimentalSANNCandidateResultFut, + userInterestedInSANN1CandidateResultFut, + userNextInterestedInSANN1CandidateResultFut, + userAddressBookSANN1CandidateResultFut, + userInterestedInSANN2CandidateResultFut, + userNextInterestedInSANN2CandidateResultFut, + userAddressBookSANN2CandidateResultFut, + userInterestedInSANN3CandidateResultFut, + userNextInterestedInSANN3CandidateResultFut, + userAddressBookSANN3CandidateResultFut, + userInterestedInSANN5CandidateResultFut, + userNextInterestedInSANN5CandidateResultFut, + userAddressBookSANN5CandidateResultFut, + userInterestedInSANN4CandidateResultFut, + userNextInterestedInSANN4CandidateResultFut, + userAddressBookSANN4CandidateResultFut + ) + ).map { candidateResults => + Some( + candidateResults.map(candidateResult => candidateResult.getOrElse(Seq.empty)) + ) + } + } + case _ => + stats.counter("sourceId_is_not_userId_cnt").incr() + Future.None + } + } + + private def simClustersCandidateMinScoreFilter( + simClustersAnnCandidates: Seq[TweetWithScore], + simClustersInterestedInMinScore: Double, + simClustersANNConfigId: String + ): Seq[TweetWithScore] = { + val filteredCandidates = simClustersAnnCandidates + .filter { candidate => + candidate.score > simClustersInterestedInMinScore + } + + stats.stat(simClustersANNConfigId, "simClustersAnnCandidates_size").add(filteredCandidates.size) + stats.counter(simClustersANNConfigId, "simClustersAnnRequests").incr() + if (filteredCandidates.isEmpty) + stats.counter(simClustersANNConfigId, "emptyFilteredSimClustersAnnCandidates").incr() + + filteredCandidates.map { candidate => + TweetWithScore(candidate.tweetId, candidate.score) + } + } + + private def getInterestedInCandidateResult( + simClustersANNSimilarityEngine: StandardSimilarityEngine[ + SimClustersANNSimilarityEngine.Query, + TweetWithScore + ], + simClustersANNQuery: EngineQuery[SimClustersANNSimilarityEngine.Query], + simClustersInterestedInMinScore: Double, + ): Future[Option[Seq[TweetWithCandidateGenerationInfo]]] = { + val interestedInCandidatesFut = + simClustersANNSimilarityEngine.getCandidates(simClustersANNQuery) + + val interestedInCandidateResultFut = interestedInCandidatesFut.map { interestedInCandidates => + stats.stat("candidateSize").add(interestedInCandidates.size) + + val embeddingCandidatesStat = stats.scope( + simClustersANNQuery.storeQuery.simClustersANNQuery.sourceEmbeddingId.embeddingType.name) + + embeddingCandidatesStat.stat("candidateSize").add(interestedInCandidates.size) + if (interestedInCandidates.isEmpty) { + embeddingCandidatesStat.counter("empty_results").incr() + } + embeddingCandidatesStat.counter("requests").incr() + + val filteredTweets = simClustersCandidateMinScoreFilter( + interestedInCandidates.toSeq.flatten, + simClustersInterestedInMinScore, + simClustersANNQuery.storeQuery.simClustersANNConfigId) + + val interestedInTweetsWithCGInfo = filteredTweets.map { tweetWithScore => + TweetWithCandidateGenerationInfo( + tweetWithScore.tweetId, + CandidateGenerationInfo( + None, + SimClustersANNSimilarityEngine + .toSimilarityEngineInfo(simClustersANNQuery, tweetWithScore.score), + Seq.empty // SANN is an atomic SE, and hence it has no contributing SEs + ) + ) + } + + val interestedInResults = if (interestedInTweetsWithCGInfo.nonEmpty) { + Some(interestedInTweetsWithCGInfo) + } else None + interestedInResults + } + interestedInCandidateResultFut + } +} + +object SimClustersInterestedInCandidateGeneration { + + case class Query( + internalId: InternalId, + enableUserInterestedIn: Boolean, + enableUserNextInterestedIn: Boolean, + enableAddressBookNextInterestedIn: Boolean, + enableProdSimClustersANNSimilarityEngine: Boolean, + enableExperimentalSimClustersANNSimilarityEngine: Boolean, + enableSimClustersANN1SimilarityEngine: Boolean, + enableSimClustersANN2SimilarityEngine: Boolean, + enableSimClustersANN3SimilarityEngine: Boolean, + enableSimClustersANN5SimilarityEngine: Boolean, + enableSimClustersANN4SimilarityEngine: Boolean, + simClustersInterestedInMinScore: Double, + simClustersNextInterestedInMinScore: Double, + simClustersAddressBookInterestedInMinScore: Double, + interestedInSimClustersANNQuery: EngineQuery[SimClustersANNSimilarityEngine.Query], + nextInterestedInSimClustersANNQuery: EngineQuery[SimClustersANNSimilarityEngine.Query], + addressbookInterestedInSimClustersANNQuery: EngineQuery[SimClustersANNSimilarityEngine.Query], + interestedInExperimentalSimClustersANNQuery: EngineQuery[SimClustersANNSimilarityEngine.Query], + nextInterestedInExperimentalSimClustersANNQuery: EngineQuery[ + SimClustersANNSimilarityEngine.Query + ], + addressbookInterestedInExperimentalSimClustersANNQuery: EngineQuery[ + SimClustersANNSimilarityEngine.Query + ], + interestedInSimClustersANN1Query: EngineQuery[SimClustersANNSimilarityEngine.Query], + nextInterestedInSimClustersANN1Query: EngineQuery[SimClustersANNSimilarityEngine.Query], + addressbookInterestedInSimClustersANN1Query: EngineQuery[SimClustersANNSimilarityEngine.Query], + interestedInSimClustersANN2Query: EngineQuery[SimClustersANNSimilarityEngine.Query], + nextInterestedInSimClustersANN2Query: EngineQuery[SimClustersANNSimilarityEngine.Query], + addressbookInterestedInSimClustersANN2Query: EngineQuery[SimClustersANNSimilarityEngine.Query], + interestedInSimClustersANN3Query: EngineQuery[SimClustersANNSimilarityEngine.Query], + nextInterestedInSimClustersANN3Query: EngineQuery[SimClustersANNSimilarityEngine.Query], + addressbookInterestedInSimClustersANN3Query: EngineQuery[SimClustersANNSimilarityEngine.Query], + interestedInSimClustersANN5Query: EngineQuery[SimClustersANNSimilarityEngine.Query], + nextInterestedInSimClustersANN5Query: EngineQuery[SimClustersANNSimilarityEngine.Query], + addressbookInterestedInSimClustersANN5Query: EngineQuery[SimClustersANNSimilarityEngine.Query], + interestedInSimClustersANN4Query: EngineQuery[SimClustersANNSimilarityEngine.Query], + nextInterestedInSimClustersANN4Query: EngineQuery[SimClustersANNSimilarityEngine.Query], + addressbookInterestedInSimClustersANN4Query: EngineQuery[SimClustersANNSimilarityEngine.Query], + ) + + def fromParams( + internalId: InternalId, + params: configapi.Params, + ): Query = { + // SimClusters common configs + val simClustersModelVersion = + ModelVersions.Enum.enumToSimClustersModelVersionMap(params(GlobalParams.ModelVersionParam)) + val simClustersANNConfigId = params(SimClustersANNParams.SimClustersANNConfigId) + val experimentalSimClustersANNConfigId = params( + SimClustersANNParams.ExperimentalSimClustersANNConfigId) + val simClustersANN1ConfigId = params(SimClustersANNParams.SimClustersANN1ConfigId) + val simClustersANN2ConfigId = params(SimClustersANNParams.SimClustersANN2ConfigId) + val simClustersANN3ConfigId = params(SimClustersANNParams.SimClustersANN3ConfigId) + val simClustersANN5ConfigId = params(SimClustersANNParams.SimClustersANN5ConfigId) + val simClustersANN4ConfigId = params(SimClustersANNParams.SimClustersANN4ConfigId) + + val simClustersInterestedInMinScore = params(InterestedInParams.MinScoreParam) + val simClustersNextInterestedInMinScore = params( + InterestedInParams.MinScoreSequentialModelParam) + val simClustersAddressBookInterestedInMinScore = params( + InterestedInParams.MinScoreAddressBookParam) + + // InterestedIn embeddings parameters + val interestedInEmbedding = params(InterestedInParams.InterestedInEmbeddingIdParam) + val nextInterestedInEmbedding = params(InterestedInParams.NextInterestedInEmbeddingIdParam) + val addressbookInterestedInEmbedding = params( + InterestedInParams.AddressBookInterestedInEmbeddingIdParam) + + // Prod SimClustersANN Query + val interestedInSimClustersANNQuery = + SimClustersANNSimilarityEngine.fromParams( + internalId, + interestedInEmbedding.embeddingType, + simClustersModelVersion, + simClustersANNConfigId, + params) + + val nextInterestedInSimClustersANNQuery = + SimClustersANNSimilarityEngine.fromParams( + internalId, + nextInterestedInEmbedding.embeddingType, + simClustersModelVersion, + simClustersANNConfigId, + params) + + val addressbookInterestedInSimClustersANNQuery = + SimClustersANNSimilarityEngine.fromParams( + internalId, + addressbookInterestedInEmbedding.embeddingType, + simClustersModelVersion, + simClustersANNConfigId, + params) + + // Experimental SANN cluster Query + val interestedInExperimentalSimClustersANNQuery = + SimClustersANNSimilarityEngine.fromParams( + internalId, + interestedInEmbedding.embeddingType, + simClustersModelVersion, + experimentalSimClustersANNConfigId, + params) + + val nextInterestedInExperimentalSimClustersANNQuery = + SimClustersANNSimilarityEngine.fromParams( + internalId, + nextInterestedInEmbedding.embeddingType, + simClustersModelVersion, + experimentalSimClustersANNConfigId, + params) + + val addressbookInterestedInExperimentalSimClustersANNQuery = + SimClustersANNSimilarityEngine.fromParams( + internalId, + addressbookInterestedInEmbedding.embeddingType, + simClustersModelVersion, + experimentalSimClustersANNConfigId, + params) + + // SimClusters ANN cluster 1 Query + val interestedInSimClustersANN1Query = + SimClustersANNSimilarityEngine.fromParams( + internalId, + interestedInEmbedding.embeddingType, + simClustersModelVersion, + simClustersANN1ConfigId, + params) + + val nextInterestedInSimClustersANN1Query = + SimClustersANNSimilarityEngine.fromParams( + internalId, + nextInterestedInEmbedding.embeddingType, + simClustersModelVersion, + simClustersANN1ConfigId, + params) + + val addressbookInterestedInSimClustersANN1Query = + SimClustersANNSimilarityEngine.fromParams( + internalId, + addressbookInterestedInEmbedding.embeddingType, + simClustersModelVersion, + simClustersANN1ConfigId, + params) + + // SimClusters ANN cluster 2 Query + val interestedInSimClustersANN2Query = + SimClustersANNSimilarityEngine.fromParams( + internalId, + interestedInEmbedding.embeddingType, + simClustersModelVersion, + simClustersANN2ConfigId, + params) + + val nextInterestedInSimClustersANN2Query = + SimClustersANNSimilarityEngine.fromParams( + internalId, + nextInterestedInEmbedding.embeddingType, + simClustersModelVersion, + simClustersANN2ConfigId, + params) + + val addressbookInterestedInSimClustersANN2Query = + SimClustersANNSimilarityEngine.fromParams( + internalId, + addressbookInterestedInEmbedding.embeddingType, + simClustersModelVersion, + simClustersANN2ConfigId, + params) + + // SimClusters ANN cluster 3 Query + val interestedInSimClustersANN3Query = + SimClustersANNSimilarityEngine.fromParams( + internalId, + interestedInEmbedding.embeddingType, + simClustersModelVersion, + simClustersANN3ConfigId, + params) + + val nextInterestedInSimClustersANN3Query = + SimClustersANNSimilarityEngine.fromParams( + internalId, + nextInterestedInEmbedding.embeddingType, + simClustersModelVersion, + simClustersANN3ConfigId, + params) + + val addressbookInterestedInSimClustersANN3Query = + SimClustersANNSimilarityEngine.fromParams( + internalId, + addressbookInterestedInEmbedding.embeddingType, + simClustersModelVersion, + simClustersANN3ConfigId, + params) + + // SimClusters ANN cluster 5 Query + val interestedInSimClustersANN5Query = + SimClustersANNSimilarityEngine.fromParams( + internalId, + interestedInEmbedding.embeddingType, + simClustersModelVersion, + simClustersANN5ConfigId, + params) + // SimClusters ANN cluster 4 Query + val interestedInSimClustersANN4Query = + SimClustersANNSimilarityEngine.fromParams( + internalId, + interestedInEmbedding.embeddingType, + simClustersModelVersion, + simClustersANN4ConfigId, + params) + + val nextInterestedInSimClustersANN5Query = + SimClustersANNSimilarityEngine.fromParams( + internalId, + nextInterestedInEmbedding.embeddingType, + simClustersModelVersion, + simClustersANN5ConfigId, + params) + + val nextInterestedInSimClustersANN4Query = + SimClustersANNSimilarityEngine.fromParams( + internalId, + nextInterestedInEmbedding.embeddingType, + simClustersModelVersion, + simClustersANN4ConfigId, + params) + + val addressbookInterestedInSimClustersANN5Query = + SimClustersANNSimilarityEngine.fromParams( + internalId, + addressbookInterestedInEmbedding.embeddingType, + simClustersModelVersion, + simClustersANN5ConfigId, + params) + + val addressbookInterestedInSimClustersANN4Query = + SimClustersANNSimilarityEngine.fromParams( + internalId, + addressbookInterestedInEmbedding.embeddingType, + simClustersModelVersion, + simClustersANN4ConfigId, + params) + + Query( + internalId = internalId, + enableUserInterestedIn = params(InterestedInParams.EnableSourceParam), + enableUserNextInterestedIn = params(InterestedInParams.EnableSourceSequentialModelParam), + enableAddressBookNextInterestedIn = params(InterestedInParams.EnableSourceAddressBookParam), + enableProdSimClustersANNSimilarityEngine = + params(InterestedInParams.EnableProdSimClustersANNParam), + enableExperimentalSimClustersANNSimilarityEngine = + params(InterestedInParams.EnableExperimentalSimClustersANNParam), + enableSimClustersANN1SimilarityEngine = params(InterestedInParams.EnableSimClustersANN1Param), + enableSimClustersANN2SimilarityEngine = params(InterestedInParams.EnableSimClustersANN2Param), + enableSimClustersANN3SimilarityEngine = params(InterestedInParams.EnableSimClustersANN3Param), + enableSimClustersANN5SimilarityEngine = params(InterestedInParams.EnableSimClustersANN5Param), + enableSimClustersANN4SimilarityEngine = params(InterestedInParams.EnableSimClustersANN4Param), + simClustersInterestedInMinScore = simClustersInterestedInMinScore, + simClustersNextInterestedInMinScore = simClustersNextInterestedInMinScore, + simClustersAddressBookInterestedInMinScore = simClustersAddressBookInterestedInMinScore, + interestedInSimClustersANNQuery = interestedInSimClustersANNQuery, + nextInterestedInSimClustersANNQuery = nextInterestedInSimClustersANNQuery, + addressbookInterestedInSimClustersANNQuery = addressbookInterestedInSimClustersANNQuery, + interestedInExperimentalSimClustersANNQuery = interestedInExperimentalSimClustersANNQuery, + nextInterestedInExperimentalSimClustersANNQuery = + nextInterestedInExperimentalSimClustersANNQuery, + addressbookInterestedInExperimentalSimClustersANNQuery = + addressbookInterestedInExperimentalSimClustersANNQuery, + interestedInSimClustersANN1Query = interestedInSimClustersANN1Query, + nextInterestedInSimClustersANN1Query = nextInterestedInSimClustersANN1Query, + addressbookInterestedInSimClustersANN1Query = addressbookInterestedInSimClustersANN1Query, + interestedInSimClustersANN2Query = interestedInSimClustersANN2Query, + nextInterestedInSimClustersANN2Query = nextInterestedInSimClustersANN2Query, + addressbookInterestedInSimClustersANN2Query = addressbookInterestedInSimClustersANN2Query, + interestedInSimClustersANN3Query = interestedInSimClustersANN3Query, + nextInterestedInSimClustersANN3Query = nextInterestedInSimClustersANN3Query, + addressbookInterestedInSimClustersANN3Query = addressbookInterestedInSimClustersANN3Query, + interestedInSimClustersANN5Query = interestedInSimClustersANN5Query, + nextInterestedInSimClustersANN5Query = nextInterestedInSimClustersANN5Query, + addressbookInterestedInSimClustersANN5Query = addressbookInterestedInSimClustersANN5Query, + interestedInSimClustersANN4Query = interestedInSimClustersANN4Query, + nextInterestedInSimClustersANN4Query = nextInterestedInSimClustersANN4Query, + addressbookInterestedInSimClustersANN4Query = addressbookInterestedInSimClustersANN4Query, + ) + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/candidate_generation/TopicTweetCandidateGenerator.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/candidate_generation/TopicTweetCandidateGenerator.scala new file mode 100644 index 0000000000..690fda482f --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/candidate_generation/TopicTweetCandidateGenerator.scala @@ -0,0 +1,232 @@ +package com.twitter.cr_mixer.candidate_generation + +import com.twitter.contentrecommender.thriftscala.TweetInfo +import com.twitter.cr_mixer.config.TimeoutConfig +import com.twitter.cr_mixer.model.CandidateGenerationInfo +import com.twitter.cr_mixer.model.InitialCandidate +import com.twitter.cr_mixer.model.SimilarityEngineInfo +import com.twitter.cr_mixer.model.TopicTweetCandidateGeneratorQuery +import com.twitter.cr_mixer.model.TopicTweetWithScore +import com.twitter.cr_mixer.param.TopicTweetParams +import com.twitter.cr_mixer.similarity_engine.CertoTopicTweetSimilarityEngine +import com.twitter.cr_mixer.similarity_engine.SkitHighPrecisionTopicTweetSimilarityEngine +import com.twitter.cr_mixer.similarity_engine.SkitTopicTweetSimilarityEngine +import com.twitter.cr_mixer.thriftscala.SimilarityEngineType +import com.twitter.cr_mixer.thriftscala.TopicTweet +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.finagle.util.DefaultTimer +import com.twitter.frigate.common.util.StatsUtil +import com.twitter.servo.util.MemoizingStatsReceiver +import com.twitter.simclusters_v2.common.TweetId +import com.twitter.simclusters_v2.thriftscala.TopicId +import com.twitter.snowflake.id.SnowflakeId +import com.twitter.storehaus.ReadableStore +import com.twitter.util.Duration +import com.twitter.util.Future +import com.twitter.util.Time +import javax.inject.Inject +import javax.inject.Singleton + +/** + * Formerly CrTopic in legacy Content Recommender. This generator finds top Tweets per Topic. + */ +@Singleton +class TopicTweetCandidateGenerator @Inject() ( + certoTopicTweetSimilarityEngine: CertoTopicTweetSimilarityEngine, + skitTopicTweetSimilarityEngine: SkitTopicTweetSimilarityEngine, + skitHighPrecisionTopicTweetSimilarityEngine: SkitHighPrecisionTopicTweetSimilarityEngine, + tweetInfoStore: ReadableStore[TweetId, TweetInfo], + timeoutConfig: TimeoutConfig, + globalStats: StatsReceiver) { + private val timer = DefaultTimer + private val stats: StatsReceiver = globalStats.scope(this.getClass.getCanonicalName) + private val fetchCandidatesStats = stats.scope("fetchCandidates") + private val filterCandidatesStats = stats.scope("filterCandidates") + private val tweetyPieFilteredStats = filterCandidatesStats.stat("tweetypie_filtered") + private val memoizedStatsReceiver = new MemoizingStatsReceiver(stats) + + def get( + query: TopicTweetCandidateGeneratorQuery + ): Future[Map[Long, Seq[TopicTweet]]] = { + val maxTweetAge = query.params(TopicTweetParams.MaxTweetAge) + val product = query.product + val allStats = memoizedStatsReceiver.scope("all") + val perProductStats = memoizedStatsReceiver.scope("perProduct", product.name) + StatsUtil.trackMapValueStats(allStats) { + StatsUtil.trackMapValueStats(perProductStats) { + val result = for { + retrievedTweets <- fetchCandidates(query) + initialTweetCandidates <- convertToInitialCandidates(retrievedTweets) + filteredTweetCandidates <- filterCandidates( + initialTweetCandidates, + maxTweetAge, + query.isVideoOnly, + query.impressedTweetList) + rankedTweetCandidates = rankCandidates(filteredTweetCandidates) + hydratedTweetCandidates = hydrateCandidates(rankedTweetCandidates) + } yield { + hydratedTweetCandidates.map { + case (topicId, topicTweets) => + val topKTweets = topicTweets.take(query.maxNumResults) + topicId -> topKTweets + } + } + result.raiseWithin(timeoutConfig.topicTweetEndpointTimeout)(timer) + } + } + } + + private def fetchCandidates( + query: TopicTweetCandidateGeneratorQuery + ): Future[Map[TopicId, Option[Seq[TopicTweetWithScore]]]] = { + Future.collect { + query.topicIds.map { topicId => + topicId -> StatsUtil.trackOptionStats(fetchCandidatesStats) { + Future + .join( + certoTopicTweetSimilarityEngine.get(CertoTopicTweetSimilarityEngine + .fromParams(topicId, query.isVideoOnly, query.params)), + skitTopicTweetSimilarityEngine + .get(SkitTopicTweetSimilarityEngine + .fromParams(topicId, query.isVideoOnly, query.params)), + skitHighPrecisionTopicTweetSimilarityEngine + .get(SkitHighPrecisionTopicTweetSimilarityEngine + .fromParams(topicId, query.isVideoOnly, query.params)) + ).map { + case (certoTopicTweets, skitTfgTopicTweets, skitHighPrecisionTopicTweets) => + val uniqueCandidates = (certoTopicTweets.getOrElse(Nil) ++ + skitTfgTopicTweets.getOrElse(Nil) ++ + skitHighPrecisionTopicTweets.getOrElse(Nil)) + .groupBy(_.tweetId).map { + case (_, dupCandidates) => dupCandidates.head + }.toSeq + Some(uniqueCandidates) + } + } + }.toMap + } + } + + private def convertToInitialCandidates( + candidatesMap: Map[TopicId, Option[Seq[TopicTweetWithScore]]] + ): Future[Map[TopicId, Seq[InitialCandidate]]] = { + val initialCandidates = candidatesMap.map { + case (topicId, candidatesOpt) => + val candidates = candidatesOpt.getOrElse(Nil) + val tweetIds = candidates.map(_.tweetId).toSet + val numTweetsPreFilter = tweetIds.size + Future.collect(tweetInfoStore.multiGet(tweetIds)).map { tweetInfos => + /** * + * If tweetInfo does not exist, we will filter out this tweet candidate. + */ + val tweetyPieFilteredInitialCandidates = candidates.collect { + case candidate if tweetInfos.getOrElse(candidate.tweetId, None).isDefined => + val tweetInfo = tweetInfos(candidate.tweetId) + .getOrElse(throw new IllegalStateException("Check previous line's condition")) + + InitialCandidate( + tweetId = candidate.tweetId, + tweetInfo = tweetInfo, + CandidateGenerationInfo( + None, + SimilarityEngineInfo( + similarityEngineType = candidate.similarityEngineType, + modelId = None, + score = Some(candidate.score)), + Seq.empty + ) + ) + } + val numTweetsPostFilter = tweetyPieFilteredInitialCandidates.size + tweetyPieFilteredStats.add(numTweetsPreFilter - numTweetsPostFilter) + topicId -> tweetyPieFilteredInitialCandidates + } + } + + Future.collect(initialCandidates.toSeq).map(_.toMap) + } + + private def filterCandidates( + topicTweetMap: Map[TopicId, Seq[InitialCandidate]], + maxTweetAge: Duration, + isVideoOnly: Boolean, + excludeTweetIds: Set[TweetId] + ): Future[Map[TopicId, Seq[InitialCandidate]]] = { + + val earliestTweetId = SnowflakeId.firstIdFor(Time.now - maxTweetAge) + + val filteredResults = topicTweetMap.map { + case (topicId, tweetsWithScore) => + topicId -> StatsUtil.trackItemsStats(filterCandidatesStats) { + + val timeFilteredTweets = + tweetsWithScore.filter { tweetWithScore => + tweetWithScore.tweetId >= earliestTweetId && !excludeTweetIds.contains( + tweetWithScore.tweetId) + } + + filterCandidatesStats + .stat("exclude_and_time_filtered").add(tweetsWithScore.size - timeFilteredTweets.size) + + val tweetNudityFilteredTweets = + timeFilteredTweets.collect { + case tweet if tweet.tweetInfo.isPassTweetMediaNudityTag.contains(true) => tweet + } + + filterCandidatesStats + .stat("tweet_nudity_filtered").add( + timeFilteredTweets.size - tweetNudityFilteredTweets.size) + + val userNudityFilteredTweets = + tweetNudityFilteredTweets.collect { + case tweet if tweet.tweetInfo.isPassUserNudityRateStrict.contains(true) => tweet + } + + filterCandidatesStats + .stat("user_nudity_filtered").add( + tweetNudityFilteredTweets.size - userNudityFilteredTweets.size) + + val videoFilteredTweets = { + if (isVideoOnly) { + userNudityFilteredTweets.collect { + case tweet if tweet.tweetInfo.hasVideo.contains(true) => tweet + } + } else { + userNudityFilteredTweets + } + } + + Future.value(videoFilteredTweets) + } + } + Future.collect(filteredResults) + } + + private def rankCandidates( + tweetCandidatesMap: Map[TopicId, Seq[InitialCandidate]] + ): Map[TopicId, Seq[InitialCandidate]] = { + tweetCandidatesMap.mapValues { tweetCandidates => + tweetCandidates.sortBy { candidate => + -candidate.tweetInfo.favCount + } + } + } + + private def hydrateCandidates( + topicCandidatesMap: Map[TopicId, Seq[InitialCandidate]] + ): Map[Long, Seq[TopicTweet]] = { + topicCandidatesMap.map { + case (topicId, tweetsWithScore) => + topicId.entityId -> + tweetsWithScore.map { tweetWithScore => + val similarityEngineType: SimilarityEngineType = + tweetWithScore.candidateGenerationInfo.similarityEngineInfo.similarityEngineType + TopicTweet( + tweetId = tweetWithScore.tweetId, + score = tweetWithScore.getSimilarityScore, + similarityEngineType = similarityEngineType + ) + } + } + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/candidate_generation/UtegTweetCandidateGenerator.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/candidate_generation/UtegTweetCandidateGenerator.scala new file mode 100644 index 0000000000..ecf0bb98ee --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/candidate_generation/UtegTweetCandidateGenerator.scala @@ -0,0 +1,179 @@ +package com.twitter.cr_mixer.candidate_generation + +import com.twitter.contentrecommender.thriftscala.TweetInfo +import com.twitter.cr_mixer.logging.UtegTweetScribeLogger +import com.twitter.cr_mixer.filter.UtegFilterRunner +import com.twitter.cr_mixer.model.CandidateGenerationInfo +import com.twitter.cr_mixer.model.InitialCandidate +import com.twitter.cr_mixer.model.ModuleNames +import com.twitter.cr_mixer.model.RankedCandidate +import com.twitter.cr_mixer.model.SimilarityEngineInfo +import com.twitter.cr_mixer.model.TweetWithScoreAndSocialProof +import com.twitter.cr_mixer.model.UtegTweetCandidateGeneratorQuery +import com.twitter.cr_mixer.similarity_engine.UserTweetEntityGraphSimilarityEngine +import com.twitter.cr_mixer.similarity_engine.StandardSimilarityEngine +import com.twitter.cr_mixer.source_signal.RealGraphInSourceGraphFetcher +import com.twitter.cr_mixer.source_signal.SourceFetcher.FetcherQuery +import com.twitter.cr_mixer.thriftscala.SimilarityEngineType +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.frigate.common.util.StatsUtil +import com.twitter.simclusters_v2.common.TweetId +import com.twitter.simclusters_v2.common.UserId +import com.twitter.storehaus.ReadableStore +import com.twitter.util.Future +import javax.inject.Inject +import javax.inject.Named +import javax.inject.Singleton + +@Singleton +class UtegTweetCandidateGenerator @Inject() ( + @Named(ModuleNames.UserTweetEntityGraphSimilarityEngine) userTweetEntityGraphSimilarityEngine: StandardSimilarityEngine[ + UserTweetEntityGraphSimilarityEngine.Query, + TweetWithScoreAndSocialProof + ], + utegTweetScribeLogger: UtegTweetScribeLogger, + tweetInfoStore: ReadableStore[TweetId, TweetInfo], + realGraphInSourceGraphFetcher: RealGraphInSourceGraphFetcher, + utegFilterRunner: UtegFilterRunner, + globalStats: StatsReceiver) { + + private val stats: StatsReceiver = globalStats.scope(this.getClass.getCanonicalName) + private val fetchSeedsStats = stats.scope("fetchSeeds") + private val fetchCandidatesStats = stats.scope("fetchCandidates") + private val utegFilterStats = stats.scope("utegFilter") + private val rankStats = stats.scope("rank") + + def get( + query: UtegTweetCandidateGeneratorQuery + ): Future[Seq[TweetWithScoreAndSocialProof]] = { + + val allStats = stats.scope("all") + val perProductStats = stats.scope("perProduct", query.product.toString) + StatsUtil.trackItemsStats(allStats) { + StatsUtil.trackItemsStats(perProductStats) { + + /** + * The candidate we return in the end needs a social proof field, which isn't + * supported by the any existing Candidate type, so we created TweetWithScoreAndSocialProof + * instead. + * + * However, filters and light ranker expect Candidate-typed param to work. In order to minimise the + * changes to them, we are doing conversions from/to TweetWithScoreAndSocialProof to/from Candidate + * in this method. + */ + for { + realGraphSeeds <- StatsUtil.trackItemMapStats(fetchSeedsStats) { + fetchSeeds(query) + } + initialTweets <- StatsUtil.trackItemsStats(fetchCandidatesStats) { + fetchCandidates(query, realGraphSeeds) + } + initialCandidates <- convertToInitialCandidates(initialTweets) + filteredCandidates <- StatsUtil.trackItemsStats(utegFilterStats) { + utegFilter(query, initialCandidates) + } + rankedCandidates <- StatsUtil.trackItemsStats(rankStats) { + rankCandidates(query, filteredCandidates) + } + } yield { + val topTweets = rankedCandidates.take(query.maxNumResults) + convertToTweets(topTweets, initialTweets.map(tweet => tweet.tweetId -> tweet).toMap) + } + } + } + } + + private def utegFilter( + query: UtegTweetCandidateGeneratorQuery, + candidates: Seq[InitialCandidate] + ): Future[Seq[InitialCandidate]] = { + utegFilterRunner.runSequentialFilters(query, Seq(candidates)).map(_.flatten) + } + + private def fetchSeeds( + query: UtegTweetCandidateGeneratorQuery + ): Future[Map[UserId, Double]] = { + realGraphInSourceGraphFetcher + .get(FetcherQuery(query.userId, query.product, query.userState, query.params)) + .map(_.map(_.seedWithScores).getOrElse(Map.empty)) + } + + private[candidate_generation] def rankCandidates( + query: UtegTweetCandidateGeneratorQuery, + filteredCandidates: Seq[InitialCandidate], + ): Future[Seq[RankedCandidate]] = { + val blendedCandidates = filteredCandidates.map(candidate => + candidate.toBlendedCandidate(Seq(candidate.candidateGenerationInfo))) + + Future( + blendedCandidates.map { candidate => + val score = candidate.getSimilarityScore + candidate.toRankedCandidate(score) + } + ) + + } + + def fetchCandidates( + query: UtegTweetCandidateGeneratorQuery, + realGraphSeeds: Map[UserId, Double], + ): Future[Seq[TweetWithScoreAndSocialProof]] = { + val engineQuery = UserTweetEntityGraphSimilarityEngine.fromParams( + query.userId, + realGraphSeeds, + Some(query.impressedTweetList.toSeq), + query.params + ) + + utegTweetScribeLogger.scribeInitialCandidates( + query, + userTweetEntityGraphSimilarityEngine.getCandidates(engineQuery).map(_.toSeq.flatten) + ) + } + + private[candidate_generation] def convertToInitialCandidates( + candidates: Seq[TweetWithScoreAndSocialProof], + ): Future[Seq[InitialCandidate]] = { + val tweetIds = candidates.map(_.tweetId).toSet + Future.collect(tweetInfoStore.multiGet(tweetIds)).map { tweetInfos => + /** * + * If tweetInfo does not exist, we will filter out this tweet candidate. + */ + candidates.collect { + case candidate if tweetInfos.getOrElse(candidate.tweetId, None).isDefined => + val tweetInfo = tweetInfos(candidate.tweetId) + .getOrElse(throw new IllegalStateException("Check previous line's condition")) + + InitialCandidate( + tweetId = candidate.tweetId, + tweetInfo = tweetInfo, + CandidateGenerationInfo( + None, + SimilarityEngineInfo( + similarityEngineType = SimilarityEngineType.Uteg, + modelId = None, + score = Some(candidate.score)), + Seq.empty + ) + ) + } + } + } + + private[candidate_generation] def convertToTweets( + candidates: Seq[RankedCandidate], + tweetMap: Map[TweetId, TweetWithScoreAndSocialProof] + ): Seq[TweetWithScoreAndSocialProof] = { + candidates.map { candidate => + tweetMap + .get(candidate.tweetId).map { tweet => + TweetWithScoreAndSocialProof( + tweet.tweetId, + candidate.predictionScore, + tweet.socialProofByType + ) + // The exception should never be thrown + }.getOrElse(throw new Exception("Cannot find ranked candidate in original UTEG tweets")) + } + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/config/BUILD b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/config/BUILD new file mode 100644 index 0000000000..11b558321c --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/config/BUILD @@ -0,0 +1,13 @@ +scala_library( + sources = ["*.scala"], + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/javax/inject:javax.inject", + "configapi/configapi-core", + "cr-mixer/server/src/main/scala/com/twitter/cr_mixer/exception", + "cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param", + "finatra/inject/inject-core/src/main/scala", + "simclusters-ann/thrift/src/main/thrift:thrift-scala", + "src/thrift/com/twitter/simclusters_v2:simclusters_v2-thrift-scala", + ], +) diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/config/SimClustersANNConfig.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/config/SimClustersANNConfig.scala new file mode 100644 index 0000000000..dbf3ad6fdd --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/config/SimClustersANNConfig.scala @@ -0,0 +1,473 @@ +package com.twitter.cr_mixer.config + +import com.twitter.conversions.DurationOps._ +import com.twitter.cr_mixer.exception.InvalidSANNConfigException +import com.twitter.simclusters_v2.thriftscala.EmbeddingType +import com.twitter.simclustersann.thriftscala.ScoringAlgorithm +import com.twitter.simclustersann.thriftscala.{SimClustersANNConfig => ThriftSimClustersANNConfig} +import com.twitter.util.Duration + +case class SimClustersANNConfig( + maxNumResults: Int, + minScore: Double, + candidateEmbeddingType: EmbeddingType, + maxTopTweetsPerCluster: Int, + maxScanClusters: Int, + maxTweetCandidateAge: Duration, + minTweetCandidateAge: Duration, + annAlgorithm: ScoringAlgorithm) { + val toSANNConfigThrift: ThriftSimClustersANNConfig = ThriftSimClustersANNConfig( + maxNumResults = maxNumResults, + minScore = minScore, + candidateEmbeddingType = candidateEmbeddingType, + maxTopTweetsPerCluster = maxTopTweetsPerCluster, + maxScanClusters = maxScanClusters, + maxTweetCandidateAgeHours = maxTweetCandidateAge.inHours, + minTweetCandidateAgeHours = minTweetCandidateAge.inHours, + annAlgorithm = annAlgorithm, + ) +} + +object SimClustersANNConfig { + + final val DefaultConfig = SimClustersANNConfig( + maxNumResults = 200, + minScore = 0.0, + candidateEmbeddingType = EmbeddingType.LogFavBasedTweet, + maxTopTweetsPerCluster = 800, + maxScanClusters = 50, + maxTweetCandidateAge = 24.hours, + minTweetCandidateAge = 0.hours, + annAlgorithm = ScoringAlgorithm.CosineSimilarity, + ) + + /* + SimClustersANNConfigId: String + Format: Prod - “EmbeddingType_ModelVersion_Default” + Format: Experiment - “EmbeddingType_ModelVersion_Date_Two-Digit-Serial-Number”. Date : YYYYMMDD + */ + + private val FavBasedProducer_Model20m145k2020_Default = DefaultConfig.copy() + + // Chunnan's exp on maxTweetCandidateAgeDays 2 + private val FavBasedProducer_Model20m145k2020_20220617_06 = + FavBasedProducer_Model20m145k2020_Default.copy( + maxTweetCandidateAge = 48.hours, + ) + + // Experimental SANN config + private val FavBasedProducer_Model20m145k2020_20220801 = + FavBasedProducer_Model20m145k2020_Default.copy( + candidateEmbeddingType = EmbeddingType.VideoPlayBack50LogFavBasedTweet, + ) + + // SANN-1 config + private val FavBasedProducer_Model20m145k2020_20220810 = + FavBasedProducer_Model20m145k2020_Default.copy( + maxNumResults = 100, + candidateEmbeddingType = EmbeddingType.LogFavBasedAdsTweet, + maxTweetCandidateAge = 175200.hours, + maxTopTweetsPerCluster = 1600 + ) + + // SANN-2 config + private val FavBasedProducer_Model20m145k2020_20220818 = + FavBasedProducer_Model20m145k2020_Default.copy( + maxNumResults = 100, + candidateEmbeddingType = EmbeddingType.LogFavClickBasedAdsTweet, + maxTweetCandidateAge = 175200.hours, + maxTopTweetsPerCluster = 1600 + ) + + // SANN-3 config + private val FavBasedProducer_Model20m145k2020_20220819 = + FavBasedProducer_Model20m145k2020_Default.copy( + candidateEmbeddingType = EmbeddingType.PushOpenLogFavBasedTweet, + ) + + // SANN-5 config + private val FavBasedProducer_Model20m145k2020_20221221 = + FavBasedProducer_Model20m145k2020_Default.copy( + candidateEmbeddingType = EmbeddingType.LogFavBasedRealTimeTweet, + maxTweetCandidateAge = 1.hours + ) + + // SANN-4 config + private val FavBasedProducer_Model20m145k2020_20221220 = + FavBasedProducer_Model20m145k2020_Default.copy( + candidateEmbeddingType = EmbeddingType.LogFavBasedEvergreenTweet, + maxTweetCandidateAge = 48.hours + ) + private val LogFavLongestL2EmbeddingTweet_Model20m145k2020_Default = DefaultConfig.copy() + + // Chunnan's exp on maxTweetCandidateAgeDays 2 + private val LogFavLongestL2EmbeddingTweet_Model20m145k2020_20220617_06 = + LogFavLongestL2EmbeddingTweet_Model20m145k2020_Default.copy( + maxTweetCandidateAge = 48.hours, + ) + + // Experimental SANN config + private val LogFavLongestL2EmbeddingTweet_Model20m145k2020_20220801 = + LogFavLongestL2EmbeddingTweet_Model20m145k2020_Default.copy( + candidateEmbeddingType = EmbeddingType.VideoPlayBack50LogFavBasedTweet, + ) + + // SANN-1 config + private val LogFavLongestL2EmbeddingTweet_Model20m145k2020_20220810 = + LogFavLongestL2EmbeddingTweet_Model20m145k2020_Default.copy( + maxNumResults = 100, + candidateEmbeddingType = EmbeddingType.LogFavBasedAdsTweet, + maxTweetCandidateAge = 175200.hours, + maxTopTweetsPerCluster = 1600 + ) + + // SANN-2 config + private val LogFavLongestL2EmbeddingTweet_Model20m145k2020_20220818 = + LogFavLongestL2EmbeddingTweet_Model20m145k2020_Default.copy( + maxNumResults = 100, + candidateEmbeddingType = EmbeddingType.LogFavClickBasedAdsTweet, + maxTweetCandidateAge = 175200.hours, + maxTopTweetsPerCluster = 1600 + ) + + // SANN-3 config + private val LogFavLongestL2EmbeddingTweet_Model20m145k2020_20220819 = + LogFavLongestL2EmbeddingTweet_Model20m145k2020_Default.copy( + candidateEmbeddingType = EmbeddingType.PushOpenLogFavBasedTweet, + ) + + // SANN-5 config + private val LogFavLongestL2EmbeddingTweet_Model20m145k2020_20221221 = + LogFavLongestL2EmbeddingTweet_Model20m145k2020_Default.copy( + candidateEmbeddingType = EmbeddingType.LogFavBasedRealTimeTweet, + maxTweetCandidateAge = 1.hours + ) + // SANN-4 config + private val LogFavLongestL2EmbeddingTweet_Model20m145k2020_20221220 = + LogFavLongestL2EmbeddingTweet_Model20m145k2020_Default.copy( + candidateEmbeddingType = EmbeddingType.LogFavBasedEvergreenTweet, + maxTweetCandidateAge = 48.hours + ) + private val UnfilteredUserInterestedIn_Model20m145k2020_Default = DefaultConfig.copy() + + // Chunnan's exp on maxTweetCandidateAgeDays 2 + private val UnfilteredUserInterestedIn_Model20m145k2020_20220617_06 = + UnfilteredUserInterestedIn_Model20m145k2020_Default.copy( + maxTweetCandidateAge = 48.hours, + ) + + // Experimental SANN config + private val UnfilteredUserInterestedIn_Model20m145k2020_20220801 = + UnfilteredUserInterestedIn_Model20m145k2020_20220617_06.copy( + candidateEmbeddingType = EmbeddingType.VideoPlayBack50LogFavBasedTweet, + ) + + // SANN-1 config + private val UnfilteredUserInterestedIn_Model20m145k2020_20220810 = + UnfilteredUserInterestedIn_Model20m145k2020_Default.copy( + maxNumResults = 100, + candidateEmbeddingType = EmbeddingType.LogFavBasedAdsTweet, + maxTweetCandidateAge = 175200.hours, + maxTopTweetsPerCluster = 1600 + ) + + // SANN-2 config + private val UnfilteredUserInterestedIn_Model20m145k2020_20220818 = + UnfilteredUserInterestedIn_Model20m145k2020_Default.copy( + maxNumResults = 100, + candidateEmbeddingType = EmbeddingType.LogFavClickBasedAdsTweet, + maxTweetCandidateAge = 175200.hours, + maxTopTweetsPerCluster = 1600 + ) + + // SANN-3 config + private val UnfilteredUserInterestedIn_Model20m145k2020_20220819 = + UnfilteredUserInterestedIn_Model20m145k2020_Default.copy( + candidateEmbeddingType = EmbeddingType.PushOpenLogFavBasedTweet, + ) + + // SANN-5 config + private val UnfilteredUserInterestedIn_Model20m145k2020_20221221 = + UnfilteredUserInterestedIn_Model20m145k2020_Default.copy( + candidateEmbeddingType = EmbeddingType.LogFavBasedRealTimeTweet, + maxTweetCandidateAge = 1.hours + ) + + // SANN-4 config + private val UnfilteredUserInterestedIn_Model20m145k2020_20221220 = + UnfilteredUserInterestedIn_Model20m145k2020_Default.copy( + candidateEmbeddingType = EmbeddingType.LogFavBasedEvergreenTweet, + maxTweetCandidateAge = 48.hours + ) + private val LogFavBasedUserInterestedInFromAPE_Model20m145k2020_Default = DefaultConfig.copy() + + // Chunnan's exp on maxTweetCandidateAgeDays 2 + private val LogFavBasedUserInterestedInFromAPE_Model20m145k2020_20220617_06 = + LogFavBasedUserInterestedInFromAPE_Model20m145k2020_Default.copy( + maxTweetCandidateAge = 48.hours, + ) + + // Experimental SANN config + private val LogFavBasedUserInterestedInFromAPE_Model20m145k2020_20220801 = + LogFavBasedUserInterestedInFromAPE_Model20m145k2020_Default.copy( + candidateEmbeddingType = EmbeddingType.VideoPlayBack50LogFavBasedTweet, + ) + + // SANN-1 config + private val LogFavBasedUserInterestedInFromAPE_Model20m145k2020_20220810 = + LogFavBasedUserInterestedInFromAPE_Model20m145k2020_Default.copy( + maxNumResults = 100, + candidateEmbeddingType = EmbeddingType.LogFavBasedAdsTweet, + maxTweetCandidateAge = 175200.hours, + maxTopTweetsPerCluster = 1600 + ) + + // SANN-2 config + private val LogFavBasedUserInterestedInFromAPE_Model20m145k2020_20220818 = + LogFavBasedUserInterestedInFromAPE_Model20m145k2020_Default.copy( + maxNumResults = 100, + candidateEmbeddingType = EmbeddingType.LogFavClickBasedAdsTweet, + maxTweetCandidateAge = 175200.hours, + maxTopTweetsPerCluster = 1600 + ) + + // SANN-3 config + private val LogFavBasedUserInterestedInFromAPE_Model20m145k2020_20220819 = + LogFavBasedUserInterestedInFromAPE_Model20m145k2020_Default.copy( + candidateEmbeddingType = EmbeddingType.PushOpenLogFavBasedTweet, + ) + + // SANN-5 config + private val LogFavBasedUserInterestedInFromAPE_Model20m145k2020_20221221 = + LogFavBasedUserInterestedInFromAPE_Model20m145k2020_Default.copy( + candidateEmbeddingType = EmbeddingType.LogFavBasedRealTimeTweet, + maxTweetCandidateAge = 1.hours + ) + + // SANN-4 config + private val LogFavBasedUserInterestedInFromAPE_Model20m145k2020_20221220 = + LogFavBasedUserInterestedInFromAPE_Model20m145k2020_Default.copy( + candidateEmbeddingType = EmbeddingType.LogFavBasedEvergreenTweet, + maxTweetCandidateAge = 48.hours + ) + private val LogFavBasedUserInterestedLouvainMaxpoolingAddressBookFromIIAPE_Model20m145k2020_Default = + DefaultConfig.copy() + + // Chunnan's exp on maxTweetCandidateAgeDays 2 + private val LogFavBasedUserInterestedLouvainMaxpoolingAddressBookFromIIAPE_Model20m145k2020_20220617_06 = + LogFavBasedUserInterestedLouvainMaxpoolingAddressBookFromIIAPE_Model20m145k2020_Default.copy( + maxTweetCandidateAge = 48.hours, + ) + + // Experimental SANN config + private val LogFavBasedUserInterestedLouvainMaxpoolingAddressBookFromIIAPE_Model20m145k2020_20220801 = + LogFavBasedUserInterestedLouvainMaxpoolingAddressBookFromIIAPE_Model20m145k2020_Default.copy( + candidateEmbeddingType = EmbeddingType.VideoPlayBack50LogFavBasedTweet, + ) + + // SANN-1 config + private val LogFavBasedUserInterestedLouvainMaxpoolingAddressBookFromIIAPE_Model20m145k2020_20220810 = + LogFavBasedUserInterestedLouvainMaxpoolingAddressBookFromIIAPE_Model20m145k2020_Default.copy( + maxNumResults = 100, + candidateEmbeddingType = EmbeddingType.LogFavBasedAdsTweet, + maxTweetCandidateAge = 175200.hours, + maxTopTweetsPerCluster = 1600 + ) + + // SANN-2 config + private val LogFavBasedUserInterestedLouvainMaxpoolingAddressBookFromIIAPE_Model20m145k2020_20220818 = + LogFavBasedUserInterestedLouvainMaxpoolingAddressBookFromIIAPE_Model20m145k2020_Default.copy( + maxNumResults = 100, + candidateEmbeddingType = EmbeddingType.LogFavClickBasedAdsTweet, + maxTweetCandidateAge = 175200.hours, + maxTopTweetsPerCluster = 1600 + ) + + // SANN-3 config + private val LogFavBasedUserInterestedLouvainMaxpoolingAddressBookFromIIAPE_Model20m145k2020_20220819 = + LogFavBasedUserInterestedLouvainMaxpoolingAddressBookFromIIAPE_Model20m145k2020_Default.copy( + candidateEmbeddingType = EmbeddingType.PushOpenLogFavBasedTweet, + ) + + // SANN-5 config + private val LogFavBasedUserInterestedLouvainMaxpoolingAddressBookFromIIAPE_Model20m145k2020_20221221 = + LogFavBasedUserInterestedLouvainMaxpoolingAddressBookFromIIAPE_Model20m145k2020_Default.copy( + candidateEmbeddingType = EmbeddingType.LogFavBasedRealTimeTweet, + maxTweetCandidateAge = 1.hours + ) + + // SANN-4 config + private val LogFavBasedUserInterestedLouvainMaxpoolingAddressBookFromIIAPE_Model20m145k2020_20221220 = + LogFavBasedUserInterestedLouvainMaxpoolingAddressBookFromIIAPE_Model20m145k2020_Default.copy( + candidateEmbeddingType = EmbeddingType.LogFavBasedEvergreenTweet, + maxTweetCandidateAge = 48.hours + ) + private val UserNextInterestedIn_Model20m145k2020_Default = DefaultConfig.copy() + + // Chunnan's exp on maxTweetCandidateAgeDays 2 + private val UserNextInterestedIn_Model20m145k2020_20220617_06 = + UserNextInterestedIn_Model20m145k2020_Default.copy( + maxTweetCandidateAge = 48.hours, + ) + + // Experimental SANN config + private val UserNextInterestedIn_Model20m145k2020_20220801 = + UserNextInterestedIn_Model20m145k2020_Default.copy( + candidateEmbeddingType = EmbeddingType.VideoPlayBack50LogFavBasedTweet, + ) + + // SANN-1 config + private val UserNextInterestedIn_Model20m145k2020_20220810 = + UserNextInterestedIn_Model20m145k2020_Default.copy( + maxNumResults = 100, + candidateEmbeddingType = EmbeddingType.LogFavBasedAdsTweet, + maxTweetCandidateAge = 175200.hours, + maxTopTweetsPerCluster = 1600 + ) + + // SANN-2 config + private val UserNextInterestedIn_Model20m145k2020_20220818 = + UserNextInterestedIn_Model20m145k2020_Default.copy( + maxNumResults = 100, + candidateEmbeddingType = EmbeddingType.LogFavClickBasedAdsTweet, + maxTweetCandidateAge = 175200.hours, + maxTopTweetsPerCluster = 1600 + ) + + // SANN-3 config + private val UserNextInterestedIn_Model20m145k2020_20220819 = + UserNextInterestedIn_Model20m145k2020_Default.copy( + candidateEmbeddingType = EmbeddingType.PushOpenLogFavBasedTweet, + ) + + // SANN-5 config + private val UserNextInterestedIn_Model20m145k2020_20221221 = + UserNextInterestedIn_Model20m145k2020_Default.copy( + candidateEmbeddingType = EmbeddingType.LogFavBasedRealTimeTweet, + maxTweetCandidateAge = 1.hours + ) + + // SANN-4 config + private val UserNextInterestedIn_Model20m145k2020_20221220 = + UserNextInterestedIn_Model20m145k2020_Default.copy( + candidateEmbeddingType = EmbeddingType.LogFavBasedEvergreenTweet, + maxTweetCandidateAge = 48.hours + ) + // Vincent's experiment on using FollowBasedProducer as query embedding type for UserFollow + private val FollowBasedProducer_Model20m145k2020_Default = + FavBasedProducer_Model20m145k2020_Default.copy() + + // Experimental SANN config + private val FollowBasedProducer_Model20m145k2020_20220801 = + FavBasedProducer_Model20m145k2020_Default.copy( + candidateEmbeddingType = EmbeddingType.VideoPlayBack50LogFavBasedTweet, + ) + + // SANN-1 config + private val FollowBasedProducer_Model20m145k2020_20220810 = + FavBasedProducer_Model20m145k2020_Default.copy( + maxNumResults = 100, + candidateEmbeddingType = EmbeddingType.LogFavBasedAdsTweet, + maxTweetCandidateAge = 175200.hours, + maxTopTweetsPerCluster = 1600 + ) + + // SANN-2 config + private val FollowBasedProducer_Model20m145k2020_20220818 = + FavBasedProducer_Model20m145k2020_Default.copy( + maxNumResults = 100, + candidateEmbeddingType = EmbeddingType.LogFavClickBasedAdsTweet, + maxTweetCandidateAge = 175200.hours, + maxTopTweetsPerCluster = 1600 + ) + + // SANN-3 config + private val FollowBasedProducer_Model20m145k2020_20220819 = + FavBasedProducer_Model20m145k2020_Default.copy( + candidateEmbeddingType = EmbeddingType.PushOpenLogFavBasedTweet, + ) + + // SANN-5 config + private val FollowBasedProducer_Model20m145k2020_20221221 = + FavBasedProducer_Model20m145k2020_Default.copy( + candidateEmbeddingType = EmbeddingType.LogFavBasedRealTimeTweet, + maxTweetCandidateAge = 1.hours + ) + + // SANN-4 config + private val FollowBasedProducer_Model20m145k2020_20221220 = + FavBasedProducer_Model20m145k2020_Default.copy( + candidateEmbeddingType = EmbeddingType.LogFavBasedEvergreenTweet, + maxTweetCandidateAge = 48.hours + ) + val DefaultConfigMappings: Map[String, SimClustersANNConfig] = Map( + "FavBasedProducer_Model20m145k2020_Default" -> FavBasedProducer_Model20m145k2020_Default, + "FavBasedProducer_Model20m145k2020_20220617_06" -> FavBasedProducer_Model20m145k2020_20220617_06, + "FavBasedProducer_Model20m145k2020_20220801" -> FavBasedProducer_Model20m145k2020_20220801, + "FavBasedProducer_Model20m145k2020_20220810" -> FavBasedProducer_Model20m145k2020_20220810, + "FavBasedProducer_Model20m145k2020_20220818" -> FavBasedProducer_Model20m145k2020_20220818, + "FavBasedProducer_Model20m145k2020_20220819" -> FavBasedProducer_Model20m145k2020_20220819, + "FavBasedProducer_Model20m145k2020_20221221" -> FavBasedProducer_Model20m145k2020_20221221, + "FavBasedProducer_Model20m145k2020_20221220" -> FavBasedProducer_Model20m145k2020_20221220, + "FollowBasedProducer_Model20m145k2020_Default" -> FollowBasedProducer_Model20m145k2020_Default, + "FollowBasedProducer_Model20m145k2020_20220801" -> FollowBasedProducer_Model20m145k2020_20220801, + "FollowBasedProducer_Model20m145k2020_20220810" -> FollowBasedProducer_Model20m145k2020_20220810, + "FollowBasedProducer_Model20m145k2020_20220818" -> FollowBasedProducer_Model20m145k2020_20220818, + "FollowBasedProducer_Model20m145k2020_20220819" -> FollowBasedProducer_Model20m145k2020_20220819, + "FollowBasedProducer_Model20m145k2020_20221221" -> FollowBasedProducer_Model20m145k2020_20221221, + "FollowBasedProducer_Model20m145k2020_20221220" -> FollowBasedProducer_Model20m145k2020_20221220, + "LogFavLongestL2EmbeddingTweet_Model20m145k2020_Default" -> LogFavLongestL2EmbeddingTweet_Model20m145k2020_Default, + "LogFavLongestL2EmbeddingTweet_Model20m145k2020_20220617_06" -> LogFavLongestL2EmbeddingTweet_Model20m145k2020_20220617_06, + "LogFavLongestL2EmbeddingTweet_Model20m145k2020_20220801" -> LogFavLongestL2EmbeddingTweet_Model20m145k2020_20220801, + "LogFavLongestL2EmbeddingTweet_Model20m145k2020_20220810" -> LogFavLongestL2EmbeddingTweet_Model20m145k2020_20220810, + "LogFavLongestL2EmbeddingTweet_Model20m145k2020_20220818" -> LogFavLongestL2EmbeddingTweet_Model20m145k2020_20220818, + "LogFavLongestL2EmbeddingTweet_Model20m145k2020_20220819" -> LogFavLongestL2EmbeddingTweet_Model20m145k2020_20220819, + "LogFavLongestL2EmbeddingTweet_Model20m145k2020_20221221" -> LogFavLongestL2EmbeddingTweet_Model20m145k2020_20221221, + "LogFavLongestL2EmbeddingTweet_Model20m145k2020_20221220" -> LogFavLongestL2EmbeddingTweet_Model20m145k2020_20221220, + "UnfilteredUserInterestedIn_Model20m145k2020_Default" -> UnfilteredUserInterestedIn_Model20m145k2020_Default, + "UnfilteredUserInterestedIn_Model20m145k2020_20220617_06" -> UnfilteredUserInterestedIn_Model20m145k2020_20220617_06, + "UnfilteredUserInterestedIn_Model20m145k2020_20220801" -> UnfilteredUserInterestedIn_Model20m145k2020_20220801, + "UnfilteredUserInterestedIn_Model20m145k2020_20220810" -> UnfilteredUserInterestedIn_Model20m145k2020_20220810, + "UnfilteredUserInterestedIn_Model20m145k2020_20220818" -> UnfilteredUserInterestedIn_Model20m145k2020_20220818, + "UnfilteredUserInterestedIn_Model20m145k2020_20220819" -> UnfilteredUserInterestedIn_Model20m145k2020_20220819, + "UnfilteredUserInterestedIn_Model20m145k2020_20221221" -> UnfilteredUserInterestedIn_Model20m145k2020_20221221, + "UnfilteredUserInterestedIn_Model20m145k2020_20221220" -> UnfilteredUserInterestedIn_Model20m145k2020_20221220, + "LogFavBasedUserInterestedInFromAPE_Model20m145k2020_Default" -> LogFavBasedUserInterestedInFromAPE_Model20m145k2020_Default, + "LogFavBasedUserInterestedInFromAPE_Model20m145k2020_20220617_06" -> LogFavBasedUserInterestedInFromAPE_Model20m145k2020_20220617_06, + "LogFavBasedUserInterestedInFromAPE_Model20m145k2020_20220801" -> LogFavBasedUserInterestedInFromAPE_Model20m145k2020_20220801, + "LogFavBasedUserInterestedInFromAPE_Model20m145k2020_20220810" -> LogFavBasedUserInterestedInFromAPE_Model20m145k2020_20220810, + "LogFavBasedUserInterestedInFromAPE_Model20m145k2020_20220818" -> LogFavBasedUserInterestedInFromAPE_Model20m145k2020_20220818, + "LogFavBasedUserInterestedInFromAPE_Model20m145k2020_20220819" -> LogFavBasedUserInterestedInFromAPE_Model20m145k2020_20220819, + "LogFavBasedUserInterestedInFromAPE_Model20m145k2020_20221221" -> LogFavBasedUserInterestedInFromAPE_Model20m145k2020_20221221, + "LogFavBasedUserInterestedInFromAPE_Model20m145k2020_20221220" -> LogFavBasedUserInterestedInFromAPE_Model20m145k2020_20221220, + "LogFavBasedUserInterestedLouvainMaxpoolingAddressBookFromIIAPE_Model20m145k2020_Default" -> LogFavBasedUserInterestedLouvainMaxpoolingAddressBookFromIIAPE_Model20m145k2020_Default, + "LogFavBasedUserInterestedLouvainMaxpoolingAddressBookFromIIAPE_Model20m145k2020_20220617_06" -> LogFavBasedUserInterestedLouvainMaxpoolingAddressBookFromIIAPE_Model20m145k2020_20220617_06, + "LogFavBasedUserInterestedLouvainMaxpoolingAddressBookFromIIAPE_Model20m145k2020_20220801" -> LogFavBasedUserInterestedLouvainMaxpoolingAddressBookFromIIAPE_Model20m145k2020_20220801, + "LogFavBasedUserInterestedLouvainMaxpoolingAddressBookFromIIAPE_Model20m145k2020_20220810" -> LogFavBasedUserInterestedLouvainMaxpoolingAddressBookFromIIAPE_Model20m145k2020_20220810, + "LogFavBasedUserInterestedLouvainMaxpoolingAddressBookFromIIAPE_Model20m145k2020_20220818" -> LogFavBasedUserInterestedLouvainMaxpoolingAddressBookFromIIAPE_Model20m145k2020_20220818, + "LogFavBasedUserInterestedLouvainMaxpoolingAddressBookFromIIAPE_Model20m145k2020_20220819" -> LogFavBasedUserInterestedLouvainMaxpoolingAddressBookFromIIAPE_Model20m145k2020_20220819, + "LogFavBasedUserInterestedLouvainMaxpoolingAddressBookFromIIAPE_Model20m145k2020_20221221" -> LogFavBasedUserInterestedLouvainMaxpoolingAddressBookFromIIAPE_Model20m145k2020_20221221, + "LogFavBasedUserInterestedLouvainMaxpoolingAddressBookFromIIAPE_Model20m145k2020_20221220" -> LogFavBasedUserInterestedLouvainMaxpoolingAddressBookFromIIAPE_Model20m145k2020_20221220, + "UserNextInterestedIn_Model20m145k2020_Default" -> UserNextInterestedIn_Model20m145k2020_Default, + "UserNextInterestedIn_Model20m145k2020_20220617_06" -> UserNextInterestedIn_Model20m145k2020_20220617_06, + "UserNextInterestedIn_Model20m145k2020_20220801" -> UserNextInterestedIn_Model20m145k2020_20220801, + "UserNextInterestedIn_Model20m145k2020_20220810" -> UserNextInterestedIn_Model20m145k2020_20220810, + "UserNextInterestedIn_Model20m145k2020_20220818" -> UserNextInterestedIn_Model20m145k2020_20220818, + "UserNextInterestedIn_Model20m145k2020_20220819" -> UserNextInterestedIn_Model20m145k2020_20220819, + "UserNextInterestedIn_Model20m145k2020_20221221" -> UserNextInterestedIn_Model20m145k2020_20221221, + "UserNextInterestedIn_Model20m145k2020_20221220" -> UserNextInterestedIn_Model20m145k2020_20221220, + ) + + def getConfig( + embeddingType: String, + modelVersion: String, + id: String + ): SimClustersANNConfig = { + val configName = embeddingType + "_" + modelVersion + "_" + id + DefaultConfigMappings.get(configName) match { + case Some(config) => config + case None => + throw InvalidSANNConfigException(s"Incorrect config id passed in for SANN $configName") + } + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/config/TimeoutConfig.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/config/TimeoutConfig.scala new file mode 100644 index 0000000000..46e32990b4 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/config/TimeoutConfig.scala @@ -0,0 +1,24 @@ +package com.twitter.cr_mixer.config + +import com.twitter.util.Duration + +case class TimeoutConfig( + /* Default timeouts for candidate generator */ + serviceTimeout: Duration, + signalFetchTimeout: Duration, + similarityEngineTimeout: Duration, + annServiceClientTimeout: Duration, + /* For Uteg Candidate Generator */ + utegSimilarityEngineTimeout: Duration, + /* For User State Store */ + userStateUnderlyingStoreTimeout: Duration, + userStateStoreTimeout: Duration, + /* For FRS based tweets */ + // Timeout passed to EarlyBird server + earlybirdServerTimeout: Duration, + // Timeout set on CrMixer side + earlybirdSimilarityEngineTimeout: Duration, + frsBasedTweetEndpointTimeout: Duration, + topicTweetEndpointTimeout: Duration, + // Timeout Settings for Navi gRPC Client + naviRequestTimeout: Duration) diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/controller/BUILD.bazel b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/controller/BUILD.bazel new file mode 100644 index 0000000000..b2f7d2f7d5 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/controller/BUILD.bazel @@ -0,0 +1,48 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + strict_deps = True, + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/com/twitter/storehaus:core", + "3rdparty/src/jvm/com/twitter/storehaus:core", + "content-recommender/thrift/src/main/thrift:content-recommender-common-scala", + "cr-mixer/server/src/main/scala/com/twitter/cr_mixer/candidate_generation", + "cr-mixer/server/src/main/scala/com/twitter/cr_mixer/debug", + "cr-mixer/server/src/main/scala/com/twitter/cr_mixer/featureswitch", + "cr-mixer/server/src/main/scala/com/twitter/cr_mixer/logging", + "cr-mixer/server/src/main/scala/com/twitter/cr_mixer/model", + "cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param", + "cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/decider", + "cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine", + "cr-mixer/server/src/main/scala/com/twitter/cr_mixer/source_signal", + "cr-mixer/server/src/main/scala/com/twitter/cr_mixer/util", + "cr-mixer/thrift/src/main/thrift:thrift-scala", + "finagle/finagle-base-http/src/main", + "finagle/finagle-core/src/main", + "finagle/finagle-http/src/main/scala", + "finatra/http-server/src/main/scala/com/twitter/finatra/http:controller", + "finatra/thrift/src/main/scala/com/twitter/finatra/thrift:controller", + "frigate/frigate-common/src/main/scala/com/twitter/frigate/common/base", + "hermit/hermit-core/src/main/scala/com/twitter/hermit/store/common", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/configapi", + "product-mixer/core/src/main/thrift/com/twitter/product_mixer/core:thrift-scala", + "simclusters-ann/thrift/src/main/thrift:thrift-scala", + "src/scala/com/twitter/simclusters_v2/common", + "src/thrift/com/twitter/ads/schema:common-scala", + "src/thrift/com/twitter/context:twitter-context-scala", + "src/thrift/com/twitter/core_workflows/user_model:user_model-scala", + "src/thrift/com/twitter/frigate/data_pipeline/scalding:blue_verified_annotations-scala", + "src/thrift/com/twitter/onboarding/relevance/coldstart_lookalike:coldstartlookalike-thrift-scala", + "src/thrift/com/twitter/recos:recos-common-scala", + "src/thrift/com/twitter/simclusters_v2:simclusters_v2-thrift-scala", + "src/thrift/com/twitter/timelines/render:thrift-scala", + "src/thrift/com/twitter/timelines/timeline_logging:thrift-scala", + "src/thrift/com/twitter/wtf/candidate:wtf-candidate-scala", + "stringcenter/client", + "timelines/src/main/scala/com/twitter/timelines/tracing/lensview", + "timelines/src/main/scala/com/twitter/timelines/tracing/lensview/funnelseries", + "twitter-context/src/main/scala", + "user-signal-service/thrift/src/main/thrift:thrift-scala", + ], +) diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/controller/CrMixerThriftController.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/controller/CrMixerThriftController.scala new file mode 100644 index 0000000000..c16d76de85 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/controller/CrMixerThriftController.scala @@ -0,0 +1,757 @@ +package com.twitter.cr_mixer.controller + +import com.twitter.core_workflows.user_model.thriftscala.UserState +import com.twitter.cr_mixer.candidate_generation.AdsCandidateGenerator +import com.twitter.cr_mixer.candidate_generation.CrCandidateGenerator +import com.twitter.cr_mixer.candidate_generation.FrsTweetCandidateGenerator +import com.twitter.cr_mixer.candidate_generation.RelatedTweetCandidateGenerator +import com.twitter.cr_mixer.candidate_generation.RelatedVideoTweetCandidateGenerator +import com.twitter.cr_mixer.candidate_generation.TopicTweetCandidateGenerator +import com.twitter.cr_mixer.candidate_generation.UtegTweetCandidateGenerator +import com.twitter.cr_mixer.featureswitch.ParamsBuilder +import com.twitter.cr_mixer.logging.CrMixerScribeLogger +import com.twitter.cr_mixer.logging.RelatedTweetScribeLogger +import com.twitter.cr_mixer.logging.AdsRecommendationsScribeLogger +import com.twitter.cr_mixer.logging.RelatedTweetScribeMetadata +import com.twitter.cr_mixer.logging.ScribeMetadata +import com.twitter.cr_mixer.logging.UtegTweetScribeLogger +import com.twitter.cr_mixer.model.AdsCandidateGeneratorQuery +import com.twitter.cr_mixer.model.CrCandidateGeneratorQuery +import com.twitter.cr_mixer.model.FrsTweetCandidateGeneratorQuery +import com.twitter.cr_mixer.model.InitialCandidate +import com.twitter.cr_mixer.model.RankedAdsCandidate +import com.twitter.cr_mixer.model.RankedCandidate +import com.twitter.cr_mixer.model.RelatedTweetCandidateGeneratorQuery +import com.twitter.cr_mixer.model.RelatedVideoTweetCandidateGeneratorQuery +import com.twitter.cr_mixer.model.TopicTweetCandidateGeneratorQuery +import com.twitter.cr_mixer.model.TweetWithScoreAndSocialProof +import com.twitter.cr_mixer.model.UtegTweetCandidateGeneratorQuery +import com.twitter.cr_mixer.param.AdsParams +import com.twitter.cr_mixer.param.FrsParams.FrsBasedCandidateGenerationMaxCandidatesNumParam +import com.twitter.cr_mixer.param.GlobalParams +import com.twitter.cr_mixer.param.RelatedTweetGlobalParams +import com.twitter.cr_mixer.param.RelatedVideoTweetGlobalParams +import com.twitter.cr_mixer.param.TopicTweetParams +import com.twitter.cr_mixer.param.decider.CrMixerDecider +import com.twitter.cr_mixer.param.decider.DeciderConstants +import com.twitter.cr_mixer.param.decider.EndpointLoadShedder +import com.twitter.cr_mixer.thriftscala.AdTweetRecommendation +import com.twitter.cr_mixer.thriftscala.AdsRequest +import com.twitter.cr_mixer.thriftscala.AdsResponse +import com.twitter.cr_mixer.thriftscala.CrMixerTweetRequest +import com.twitter.cr_mixer.thriftscala.CrMixerTweetResponse +import com.twitter.cr_mixer.thriftscala.FrsTweetRequest +import com.twitter.cr_mixer.thriftscala.FrsTweetResponse +import com.twitter.cr_mixer.thriftscala.RelatedTweet +import com.twitter.cr_mixer.thriftscala.RelatedTweetRequest +import com.twitter.cr_mixer.thriftscala.RelatedTweetResponse +import com.twitter.cr_mixer.thriftscala.RelatedVideoTweet +import com.twitter.cr_mixer.thriftscala.RelatedVideoTweetRequest +import com.twitter.cr_mixer.thriftscala.RelatedVideoTweetResponse +import com.twitter.cr_mixer.thriftscala.TopicTweet +import com.twitter.cr_mixer.thriftscala.TopicTweetRequest +import com.twitter.cr_mixer.thriftscala.TopicTweetResponse +import com.twitter.cr_mixer.thriftscala.TweetRecommendation +import com.twitter.cr_mixer.thriftscala.UtegTweet +import com.twitter.cr_mixer.thriftscala.UtegTweetRequest +import com.twitter.cr_mixer.thriftscala.UtegTweetResponse +import com.twitter.cr_mixer.util.MetricTagUtil +import com.twitter.cr_mixer.util.SignalTimestampStatsUtil +import com.twitter.cr_mixer.{thriftscala => t} +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.finatra.thrift.Controller +import com.twitter.hermit.store.common.ReadableWritableStore +import com.twitter.simclusters_v2.common.UserId +import com.twitter.simclusters_v2.thriftscala.TopicId +import com.twitter.storehaus.ReadableStore +import com.twitter.timelines.timeline_logging.{thriftscala => thriftlog} +import com.twitter.timelines.tracing.lensview.funnelseries.TweetScoreFunnelSeries +import com.twitter.util.Future +import com.twitter.util.Time +import java.util.UUID +import javax.inject.Inject +import org.apache.commons.lang.exception.ExceptionUtils + +class CrMixerThriftController @Inject() ( + crCandidateGenerator: CrCandidateGenerator, + relatedTweetCandidateGenerator: RelatedTweetCandidateGenerator, + relatedVideoTweetCandidateGenerator: RelatedVideoTweetCandidateGenerator, + utegTweetCandidateGenerator: UtegTweetCandidateGenerator, + frsTweetCandidateGenerator: FrsTweetCandidateGenerator, + topicTweetCandidateGenerator: TopicTweetCandidateGenerator, + crMixerScribeLogger: CrMixerScribeLogger, + relatedTweetScribeLogger: RelatedTweetScribeLogger, + utegTweetScribeLogger: UtegTweetScribeLogger, + adsRecommendationsScribeLogger: AdsRecommendationsScribeLogger, + adsCandidateGenerator: AdsCandidateGenerator, + decider: CrMixerDecider, + paramsBuilder: ParamsBuilder, + endpointLoadShedder: EndpointLoadShedder, + signalTimestampStatsUtil: SignalTimestampStatsUtil, + tweetRecommendationResultsStore: ReadableWritableStore[UserId, CrMixerTweetResponse], + userStateStore: ReadableStore[UserId, UserState], + statsReceiver: StatsReceiver) + extends Controller(t.CrMixer) { + + lazy private val tweetScoreFunnelSeries = new TweetScoreFunnelSeries(statsReceiver) + + private def logErrMessage(endpoint: String, e: Throwable): Unit = { + val msg = Seq( + s"Failed endpoint $endpoint: ${e.getLocalizedMessage}", + ExceptionUtils.getStackTrace(e) + ).mkString("\n") + + /** * + * We chose logger.info() here to print message instead of logger.error since that + * logger.error sometimes suppresses detailed stacktrace. + */ + logger.info(msg) + } + + private def generateRequestUUID(): Long = { + + /** * + * We generate unique UUID via bitwise operations. See the below link for more: + * https://stackoverflow.com/questions/15184820/how-to-generate-unique-positive-long-using-uuid + */ + UUID.randomUUID().getMostSignificantBits & Long.MaxValue + } + + handle(t.CrMixer.GetTweetRecommendations) { args: t.CrMixer.GetTweetRecommendations.Args => + val endpointName = "getTweetRecommendations" + + val requestUUID = generateRequestUUID() + val startTime = Time.now.inMilliseconds + val userId = args.request.clientContext.userId.getOrElse( + throw new IllegalArgumentException("userId must be present in the Thrift clientContext") + ) + val queryFut = buildCrCandidateGeneratorQuery(args.request, requestUUID, userId) + queryFut.flatMap { query => + val scribeMetadata = ScribeMetadata.from(query) + endpointLoadShedder(endpointName, query.product.originalName) { + + val response = crCandidateGenerator.get(query) + + val blueVerifiedScribedResponse = response.flatMap { rankedCandidates => + val hasBlueVerifiedCandidate = rankedCandidates.exists { tweet => + tweet.tweetInfo.hasBlueVerifiedAnnotation.contains(true) + } + + if (hasBlueVerifiedCandidate) { + crMixerScribeLogger.scribeGetTweetRecommendationsForBlueVerified( + scribeMetadata, + response) + } else { + response + } + } + + val thriftResponse = blueVerifiedScribedResponse.map { candidates => + if (query.product == t.Product.Home) { + scribeTweetScoreFunnelSeries(candidates) + } + buildThriftResponse(candidates) + } + + cacheTweetRecommendationResults(args.request, thriftResponse) + + crMixerScribeLogger.scribeGetTweetRecommendations( + args.request, + startTime, + scribeMetadata, + thriftResponse) + }.rescue { + case EndpointLoadShedder.LoadSheddingException => + Future(CrMixerTweetResponse(Seq.empty)) + case e => + logErrMessage(endpointName, e) + Future(CrMixerTweetResponse(Seq.empty)) + } + } + + } + + /** * + * GetRelatedTweetsForQueryTweet and GetRelatedTweetsForQueryAuthor are essentially + * doing very similar things, except that one passes in TweetId which calls TweetBased engine, + * and the other passes in AuthorId which calls ProducerBased engine. + */ + handle(t.CrMixer.GetRelatedTweetsForQueryTweet) { + args: t.CrMixer.GetRelatedTweetsForQueryTweet.Args => + val endpointName = "getRelatedTweetsForQueryTweet" + getRelatedTweets(endpointName, args.request) + } + + handle(t.CrMixer.GetRelatedVideoTweetsForQueryTweet) { + args: t.CrMixer.GetRelatedVideoTweetsForQueryTweet.Args => + val endpointName = "getRelatedVideoTweetsForQueryVideoTweet" + getRelatedVideoTweets(endpointName, args.request) + + } + + handle(t.CrMixer.GetRelatedTweetsForQueryAuthor) { + args: t.CrMixer.GetRelatedTweetsForQueryAuthor.Args => + val endpointName = "getRelatedTweetsForQueryAuthor" + getRelatedTweets(endpointName, args.request) + } + + private def getRelatedTweets( + endpointName: String, + request: RelatedTweetRequest + ): Future[RelatedTweetResponse] = { + val requestUUID = generateRequestUUID() + val startTime = Time.now.inMilliseconds + val queryFut = buildRelatedTweetQuery(request, requestUUID) + + queryFut.flatMap { query => + val relatedTweetScribeMetadata = RelatedTweetScribeMetadata.from(query) + endpointLoadShedder(endpointName, query.product.originalName) { + relatedTweetScribeLogger.scribeGetRelatedTweets( + request, + startTime, + relatedTweetScribeMetadata, + relatedTweetCandidateGenerator + .get(query) + .map(buildRelatedTweetResponse)) + }.rescue { + case EndpointLoadShedder.LoadSheddingException => + Future(RelatedTweetResponse(Seq.empty)) + case e => + logErrMessage(endpointName, e) + Future(RelatedTweetResponse(Seq.empty)) + } + } + + } + + private def getRelatedVideoTweets( + endpointName: String, + request: RelatedVideoTweetRequest + ): Future[RelatedVideoTweetResponse] = { + val requestUUID = generateRequestUUID() + val queryFut = buildRelatedVideoTweetQuery(request, requestUUID) + + queryFut.flatMap { query => + endpointLoadShedder(endpointName, query.product.originalName) { + relatedVideoTweetCandidateGenerator.get(query).map { initialCandidateSeq => + buildRelatedVideoTweetResponse(initialCandidateSeq) + } + }.rescue { + case EndpointLoadShedder.LoadSheddingException => + Future(RelatedVideoTweetResponse(Seq.empty)) + case e => + logErrMessage(endpointName, e) + Future(RelatedVideoTweetResponse(Seq.empty)) + } + } + } + + handle(t.CrMixer.GetFrsBasedTweetRecommendations) { + args: t.CrMixer.GetFrsBasedTweetRecommendations.Args => + val endpointName = "getFrsBasedTweetRecommendations" + + val requestUUID = generateRequestUUID() + val queryFut = buildFrsBasedTweetQuery(args.request, requestUUID) + queryFut.flatMap { query => + endpointLoadShedder(endpointName, query.product.originalName) { + frsTweetCandidateGenerator.get(query).map(FrsTweetResponse(_)) + }.rescue { + case e => + logErrMessage(endpointName, e) + Future(FrsTweetResponse(Seq.empty)) + } + } + } + + handle(t.CrMixer.GetTopicTweetRecommendations) { + args: t.CrMixer.GetTopicTweetRecommendations.Args => + val endpointName = "getTopicTweetRecommendations" + + val requestUUID = generateRequestUUID() + val query = buildTopicTweetQuery(args.request, requestUUID) + + endpointLoadShedder(endpointName, query.product.originalName) { + topicTweetCandidateGenerator.get(query).map(TopicTweetResponse(_)) + }.rescue { + case e => + logErrMessage(endpointName, e) + Future(TopicTweetResponse(Map.empty[Long, Seq[TopicTweet]])) + } + } + + handle(t.CrMixer.GetUtegTweetRecommendations) { + args: t.CrMixer.GetUtegTweetRecommendations.Args => + val endpointName = "getUtegTweetRecommendations" + + val requestUUID = generateRequestUUID() + val startTime = Time.now.inMilliseconds + val queryFut = buildUtegTweetQuery(args.request, requestUUID) + queryFut + .flatMap { query => + val scribeMetadata = ScribeMetadata.from(query) + endpointLoadShedder(endpointName, query.product.originalName) { + utegTweetScribeLogger.scribeGetUtegTweetRecommendations( + args.request, + startTime, + scribeMetadata, + utegTweetCandidateGenerator + .get(query) + .map(buildUtegTweetResponse) + ) + }.rescue { + case e => + logErrMessage(endpointName, e) + Future(UtegTweetResponse(Seq.empty)) + } + } + } + + handle(t.CrMixer.GetAdsRecommendations) { args: t.CrMixer.GetAdsRecommendations.Args => + val endpointName = "getAdsRecommendations" + val queryFut = buildAdsCandidateGeneratorQuery(args.request) + val startTime = Time.now.inMilliseconds + queryFut.flatMap { query => + { + val scribeMetadata = ScribeMetadata.from(query) + val response = adsCandidateGenerator + .get(query).map { candidates => + buildAdsResponse(candidates) + } + adsRecommendationsScribeLogger.scribeGetAdsRecommendations( + args.request, + startTime, + scribeMetadata, + response, + query.params(AdsParams.EnableScribe) + ) + }.rescue { + case e => + logErrMessage(endpointName, e) + Future(AdsResponse(Seq.empty)) + } + } + + } + + private def buildCrCandidateGeneratorQuery( + thriftRequest: CrMixerTweetRequest, + requestUUID: Long, + userId: Long + ): Future[CrCandidateGeneratorQuery] = { + + val product = thriftRequest.product + val productContext = thriftRequest.productContext + val scopedStats = statsReceiver + .scope(product.toString).scope("CrMixerTweetRequest") + + userStateStore + .get(userId).map { userStateOpt => + val userState = userStateOpt + .getOrElse(UserState.EnumUnknownUserState(100)) + scopedStats.scope("UserState").counter(userState.toString).incr() + + val params = + paramsBuilder.buildFromClientContext( + thriftRequest.clientContext, + thriftRequest.product, + userState + ) + + // Specify product-specific behavior mapping here + val maxNumResults = (product, productContext) match { + case (t.Product.Home, Some(t.ProductContext.HomeContext(homeContext))) => + homeContext.maxResults.getOrElse(9999) + case (t.Product.Notifications, Some(t.ProductContext.NotificationsContext(cxt))) => + params(GlobalParams.MaxCandidatesPerRequestParam) + case (t.Product.Email, None) => + params(GlobalParams.MaxCandidatesPerRequestParam) + case (t.Product.ImmersiveMediaViewer, None) => + params(GlobalParams.MaxCandidatesPerRequestParam) + case (t.Product.VideoCarousel, None) => + params(GlobalParams.MaxCandidatesPerRequestParam) + case _ => + throw new IllegalArgumentException( + s"Product ${product} and ProductContext ${productContext} are not allowed in CrMixer" + ) + } + + CrCandidateGeneratorQuery( + userId = userId, + product = product, + userState = userState, + maxNumResults = maxNumResults, + impressedTweetList = thriftRequest.excludedTweetIds.getOrElse(Nil).toSet, + params = params, + requestUUID = requestUUID, + languageCode = thriftRequest.clientContext.languageCode + ) + } + } + + private def buildRelatedTweetQuery( + thriftRequest: RelatedTweetRequest, + requestUUID: Long + ): Future[RelatedTweetCandidateGeneratorQuery] = { + + val product = thriftRequest.product + val scopedStats = statsReceiver + .scope(product.toString).scope("RelatedTweetRequest") + val userStateFut: Future[UserState] = (thriftRequest.clientContext.userId match { + case Some(userId) => userStateStore.get(userId) + case None => Future.value(Some(UserState.EnumUnknownUserState(100))) + }).map(_.getOrElse(UserState.EnumUnknownUserState(100))) + + userStateFut.map { userState => + scopedStats.scope("UserState").counter(userState.toString).incr() + val params = + paramsBuilder.buildFromClientContext( + thriftRequest.clientContext, + thriftRequest.product, + userState) + + // Specify product-specific behavior mapping here + // Currently, Home takes 10, and RUX takes 100 + val maxNumResults = params(RelatedTweetGlobalParams.MaxCandidatesPerRequestParam) + + RelatedTweetCandidateGeneratorQuery( + internalId = thriftRequest.internalId, + clientContext = thriftRequest.clientContext, + product = product, + maxNumResults = maxNumResults, + impressedTweetList = thriftRequest.excludedTweetIds.getOrElse(Nil).toSet, + params = params, + requestUUID = requestUUID + ) + } + } + + private def buildAdsCandidateGeneratorQuery( + thriftRequest: AdsRequest + ): Future[AdsCandidateGeneratorQuery] = { + val userId = thriftRequest.clientContext.userId.getOrElse( + throw new IllegalArgumentException("userId must be present in the Thrift clientContext") + ) + val product = thriftRequest.product + val requestUUID = generateRequestUUID() + userStateStore + .get(userId).map { userStateOpt => + val userState = userStateOpt + .getOrElse(UserState.EnumUnknownUserState(100)) + val params = + paramsBuilder.buildFromClientContext( + thriftRequest.clientContext, + thriftRequest.product, + userState) + val maxNumResults = params(AdsParams.AdsCandidateGenerationMaxCandidatesNumParam) + AdsCandidateGeneratorQuery( + userId = userId, + product = product, + userState = userState, + params = params, + maxNumResults = maxNumResults, + requestUUID = requestUUID + ) + } + } + + private def buildRelatedVideoTweetQuery( + thriftRequest: RelatedVideoTweetRequest, + requestUUID: Long + ): Future[RelatedVideoTweetCandidateGeneratorQuery] = { + + val product = thriftRequest.product + val scopedStats = statsReceiver + .scope(product.toString).scope("RelatedVideoTweetRequest") + val userStateFut: Future[UserState] = (thriftRequest.clientContext.userId match { + case Some(userId) => userStateStore.get(userId) + case None => Future.value(Some(UserState.EnumUnknownUserState(100))) + }).map(_.getOrElse(UserState.EnumUnknownUserState(100))) + + userStateFut.map { userState => + scopedStats.scope("UserState").counter(userState.toString).incr() + val params = + paramsBuilder.buildFromClientContext( + thriftRequest.clientContext, + thriftRequest.product, + userState) + + val maxNumResults = params(RelatedVideoTweetGlobalParams.MaxCandidatesPerRequestParam) + + RelatedVideoTweetCandidateGeneratorQuery( + internalId = thriftRequest.internalId, + clientContext = thriftRequest.clientContext, + product = product, + maxNumResults = maxNumResults, + impressedTweetList = thriftRequest.excludedTweetIds.getOrElse(Nil).toSet, + params = params, + requestUUID = requestUUID + ) + } + + } + + private def buildUtegTweetQuery( + thriftRequest: UtegTweetRequest, + requestUUID: Long + ): Future[UtegTweetCandidateGeneratorQuery] = { + + val userId = thriftRequest.clientContext.userId.getOrElse( + throw new IllegalArgumentException("userId must be present in the Thrift clientContext") + ) + val product = thriftRequest.product + val productContext = thriftRequest.productContext + val scopedStats = statsReceiver + .scope(product.toString).scope("UtegTweetRequest") + + userStateStore + .get(userId).map { userStateOpt => + val userState = userStateOpt + .getOrElse(UserState.EnumUnknownUserState(100)) + scopedStats.scope("UserState").counter(userState.toString).incr() + + val params = + paramsBuilder.buildFromClientContext( + thriftRequest.clientContext, + thriftRequest.product, + userState + ) + + // Specify product-specific behavior mapping here + val maxNumResults = (product, productContext) match { + case (t.Product.Home, Some(t.ProductContext.HomeContext(homeContext))) => + homeContext.maxResults.getOrElse(9999) + case _ => + throw new IllegalArgumentException( + s"Product ${product} and ProductContext ${productContext} are not allowed in CrMixer" + ) + } + + UtegTweetCandidateGeneratorQuery( + userId = userId, + product = product, + userState = userState, + maxNumResults = maxNumResults, + impressedTweetList = thriftRequest.excludedTweetIds.getOrElse(Nil).toSet, + params = params, + requestUUID = requestUUID + ) + } + + } + + private def buildTopicTweetQuery( + thriftRequest: TopicTweetRequest, + requestUUID: Long + ): TopicTweetCandidateGeneratorQuery = { + val userId = thriftRequest.clientContext.userId.getOrElse( + throw new IllegalArgumentException( + "userId must be present in the TopicTweetRequest clientContext") + ) + val product = thriftRequest.product + val productContext = thriftRequest.productContext + + // Specify product-specific behavior mapping here + val isVideoOnly = (product, productContext) match { + case (t.Product.ExploreTopics, Some(t.ProductContext.ExploreContext(context))) => + context.isVideoOnly + case (t.Product.TopicLandingPage, None) => + false + case (t.Product.HomeTopicsBackfill, None) => + false + case (t.Product.TopicTweetsStrato, None) => + false + case _ => + throw new IllegalArgumentException( + s"Product ${product} and ProductContext ${productContext} are not allowed in CrMixer" + ) + } + + statsReceiver.scope(product.toString).counter(TopicTweetRequest.toString).incr() + + val params = + paramsBuilder.buildFromClientContext( + thriftRequest.clientContext, + product, + UserState.EnumUnknownUserState(100) + ) + + val topicIds = thriftRequest.topicIds.map { topicId => + TopicId( + entityId = topicId, + language = thriftRequest.clientContext.languageCode, + country = None + ) + }.toSet + + TopicTweetCandidateGeneratorQuery( + userId = userId, + topicIds = topicIds, + product = product, + maxNumResults = params(TopicTweetParams.MaxTopicTweetCandidatesParam), + impressedTweetList = thriftRequest.excludedTweetIds.getOrElse(Nil).toSet, + params = params, + requestUUID = requestUUID, + isVideoOnly = isVideoOnly + ) + } + + private def buildFrsBasedTweetQuery( + thriftRequest: FrsTweetRequest, + requestUUID: Long + ): Future[FrsTweetCandidateGeneratorQuery] = { + val userId = thriftRequest.clientContext.userId.getOrElse( + throw new IllegalArgumentException( + "userId must be present in the FrsTweetRequest clientContext") + ) + val product = thriftRequest.product + val productContext = thriftRequest.productContext + + val scopedStats = statsReceiver + .scope(product.toString).scope("FrsTweetRequest") + + userStateStore + .get(userId).map { userStateOpt => + val userState = userStateOpt + .getOrElse(UserState.EnumUnknownUserState(100)) + scopedStats.scope("UserState").counter(userState.toString).incr() + + val params = + paramsBuilder.buildFromClientContext( + thriftRequest.clientContext, + thriftRequest.product, + userState + ) + val maxNumResults = (product, productContext) match { + case (t.Product.Home, Some(t.ProductContext.HomeContext(homeContext))) => + homeContext.maxResults.getOrElse( + params(FrsBasedCandidateGenerationMaxCandidatesNumParam)) + case _ => + params(FrsBasedCandidateGenerationMaxCandidatesNumParam) + } + + FrsTweetCandidateGeneratorQuery( + userId = userId, + product = product, + maxNumResults = maxNumResults, + impressedTweetList = thriftRequest.excludedTweetIds.getOrElse(Nil).toSet, + impressedUserList = thriftRequest.excludedUserIds.getOrElse(Nil).toSet, + params = params, + languageCodeOpt = thriftRequest.clientContext.languageCode, + countryCodeOpt = thriftRequest.clientContext.countryCode, + requestUUID = requestUUID + ) + } + } + + private def buildThriftResponse( + candidates: Seq[RankedCandidate] + ): CrMixerTweetResponse = { + + val tweets = candidates.map { candidate => + TweetRecommendation( + tweetId = candidate.tweetId, + score = candidate.predictionScore, + metricTags = Some(MetricTagUtil.buildMetricTags(candidate)), + latestSourceSignalTimestampInMillis = + SignalTimestampStatsUtil.buildLatestSourceSignalTimestamp(candidate) + ) + } + signalTimestampStatsUtil.statsSignalTimestamp(tweets) + CrMixerTweetResponse(tweets) + } + + private def scribeTweetScoreFunnelSeries( + candidates: Seq[RankedCandidate] + ): Seq[RankedCandidate] = { + // 202210210901 is a random number for code search of Lensview + tweetScoreFunnelSeries.startNewSpan( + name = "GetTweetRecommendationsTopLevelTweetSimilarityEngineType", + codePtr = 202210210901L) { + ( + candidates, + candidates.map { candidate => + thriftlog.TweetDimensionMeasure( + dimension = Some( + thriftlog + .RequestTweetDimension( + candidate.tweetId, + candidate.reasonChosen.similarityEngineInfo.similarityEngineType.value)), + measure = Some(thriftlog.RequestTweetMeasure(candidate.predictionScore)) + ) + } + ) + } + } + + private def buildRelatedTweetResponse(candidates: Seq[InitialCandidate]): RelatedTweetResponse = { + val tweets = candidates.map { candidate => + RelatedTweet( + tweetId = candidate.tweetId, + score = Some(candidate.getSimilarityScore), + authorId = Some(candidate.tweetInfo.authorId) + ) + } + RelatedTweetResponse(tweets) + } + + private def buildRelatedVideoTweetResponse( + candidates: Seq[InitialCandidate] + ): RelatedVideoTweetResponse = { + val tweets = candidates.map { candidate => + RelatedVideoTweet( + tweetId = candidate.tweetId, + score = Some(candidate.getSimilarityScore) + ) + } + RelatedVideoTweetResponse(tweets) + } + + private def buildUtegTweetResponse( + candidates: Seq[TweetWithScoreAndSocialProof] + ): UtegTweetResponse = { + val tweets = candidates.map { candidate => + UtegTweet( + tweetId = candidate.tweetId, + score = candidate.score, + socialProofByType = candidate.socialProofByType + ) + } + UtegTweetResponse(tweets) + } + + private def buildAdsResponse( + candidates: Seq[RankedAdsCandidate] + ): AdsResponse = { + AdsResponse(ads = candidates.map { candidate => + AdTweetRecommendation( + tweetId = candidate.tweetId, + score = candidate.predictionScore, + lineItems = Some(candidate.lineItemInfo)) + }) + } + + private def cacheTweetRecommendationResults( + request: CrMixerTweetRequest, + response: Future[CrMixerTweetResponse] + ): Unit = { + + val userId = request.clientContext.userId.getOrElse( + throw new IllegalArgumentException( + "userId must be present in getTweetRecommendations() Thrift clientContext")) + + if (decider.isAvailableForId(userId, DeciderConstants.getTweetRecommendationsCacheRate)) { + response.map { crMixerTweetResponse => + { + ( + request.product, + request.clientContext.userId, + crMixerTweetResponse.tweets.nonEmpty) match { + case (t.Product.Home, Some(userId), true) => + tweetRecommendationResultsStore.put((userId, crMixerTweetResponse)) + case _ => Future.value(Unit) + } + } + } + } + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/exception/BUILD b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/exception/BUILD new file mode 100644 index 0000000000..60521ad51a --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/exception/BUILD @@ -0,0 +1,7 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + strict_deps = True, + tags = ["bazel-compatible"], + dependencies = [], +) diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/exception/InvalidSANNConfigException.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/exception/InvalidSANNConfigException.scala new file mode 100644 index 0000000000..a8ada7abf0 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/exception/InvalidSANNConfigException.scala @@ -0,0 +1,4 @@ +package com.twitter.cr_mixer +package exception + +case class InvalidSANNConfigException(msg: String) extends Exception(msg) diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/featureswitch/BUILD b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/featureswitch/BUILD new file mode 100644 index 0000000000..d728980f64 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/featureswitch/BUILD @@ -0,0 +1,35 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + strict_deps = True, + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/javax/inject:javax.inject", + "abdecider/src/main/scala", + "configapi/configapi-abdecider", + "configapi/configapi-core", + "configapi/configapi-featureswitches:v2", + "cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param", + "cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/decider", + "cr-mixer/thrift/src/main/thrift:thrift-scala", + "decider/src/main/scala", + "discovery-common/src/main/scala/com/twitter/discovery/common/configapi", + "featureswitches/featureswitches-core", + "featureswitches/featureswitches-core/src/main/scala/com/twitter/featureswitches/v2/builder", + "finagle-internal/mtls/src/main/scala/com/twitter/finagle/mtls/authentication", + "frigate/frigate-common:util", + "frigate/frigate-common/src/main/scala/com/twitter/frigate/common/base", + "frigate/frigate-common/src/main/scala/com/twitter/frigate/common/candidate", + "frigate/frigate-common/src/main/scala/com/twitter/frigate/common/store", + "frigate/frigate-common/src/main/scala/com/twitter/frigate/common/store/health", + "frigate/frigate-common/src/main/scala/com/twitter/frigate/common/store/interests", + "frigate/frigate-common/src/main/scala/com/twitter/frigate/common/store/strato", + "frigate/frigate-common/src/main/scala/com/twitter/frigate/common/util", + "product-mixer/core/src/main/thrift/com/twitter/product_mixer/core:thrift-scala", + "scribelib/marshallers/src/main/scala/com/twitter/scribelib/marshallers", + "src/scala/com/twitter/simclusters_v2/common", + "src/thrift/com/twitter/core_workflows/user_model:user_model-scala", + "src/thrift/com/twitter/frigate:frigate-common-thrift-scala", + "src/thrift/com/twitter/simclusters_v2:simclusters_v2-thrift-scala", + ], +) diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/featureswitch/CrMixerLoggingABDecider.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/featureswitch/CrMixerLoggingABDecider.scala new file mode 100644 index 0000000000..20195921ef --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/featureswitch/CrMixerLoggingABDecider.scala @@ -0,0 +1,79 @@ +package com.twitter.cr_mixer +package featureswitch + +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.abdecider.LoggingABDecider +import com.twitter.abdecider.Recipient +import com.twitter.abdecider.Bucket +import com.twitter.frigate.common.util.StatsUtil +import com.twitter.util.Local +import scala.collection.concurrent.{Map => ConcurrentMap} + +/** + * Wraps a LoggingABDecider, so all impressed buckets are recorded to a 'LocalContext' on a given request. + * + * Contexts (https://twitter.github.io/finagle/guide/Contexts.html) are Finagle's mechanism for + * storing state/variables without having to pass these variables all around the request. + * + * In order for this class to be used the [[SetImpressedBucketsLocalContextFilter]] must be applied + * at the beginning of the request, to initialize a concurrent map used to store impressed buckets. + * + * Whenever we get an a/b impression, the bucket information is logged to the concurrent hashmap. + */ +case class CrMixerLoggingABDecider( + loggingAbDecider: LoggingABDecider, + statsReceiver: StatsReceiver) + extends LoggingABDecider { + + private val scopedStatsReceiver = statsReceiver.scope("cr_logging_ab_decider") + + override def impression( + experimentName: String, + recipient: Recipient + ): Option[Bucket] = { + + StatsUtil.trackNonFutureBlockStats(scopedStatsReceiver.scope("log_impression")) { + val maybeBuckets = loggingAbDecider.impression(experimentName, recipient) + maybeBuckets.foreach { b => + scopedStatsReceiver.counter("impressions").incr() + CrMixerImpressedBuckets.recordImpressedBucket(b) + } + maybeBuckets + } + } + + override def track( + experimentName: String, + eventName: String, + recipient: Recipient + ): Unit = { + loggingAbDecider.track(experimentName, eventName, recipient) + } + + override def bucket( + experimentName: String, + recipient: Recipient + ): Option[Bucket] = { + loggingAbDecider.bucket(experimentName, recipient) + } + + override def experiments: Seq[String] = loggingAbDecider.experiments + + override def experiment(experimentName: String) = + loggingAbDecider.experiment(experimentName) +} + +object CrMixerImpressedBuckets { + private[featureswitch] val localImpressedBucketsMap = new Local[ConcurrentMap[Bucket, Boolean]] + + /** + * Gets all impressed buckets for this request. + **/ + def getAllImpressedBuckets: Option[List[Bucket]] = { + localImpressedBucketsMap.apply().map(_.map { case (k, _) => k }.toList) + } + + private[featureswitch] def recordImpressedBucket(bucket: Bucket) = { + localImpressedBucketsMap().foreach { m => m += bucket -> true } + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/featureswitch/ParamsBuilder.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/featureswitch/ParamsBuilder.scala new file mode 100644 index 0000000000..c322c456e6 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/featureswitch/ParamsBuilder.scala @@ -0,0 +1,151 @@ +package com.twitter.cr_mixer.featureswitch + +import com.twitter.abdecider.LoggingABDecider +import com.twitter.abdecider.UserRecipient +import com.twitter.cr_mixer.{thriftscala => t} +import com.twitter.core_workflows.user_model.thriftscala.UserState +import com.twitter.discovery.common.configapi.FeatureContextBuilder +import com.twitter.featureswitches.FSRecipient +import com.twitter.featureswitches.UserAgent +import com.twitter.featureswitches.{Recipient => FeatureSwitchRecipient} +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.product_mixer.core.thriftscala.ClientContext +import com.twitter.timelines.configapi.Config +import com.twitter.timelines.configapi.FeatureValue +import com.twitter.timelines.configapi.ForcedFeatureContext +import com.twitter.timelines.configapi.OrElseFeatureContext +import com.twitter.timelines.configapi.Params +import com.twitter.timelines.configapi.RequestContext +import com.twitter.timelines.configapi.abdecider.LoggingABDeciderExperimentContext +import javax.inject.Inject +import javax.inject.Singleton + +/** Singleton object for building [[Params]] to override */ +@Singleton +class ParamsBuilder @Inject() ( + globalStats: StatsReceiver, + abDecider: LoggingABDecider, + featureContextBuilder: FeatureContextBuilder, + config: Config) { + + private val stats = globalStats.scope("params") + + def buildFromClientContext( + clientContext: ClientContext, + product: t.Product, + userState: UserState, + userRoleOverride: Option[Set[String]] = None, + featureOverrides: Map[String, FeatureValue] = Map.empty, + ): Params = { + clientContext.userId match { + case Some(userId) => + val userRecipient = buildFeatureSwitchRecipient( + userId, + userRoleOverride, + clientContext, + product, + userState + ) + + val featureContext = OrElseFeatureContext( + ForcedFeatureContext(featureOverrides), + featureContextBuilder( + Some(userId), + Some(userRecipient) + )) + + config( + requestContext = RequestContext( + userId = Some(userId), + experimentContext = LoggingABDeciderExperimentContext( + abDecider, + Some(UserRecipient(userId, Some(userId)))), + featureContext = featureContext + ), + stats + ) + case None => + val guestRecipient = + buildFeatureSwitchRecipientWithGuestId(clientContext: ClientContext, product, userState) + + val featureContext = OrElseFeatureContext( + ForcedFeatureContext(featureOverrides), + featureContextBuilder( + clientContext.userId, + Some(guestRecipient) + ) + ) //ExperimentContext with GuestRecipient is not supported as there is no active use-cases yet in CrMixer + + config( + requestContext = RequestContext( + userId = clientContext.userId, + featureContext = featureContext + ), + stats + ) + } + } + + private def buildFeatureSwitchRecipientWithGuestId( + clientContext: ClientContext, + product: t.Product, + userState: UserState + ): FeatureSwitchRecipient = { + + val recipient = FSRecipient( + userId = None, + userRoles = None, + deviceId = clientContext.deviceId, + guestId = clientContext.guestId, + languageCode = clientContext.languageCode, + countryCode = clientContext.countryCode, + userAgent = clientContext.userAgent.flatMap(UserAgent(_)), + isVerified = None, + isTwoffice = None, + tooClient = None, + highWaterMark = None + ) + + recipient.withCustomFields( + (ParamsBuilder.ProductCustomField, product.toString), + (ParamsBuilder.UserStateCustomField, userState.toString) + ) + } + + private def buildFeatureSwitchRecipient( + userId: Long, + userRolesOverride: Option[Set[String]], + clientContext: ClientContext, + product: t.Product, + userState: UserState + ): FeatureSwitchRecipient = { + val userRoles = userRolesOverride match { + case Some(overrides) => Some(overrides) + case _ => clientContext.userRoles.map(_.toSet) + } + + val recipient = FSRecipient( + userId = Some(userId), + userRoles = userRoles, + deviceId = clientContext.deviceId, + guestId = clientContext.guestId, + languageCode = clientContext.languageCode, + countryCode = clientContext.countryCode, + userAgent = clientContext.userAgent.flatMap(UserAgent(_)), + isVerified = None, + isTwoffice = None, + tooClient = None, + highWaterMark = None + ) + + recipient.withCustomFields( + (ParamsBuilder.ProductCustomField, product.toString), + (ParamsBuilder.UserStateCustomField, userState.toString) + ) + } +} + +object ParamsBuilder { + private val ProductCustomField = "product_id" + private val UserStateCustomField = "user_state" +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/featureswitch/SetImpressedBucketsLocalContextFilter.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/featureswitch/SetImpressedBucketsLocalContextFilter.scala new file mode 100644 index 0000000000..905c99bea5 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/featureswitch/SetImpressedBucketsLocalContextFilter.scala @@ -0,0 +1,22 @@ +package com.twitter.cr_mixer.featureswitch + +import com.twitter.finagle.Filter +import javax.inject.Inject +import javax.inject.Singleton +import scala.collection.concurrent.TrieMap +import com.twitter.abdecider.Bucket +import com.twitter.finagle.Service + +@Singleton +class SetImpressedBucketsLocalContextFilter @Inject() () extends Filter.TypeAgnostic { + override def toFilter[Req, Rep]: Filter[Req, Rep, Req, Rep] = + (request: Req, service: Service[Req, Rep]) => { + + val concurrentTrieMap = TrieMap + .empty[Bucket, Boolean] // Trie map has no locks and O(1) inserts + CrMixerImpressedBuckets.localImpressedBucketsMap.let(concurrentTrieMap) { + service(request) + } + } + +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/filter/BUILD b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/filter/BUILD new file mode 100644 index 0000000000..e9db597989 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/filter/BUILD @@ -0,0 +1,22 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + strict_deps = True, + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/com/twitter/storehaus:core", + "3rdparty/jvm/javax/inject:javax.inject", + "configapi/configapi-core", + "content-recommender/thrift/src/main/thrift:thrift-scala", + "cr-mixer/server/src/main/scala/com/twitter/cr_mixer/model", + "cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param", + "cr-mixer/thrift/src/main/thrift:thrift-scala", + "finagle/finagle-core/src/main", + "frigate/frigate-common:util", + "snowflake/src/main/scala/com/twitter/snowflake/id", + "src/scala/com/twitter/simclusters_v2/common", + "src/thrift/com/twitter/core_workflows/user_model:user_model-scala", + "src/thrift/com/twitter/simclusters_v2:simclusters_v2-thrift-scala", + "src/thrift/com/twitter/wtf/candidate:wtf-candidate-scala", + ], +) diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/filter/FilterBase.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/filter/FilterBase.scala new file mode 100644 index 0000000000..1be4ebbaa1 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/filter/FilterBase.scala @@ -0,0 +1,22 @@ +package com.twitter.cr_mixer.filter + +import com.twitter.cr_mixer.model.CandidateGeneratorQuery +import com.twitter.cr_mixer.model.InitialCandidate +import com.twitter.util.Future + +trait FilterBase { + def name: String + + type ConfigType + + def filter( + candidates: Seq[Seq[InitialCandidate]], + config: ConfigType + ): Future[Seq[Seq[InitialCandidate]]] + + /** + * Build the config params here. passing in param() into the filter is strongly discouraged + * because param() can be slow when called many times + */ + def requestToConfig[CGQueryType <: CandidateGeneratorQuery](request: CGQueryType): ConfigType +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/filter/ImpressedTweetlistFilter.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/filter/ImpressedTweetlistFilter.scala new file mode 100644 index 0000000000..41c9b77427 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/filter/ImpressedTweetlistFilter.scala @@ -0,0 +1,63 @@ +package com.twitter.cr_mixer.filter + +import com.twitter.cr_mixer.model.CandidateGeneratorQuery +import com.twitter.cr_mixer.model.InitialCandidate +import com.twitter.simclusters_v2.common.TweetId +import com.twitter.simclusters_v2.thriftscala.InternalId +import com.twitter.util.Future +import javax.inject.Singleton + +@Singleton +case class ImpressedTweetlistFilter() extends FilterBase { + import ImpressedTweetlistFilter._ + + override val name: String = this.getClass.getCanonicalName + + override type ConfigType = FilterConfig + + /* + Filtering removes some candidates based on configurable criteria. + */ + override def filter( + candidates: Seq[Seq[InitialCandidate]], + config: FilterConfig + ): Future[Seq[Seq[InitialCandidate]]] = { + // Remove candidates which match a source tweet, or which are passed in impressedTweetList + val sourceTweetsMatch = candidates + .flatMap { + + /*** + * Within a Seq[Seq[InitialCandidate]], all candidates within a inner Seq + * are guaranteed to have the same sourceInfo. Hence, we can pick .headOption + * to represent the whole list when filtering by the internalId of the sourceInfoOpt. + * But of course the similarityEngineInfo could be different. + */ + _.headOption.flatMap { candidate => + candidate.candidateGenerationInfo.sourceInfoOpt.map(_.internalId) + } + }.collect { + case InternalId.TweetId(id) => id + } + + val impressedTweetList: Set[TweetId] = + config.impressedTweetList ++ sourceTweetsMatch + + val filteredCandidateMap: Seq[Seq[InitialCandidate]] = + candidates.map { + _.filterNot { candidate => + impressedTweetList.contains(candidate.tweetId) + } + } + Future.value(filteredCandidateMap) + } + + override def requestToConfig[CGQueryType <: CandidateGeneratorQuery]( + request: CGQueryType + ): FilterConfig = { + FilterConfig(request.impressedTweetList) + } +} + +object ImpressedTweetlistFilter { + case class FilterConfig(impressedTweetList: Set[TweetId]) +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/filter/InNetworkFilter.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/filter/InNetworkFilter.scala new file mode 100644 index 0000000000..62f4ddba5c --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/filter/InNetworkFilter.scala @@ -0,0 +1,80 @@ +package com.twitter.cr_mixer.filter + +import com.twitter.cr_mixer.model.CandidateGeneratorQuery +import com.twitter.cr_mixer.model.InitialCandidate +import com.twitter.cr_mixer.model.ModuleNames +import com.twitter.cr_mixer.model.UtegTweetCandidateGeneratorQuery +import com.twitter.cr_mixer.param.UtegTweetGlobalParams +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.frigate.common.util.StatsUtil +import com.twitter.simclusters_v2.common.UserId +import com.twitter.storehaus.ReadableStore +import com.twitter.util.Future +import com.twitter.wtf.candidate.thriftscala.CandidateSeq + +import javax.inject.Inject +import javax.inject.Named +import javax.inject.Singleton + +/*** + * Filters in-network tweets + */ +@Singleton +case class InNetworkFilter @Inject() ( + @Named(ModuleNames.RealGraphInStore) realGraphStoreMh: ReadableStore[UserId, CandidateSeq], + globalStats: StatsReceiver) + extends FilterBase { + override val name: String = this.getClass.getCanonicalName + import InNetworkFilter._ + + override type ConfigType = FilterConfig + private val stats: StatsReceiver = globalStats.scope(this.getClass.getCanonicalName) + private val filterCandidatesStats = stats.scope("filter_candidates") + + override def filter( + candidates: Seq[Seq[InitialCandidate]], + filterConfig: FilterConfig, + ): Future[Seq[Seq[InitialCandidate]]] = { + StatsUtil.trackItemsStats(filterCandidatesStats) { + filterCandidates(candidates, filterConfig) + } + } + + private def filterCandidates( + candidates: Seq[Seq[InitialCandidate]], + filterConfig: FilterConfig, + ): Future[Seq[Seq[InitialCandidate]]] = { + + if (!filterConfig.enableInNetworkFilter) { + Future.value(candidates) + } else { + filterConfig.userIdOpt match { + case Some(userId) => + realGraphStoreMh + .get(userId).map(_.map(_.candidates.map(_.userId)).getOrElse(Seq.empty).toSet).map { + realGraphInNetworkAuthorsSet => + candidates.map(_.filterNot { candidate => + realGraphInNetworkAuthorsSet.contains(candidate.tweetInfo.authorId) + }) + } + case None => Future.value(candidates) + } + } + } + + override def requestToConfig[CGQueryType <: CandidateGeneratorQuery]( + request: CGQueryType + ): FilterConfig = { + request match { + case UtegTweetCandidateGeneratorQuery(userId, _, _, _, _, params, _) => + FilterConfig(Some(userId), params(UtegTweetGlobalParams.EnableInNetworkFilterParam)) + case _ => FilterConfig(None, false) + } + } +} + +object InNetworkFilter { + case class FilterConfig( + userIdOpt: Option[UserId], + enableInNetworkFilter: Boolean) +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/filter/PostRankFilterRunner.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/filter/PostRankFilterRunner.scala new file mode 100644 index 0000000000..483f3d9564 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/filter/PostRankFilterRunner.scala @@ -0,0 +1,58 @@ +package com.twitter.cr_mixer.filter +import com.twitter.cr_mixer.model.CrCandidateGeneratorQuery +import com.twitter.cr_mixer.model.RankedCandidate +import com.twitter.cr_mixer.thriftscala.SourceType +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.util.Future +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +case class PostRankFilterRunner @Inject() ( + globalStats: StatsReceiver) { + + private val scopedStats = globalStats.scope(this.getClass.getCanonicalName) + + private val beforeCount = scopedStats.stat("candidate_count", "before") + private val afterCount = scopedStats.stat("candidate_count", "after") + + def run( + query: CrCandidateGeneratorQuery, + candidates: Seq[RankedCandidate] + ): Future[Seq[RankedCandidate]] = { + + beforeCount.add(candidates.size) + + Future( + removeBadRecentNotificationCandidates(candidates) + ).map { results => + afterCount.add(results.size) + results + } + } + + /** + * Remove "bad" quality candidates generated by recent notifications + * A candidate is bad when it is generated by a single RecentNotification + * SourceKey. + * e.x: + * tweetA {recent notification1} -> bad + * tweetB {recent notification1 recent notification2} -> good + *tweetC {recent notification1 recent follow1} -> bad + * SD-19397 + */ + private[filter] def removeBadRecentNotificationCandidates( + candidates: Seq[RankedCandidate] + ): Seq[RankedCandidate] = { + candidates.filterNot { + isBadQualityRecentNotificationCandidate + } + } + + private def isBadQualityRecentNotificationCandidate(candidate: RankedCandidate): Boolean = { + candidate.potentialReasons.size == 1 && + candidate.potentialReasons.head.sourceInfoOpt.nonEmpty && + candidate.potentialReasons.head.sourceInfoOpt.get.sourceType == SourceType.NotificationClick + } + +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/filter/PreRankFilterRunner.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/filter/PreRankFilterRunner.scala new file mode 100644 index 0000000000..7626acc7cc --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/filter/PreRankFilterRunner.scala @@ -0,0 +1,99 @@ +package com.twitter.cr_mixer.filter + +import com.twitter.cr_mixer.model.CandidateGeneratorQuery +import com.twitter.cr_mixer.model.InitialCandidate +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.util.Future +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class PreRankFilterRunner @Inject() ( + impressedTweetListFilter: ImpressedTweetlistFilter, + tweetAgeFilter: TweetAgeFilter, + videoTweetFilter: VideoTweetFilter, + tweetReplyFilter: ReplyFilter, + globalStats: StatsReceiver) { + + private val scopedStats = globalStats.scope(this.getClass.getCanonicalName) + + /*** + * The order of the filters does not matter as long as we do not apply .take(N) truncation + * across all filters. In other words, it is fine that we first do tweetAgeFilter, and then + * we do impressedTweetListFilter, or the other way around. + * Same idea applies to the signal based filter - it is ok that we apply signal based filters + * before impressedTweetListFilter. + * + * We move all signal based filters before tweetAgeFilter and impressedTweetListFilter + * as a set of early filters. + */ + val orderedFilters = Seq( + tweetAgeFilter, + impressedTweetListFilter, + videoTweetFilter, + tweetReplyFilter + ) + + def runSequentialFilters[CGQueryType <: CandidateGeneratorQuery]( + request: CGQueryType, + candidates: Seq[Seq[InitialCandidate]], + ): Future[Seq[Seq[InitialCandidate]]] = { + PreRankFilterRunner.runSequentialFilters( + request, + candidates, + orderedFilters, + scopedStats + ) + } + +} + +object PreRankFilterRunner { + private def recordCandidateStatsBeforeFilter( + candidates: Seq[Seq[InitialCandidate]], + statsReceiver: StatsReceiver + ): Unit = { + statsReceiver + .counter("empty_sources", "before").incr( + candidates.count { _.isEmpty } + ) + candidates.foreach { candidate => + statsReceiver.counter("candidates", "before").incr(candidate.size) + } + } + + private def recordCandidateStatsAfterFilter( + candidates: Seq[Seq[InitialCandidate]], + statsReceiver: StatsReceiver + ): Unit = { + statsReceiver + .counter("empty_sources", "after").incr( + candidates.count { _.isEmpty } + ) + candidates.foreach { candidate => + statsReceiver.counter("candidates", "after").incr(candidate.size) + } + } + + /* + Helper function for running some candidates through a sequence of filters + */ + private[filter] def runSequentialFilters[CGQueryType <: CandidateGeneratorQuery]( + request: CGQueryType, + candidates: Seq[Seq[InitialCandidate]], + filters: Seq[FilterBase], + statsReceiver: StatsReceiver + ): Future[Seq[Seq[InitialCandidate]]] = + filters.foldLeft(Future.value(candidates)) { + case (candsFut, filter) => + candsFut.flatMap { cands => + recordCandidateStatsBeforeFilter(cands, statsReceiver.scope(filter.name)) + filter + .filter(cands, filter.requestToConfig(request)) + .map { filteredCands => + recordCandidateStatsAfterFilter(filteredCands, statsReceiver.scope(filter.name)) + filteredCands + } + } + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/filter/ReplyFilter.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/filter/ReplyFilter.scala new file mode 100644 index 0000000000..d4d37a7da3 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/filter/ReplyFilter.scala @@ -0,0 +1,40 @@ +package com.twitter.cr_mixer.filter + +import com.twitter.cr_mixer.model.CandidateGeneratorQuery +import com.twitter.cr_mixer.model.InitialCandidate +import com.twitter.util.Future + +import javax.inject.Inject +import javax.inject.Singleton + +/*** + * Filters candidates that are replies + */ +@Singleton +case class ReplyFilter @Inject() () extends FilterBase { + override def name: String = this.getClass.getCanonicalName + override type ConfigType = Boolean + + override def filter( + candidates: Seq[Seq[InitialCandidate]], + config: ConfigType + ): Future[Seq[Seq[InitialCandidate]]] = { + if (config) { + Future.value( + candidates.map { candidateSeq => + candidateSeq.filterNot { candidate => + candidate.tweetInfo.isReply.getOrElse(false) + } + } + ) + } else { + Future.value(candidates) + } + } + + override def requestToConfig[CGQueryType <: CandidateGeneratorQuery]( + query: CGQueryType + ): ConfigType = { + true + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/filter/RetweetFilter.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/filter/RetweetFilter.scala new file mode 100644 index 0000000000..38eefadd9f --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/filter/RetweetFilter.scala @@ -0,0 +1,41 @@ +package com.twitter.cr_mixer.filter + +import com.twitter.cr_mixer.model.CandidateGeneratorQuery +import com.twitter.cr_mixer.model.InitialCandidate +import com.twitter.cr_mixer.param.UtegTweetGlobalParams +import com.twitter.util.Future + +import javax.inject.Inject +import javax.inject.Singleton + +/*** + * Filters candidates that are retweets + */ +@Singleton +case class RetweetFilter @Inject() () extends FilterBase { + override def name: String = this.getClass.getCanonicalName + override type ConfigType = Boolean + + override def filter( + candidates: Seq[Seq[InitialCandidate]], + config: ConfigType + ): Future[Seq[Seq[InitialCandidate]]] = { + if (config) { + Future.value( + candidates.map { candidateSeq => + candidateSeq.filterNot { candidate => + candidate.tweetInfo.isRetweet.getOrElse(false) + } + } + ) + } else { + Future.value(candidates) + } + } + + override def requestToConfig[CGQueryType <: CandidateGeneratorQuery]( + query: CGQueryType + ): ConfigType = { + query.params(UtegTweetGlobalParams.EnableRetweetFilterParam) + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/filter/TweetAgeFilter.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/filter/TweetAgeFilter.scala new file mode 100644 index 0000000000..d7c8889e15 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/filter/TweetAgeFilter.scala @@ -0,0 +1,39 @@ +package com.twitter.cr_mixer.filter + +import com.twitter.cr_mixer.model.CandidateGeneratorQuery +import com.twitter.cr_mixer.model.InitialCandidate +import com.twitter.cr_mixer.param.GlobalParams +import com.twitter.snowflake.id.SnowflakeId +import com.twitter.util.Duration +import com.twitter.util.Future +import com.twitter.util.Time +import javax.inject.Singleton +import com.twitter.conversions.DurationOps._ + +@Singleton +case class TweetAgeFilter() extends FilterBase { + override val name: String = this.getClass.getCanonicalName + + override type ConfigType = Duration + + override def filter( + candidates: Seq[Seq[InitialCandidate]], + maxTweetAge: Duration + ): Future[Seq[Seq[InitialCandidate]]] = { + if (maxTweetAge >= 720.hours) { + Future.value(candidates) + } else { + // Tweet IDs are approximately chronological (see http://go/snowflake), + // so we are building the earliest tweet id once, + // and pass that as the value to filter candidates for each CandidateGenerationModel. + val earliestTweetId = SnowflakeId.firstIdFor(Time.now - maxTweetAge) + Future.value(candidates.map(_.filter(_.tweetId >= earliestTweetId))) + } + } + + override def requestToConfig[CGQueryType <: CandidateGeneratorQuery]( + query: CGQueryType + ): Duration = { + query.params(GlobalParams.MaxTweetAgeHoursParam) + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/filter/TweetInfoHealthFilterBase.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/filter/TweetInfoHealthFilterBase.scala new file mode 100644 index 0000000000..5ea248424c --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/filter/TweetInfoHealthFilterBase.scala @@ -0,0 +1,39 @@ +package com.twitter.cr_mixer.filter + +import com.twitter.contentrecommender.thriftscala.TweetInfo +import com.twitter.cr_mixer.model.CandidateGeneratorQuery +import com.twitter.cr_mixer.model.CrCandidateGeneratorQuery +import com.twitter.cr_mixer.model.HealthThreshold +import com.twitter.cr_mixer.model.InitialCandidate +import com.twitter.util.Future +import javax.inject.Singleton + +@Singleton +trait TweetInfoHealthFilterBase extends FilterBase { + override def name: String = this.getClass.getCanonicalName + override type ConfigType = HealthThreshold.Enum.Value + def thresholdToPropertyMap: Map[HealthThreshold.Enum.Value, TweetInfo => Option[Boolean]] + def getFilterParamFn: CandidateGeneratorQuery => HealthThreshold.Enum.Value + + override def filter( + candidates: Seq[Seq[InitialCandidate]], + config: HealthThreshold.Enum.Value + ): Future[Seq[Seq[InitialCandidate]]] = { + Future.value(candidates.map { seq => + seq.filter(p => thresholdToPropertyMap(config)(p.tweetInfo).getOrElse(true)) + }) + } + + /** + * Build the config params here. passing in param() into the filter is strongly discouraged + * because param() can be slow when called many times + */ + override def requestToConfig[CGQueryType <: CandidateGeneratorQuery]( + query: CGQueryType + ): HealthThreshold.Enum.Value = { + query match { + case q: CrCandidateGeneratorQuery => getFilterParamFn(q) + case _ => HealthThreshold.Enum.Off + } + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/filter/UtegFilterRunner.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/filter/UtegFilterRunner.scala new file mode 100644 index 0000000000..463e026b9f --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/filter/UtegFilterRunner.scala @@ -0,0 +1,96 @@ +package com.twitter.cr_mixer.filter + +import com.twitter.cr_mixer.model.CandidateGeneratorQuery +import com.twitter.cr_mixer.model.InitialCandidate +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.util.Future + +import javax.inject.Inject +import javax.inject.Singleton + +/*** + * + * Run filters sequentially for UTEG candidate generator. The structure is copied from PreRankFilterRunner. + */ +@Singleton +class UtegFilterRunner @Inject() ( + inNetworkFilter: InNetworkFilter, + utegHealthFilter: UtegHealthFilter, + retweetFilter: RetweetFilter, + globalStats: StatsReceiver) { + + private val scopedStats = globalStats.scope(this.getClass.getCanonicalName) + + val orderedFilters: Seq[FilterBase] = Seq( + inNetworkFilter, + utegHealthFilter, + retweetFilter + ) + + def runSequentialFilters[CGQueryType <: CandidateGeneratorQuery]( + request: CGQueryType, + candidates: Seq[Seq[InitialCandidate]], + ): Future[Seq[Seq[InitialCandidate]]] = { + UtegFilterRunner.runSequentialFilters( + request, + candidates, + orderedFilters, + scopedStats + ) + } + +} + +object UtegFilterRunner { + private def recordCandidateStatsBeforeFilter( + candidates: Seq[Seq[InitialCandidate]], + statsReceiver: StatsReceiver + ): Unit = { + statsReceiver + .counter("empty_sources", "before").incr( + candidates.count { + _.isEmpty + } + ) + candidates.foreach { candidate => + statsReceiver.counter("candidates", "before").incr(candidate.size) + } + } + + private def recordCandidateStatsAfterFilter( + candidates: Seq[Seq[InitialCandidate]], + statsReceiver: StatsReceiver + ): Unit = { + statsReceiver + .counter("empty_sources", "after").incr( + candidates.count { + _.isEmpty + } + ) + candidates.foreach { candidate => + statsReceiver.counter("candidates", "after").incr(candidate.size) + } + } + + /* + Helper function for running some candidates through a sequence of filters + */ + private[filter] def runSequentialFilters[CGQueryType <: CandidateGeneratorQuery]( + request: CGQueryType, + candidates: Seq[Seq[InitialCandidate]], + filters: Seq[FilterBase], + statsReceiver: StatsReceiver + ): Future[Seq[Seq[InitialCandidate]]] = + filters.foldLeft(Future.value(candidates)) { + case (candsFut, filter) => + candsFut.flatMap { cands => + recordCandidateStatsBeforeFilter(cands, statsReceiver.scope(filter.name)) + filter + .filter(cands, filter.requestToConfig(request)) + .map { filteredCands => + recordCandidateStatsAfterFilter(filteredCands, statsReceiver.scope(filter.name)) + filteredCands + } + } + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/filter/UtegHealthFilter.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/filter/UtegHealthFilter.scala new file mode 100644 index 0000000000..4a327b1610 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/filter/UtegHealthFilter.scala @@ -0,0 +1,51 @@ +package com.twitter.cr_mixer.filter + +import com.twitter.cr_mixer.model.CandidateGeneratorQuery +import com.twitter.cr_mixer.model.InitialCandidate +import com.twitter.cr_mixer.param.UtegTweetGlobalParams +import com.twitter.util.Future + +import javax.inject.Inject +import javax.inject.Singleton + +/** + * Remove unhealthy candidates + * Currently Timeline Ranker applies a check on the following three scores: + * - toxicityScore + * - pBlockScore + * - pReportedTweetScore + * + * Where isPassTweetHealthFilterStrict checks two additions scores with the same threshold: + * - pSpammyTweetScore + * - spammyTweetContentScore + * + * We've verified that both filters behave very similarly. + */ +@Singleton +case class UtegHealthFilter @Inject() () extends FilterBase { + override def name: String = this.getClass.getCanonicalName + override type ConfigType = Boolean + + override def filter( + candidates: Seq[Seq[InitialCandidate]], + config: ConfigType + ): Future[Seq[Seq[InitialCandidate]]] = { + if (config) { + Future.value( + candidates.map { candidateSeq => + candidateSeq.filter { candidate => + candidate.tweetInfo.isPassTweetHealthFilterStrict.getOrElse(false) + } + } + ) + } else { + Future.value(candidates) + } + } + + override def requestToConfig[CGQueryType <: CandidateGeneratorQuery]( + query: CGQueryType + ): ConfigType = { + query.params(UtegTweetGlobalParams.EnableTLRHealthFilterParam) + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/filter/VideoTweetFilter.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/filter/VideoTweetFilter.scala new file mode 100644 index 0000000000..755ba8ac78 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/filter/VideoTweetFilter.scala @@ -0,0 +1,81 @@ +package com.twitter.cr_mixer.filter + +import com.twitter.cr_mixer.filter.VideoTweetFilter.FilterConfig +import com.twitter.cr_mixer.model.CandidateGeneratorQuery +import com.twitter.cr_mixer.model.CrCandidateGeneratorQuery +import com.twitter.cr_mixer.model.InitialCandidate +import com.twitter.cr_mixer.model.RelatedTweetCandidateGeneratorQuery +import com.twitter.cr_mixer.model.RelatedVideoTweetCandidateGeneratorQuery +import com.twitter.cr_mixer.param.VideoTweetFilterParams +import com.twitter.util.Future +import javax.inject.Singleton + +@Singleton +case class VideoTweetFilter() extends FilterBase { + override val name: String = this.getClass.getCanonicalName + + override type ConfigType = FilterConfig + + override def filter( + candidates: Seq[Seq[InitialCandidate]], + config: ConfigType + ): Future[Seq[Seq[InitialCandidate]]] = { + Future.value(candidates.map { + _.flatMap { + candidate => + if (!config.enableVideoTweetFilter) { + Some(candidate) + } else { + // if hasVideo is true, hasImage, hasGif should be false + val hasVideo = checkTweetInfoAttribute(candidate.tweetInfo.hasVideo) + val isHighMediaResolution = + checkTweetInfoAttribute(candidate.tweetInfo.isHighMediaResolution) + val isQuoteTweet = checkTweetInfoAttribute(candidate.tweetInfo.isQuoteTweet) + val isReply = checkTweetInfoAttribute(candidate.tweetInfo.isReply) + val hasMultipleMedia = checkTweetInfoAttribute(candidate.tweetInfo.hasMultipleMedia) + val hasUrl = checkTweetInfoAttribute(candidate.tweetInfo.hasUrl) + + if (hasVideo && isHighMediaResolution && !isQuoteTweet && + !isReply && !hasMultipleMedia && !hasUrl) { + Some(candidate) + } else { + None + } + } + } + }) + } + + def checkTweetInfoAttribute(attributeOpt: => Option[Boolean]): Boolean = { + if (attributeOpt.isDefined) + attributeOpt.get + else { + // takes Quoted Tweet (TweetInfo.isQuoteTweet) as an example, + // if the attributeOpt is None, we by default say it is not a quoted tweet + // similarly, if TweetInfo.hasVideo is a None, + // we say it does not have video. + false + } + } + + override def requestToConfig[CGQueryType <: CandidateGeneratorQuery]( + query: CGQueryType + ): FilterConfig = { + val enableVideoTweetFilter = query match { + case _: CrCandidateGeneratorQuery | _: RelatedTweetCandidateGeneratorQuery | + _: RelatedVideoTweetCandidateGeneratorQuery => + query.params(VideoTweetFilterParams.EnableVideoTweetFilterParam) + case _ => false // e.g., GetRelatedTweets() + } + FilterConfig( + enableVideoTweetFilter = enableVideoTweetFilter + ) + } +} + +object VideoTweetFilter { + // extend the filterConfig to add more flags if needed. + // now they are hardcoded according to the prod setting + case class FilterConfig( + enableVideoTweetFilter: Boolean) +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/logging/AdsRecommendationsScribeLogger.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/logging/AdsRecommendationsScribeLogger.scala new file mode 100644 index 0000000000..f786bd5867 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/logging/AdsRecommendationsScribeLogger.scala @@ -0,0 +1,139 @@ +package com.twitter.cr_mixer.logging + +import com.twitter.cr_mixer.model.AdsCandidateGeneratorQuery +import com.twitter.cr_mixer.model.InitialAdsCandidate +import com.twitter.cr_mixer.model.ModuleNames +import com.twitter.cr_mixer.logging.ScribeLoggerUtils._ +import com.twitter.cr_mixer.param.decider.CrMixerDecider +import com.twitter.cr_mixer.param.decider.DeciderConstants +import com.twitter.cr_mixer.thriftscala.AdsRecommendationTopLevelApiResult +import com.twitter.cr_mixer.thriftscala.AdsRecommendationsResult +import com.twitter.cr_mixer.thriftscala.AdsRequest +import com.twitter.cr_mixer.thriftscala.AdsResponse +import com.twitter.cr_mixer.thriftscala.FetchCandidatesResult +import com.twitter.cr_mixer.thriftscala.GetAdsRecommendationsScribe +import com.twitter.cr_mixer.thriftscala.PerformanceMetrics +import com.twitter.cr_mixer.thriftscala.TweetCandidateWithMetadata +import com.twitter.cr_mixer.util.CandidateGenerationKeyUtil +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.finagle.tracing.Trace +import com.twitter.logging.Logger +import com.twitter.simclusters_v2.common.UserId +import com.twitter.util.Future +import com.twitter.util.Stopwatch + +import javax.inject.Inject +import javax.inject.Named +import javax.inject.Singleton + +@Singleton +case class AdsRecommendationsScribeLogger @Inject() ( + @Named(ModuleNames.AdsRecommendationsLogger) adsRecommendationsScribeLogger: Logger, + decider: CrMixerDecider, + statsReceiver: StatsReceiver) { + + private val scopedStats = statsReceiver.scope(this.getClass.getCanonicalName) + private val upperFunnelsStats = scopedStats.scope("UpperFunnels") + private val topLevelApiStats = scopedStats.scope("TopLevelApi") + + /* + * Scribe first step results after fetching initial ads candidate + * */ + def scribeInitialAdsCandidates( + query: AdsCandidateGeneratorQuery, + getResultFn: => Future[Seq[Seq[InitialAdsCandidate]]], + enableScribe: Boolean // controlled by feature switch so that we can scribe for certain DDG + ): Future[Seq[Seq[InitialAdsCandidate]]] = { + val scribeMetadata = ScribeMetadata.from(query) + val timer = Stopwatch.start() + getResultFn.onSuccess { input => + val latencyMs = timer().inMilliseconds + val result = convertFetchCandidatesResult(input, scribeMetadata.userId) + val traceId = Trace.id.traceId.toLong + val scribeMsg = buildScribeMessage(result, scribeMetadata, latencyMs, traceId) + + if (enableScribe && decider.isAvailableForId( + scribeMetadata.userId, + DeciderConstants.adsRecommendationsPerExperimentScribeRate)) { + upperFunnelsStats.counter(scribeMetadata.product.originalName).incr() + scribeResult(scribeMsg) + } + } + } + + /* + * Scribe top level API results + * */ + def scribeGetAdsRecommendations( + request: AdsRequest, + startTime: Long, + scribeMetadata: ScribeMetadata, + getResultFn: => Future[AdsResponse], + enableScribe: Boolean + ): Future[AdsResponse] = { + val timer = Stopwatch.start() + getResultFn.onSuccess { response => + val latencyMs = timer().inMilliseconds + val result = AdsRecommendationsResult.AdsRecommendationTopLevelApiResult( + AdsRecommendationTopLevelApiResult( + timestamp = startTime, + request = request, + response = response + )) + val traceId = Trace.id.traceId.toLong + val scribeMsg = buildScribeMessage(result, scribeMetadata, latencyMs, traceId) + + if (enableScribe && decider.isAvailableForId( + scribeMetadata.userId, + DeciderConstants.adsRecommendationsPerExperimentScribeRate)) { + topLevelApiStats.counter(scribeMetadata.product.originalName).incr() + scribeResult(scribeMsg) + } + } + } + + private def convertFetchCandidatesResult( + candidatesSeq: Seq[Seq[InitialAdsCandidate]], + requestUserId: UserId + ): AdsRecommendationsResult = { + val tweetCandidatesWithMetadata = candidatesSeq.flatMap { candidates => + candidates.map { candidate => + TweetCandidateWithMetadata( + tweetId = candidate.tweetId, + candidateGenerationKey = Some( + CandidateGenerationKeyUtil.toThrift(candidate.candidateGenerationInfo, requestUserId)), + score = Some(candidate.getSimilarityScore), + numCandidateGenerationKeys = None // not populated yet + ) + } + } + AdsRecommendationsResult.FetchCandidatesResult( + FetchCandidatesResult(Some(tweetCandidatesWithMetadata))) + } + + private def buildScribeMessage( + result: AdsRecommendationsResult, + scribeMetadata: ScribeMetadata, + latencyMs: Long, + traceId: Long + ): GetAdsRecommendationsScribe = { + GetAdsRecommendationsScribe( + uuid = scribeMetadata.requestUUID, + userId = scribeMetadata.userId, + result = result, + traceId = Some(traceId), + performanceMetrics = Some(PerformanceMetrics(Some(latencyMs))), + impressedBuckets = getImpressedBuckets(scopedStats) + ) + } + + private def scribeResult( + scribeMsg: GetAdsRecommendationsScribe + ): Unit = { + publish( + logger = adsRecommendationsScribeLogger, + codec = GetAdsRecommendationsScribe, + message = scribeMsg) + } + +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/logging/BUILD b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/logging/BUILD new file mode 100644 index 0000000000..edf0b77f0b --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/logging/BUILD @@ -0,0 +1,34 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + strict_deps = True, + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/javax/inject:javax.inject", + "abdecider/src/main/scala", + "content-recommender/thrift/src/main/thrift:content-recommender-common-scala", + "cr-mixer/server/src/main/scala/com/twitter/cr_mixer/featureswitch", + "cr-mixer/server/src/main/scala/com/twitter/cr_mixer/model", + "cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param", + "cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/decider", + "cr-mixer/server/src/main/scala/com/twitter/cr_mixer/scribe", + "cr-mixer/server/src/main/scala/com/twitter/cr_mixer/util", + "cr-mixer/thrift/src/main/thrift:thrift-scala", + "decider/src/main/scala", + "featureswitches/featureswitches-core/src/main/scala:experimentation-settings", + "finagle/finagle-core/src/main", + "frigate/frigate-common:base", + "frigate/frigate-common:util", + "frigate/frigate-common/src/main/scala/com/twitter/frigate/common/base", + "kafka/finagle-kafka/finatra-kafka/src/main/scala", + "product-mixer/core/src/main/thrift/com/twitter/product_mixer/core:thrift-scala", + "scribelib/marshallers/src/main/scala/com/twitter/scribelib/marshallers", + "scribelib/validators/src/main/scala/com/twitter/scribelib/validators", + "scrooge/scrooge-serializer/src/main/scala", + "src/scala/com/twitter/simclusters_v2/common", + "src/thrift/com/twitter/ml/api:data-scala", + "src/thrift/com/twitter/simclusters_v2:simclusters_v2-thrift-scala", + "timelines/src/main/scala/com/twitter/timelines/clientevent", + "util-internal/scribe/src/main/scala/com/twitter/logging", + ], +) diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/logging/CrMixerScribeLogger.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/logging/CrMixerScribeLogger.scala new file mode 100644 index 0000000000..024dcf55b8 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/logging/CrMixerScribeLogger.scala @@ -0,0 +1,489 @@ +package com.twitter.cr_mixer.logging + +import com.google.common.base.CaseFormat +import com.twitter.abdecider.ScribingABDeciderUtil +import com.twitter.scribelib.marshallers.ClientDataProvider +import com.twitter.scribelib.marshallers.ScribeSerialization +import com.twitter.timelines.clientevent.MinimalClientDataProvider +import com.twitter.cr_mixer.model.BlendedCandidate +import com.twitter.cr_mixer.model.CrCandidateGeneratorQuery +import com.twitter.cr_mixer.model.InitialCandidate +import com.twitter.cr_mixer.model.RankedCandidate +import com.twitter.cr_mixer.logging.ScribeLoggerUtils._ +import com.twitter.cr_mixer.model.GraphSourceInfo +import com.twitter.cr_mixer.param.decider.CrMixerDecider +import com.twitter.cr_mixer.param.decider.DeciderConstants +import com.twitter.cr_mixer.scribe.ScribeCategories +import com.twitter.cr_mixer.thriftscala.CrMixerTweetRequest +import com.twitter.cr_mixer.thriftscala.CrMixerTweetResponse +import com.twitter.cr_mixer.thriftscala.FetchCandidatesResult +import com.twitter.cr_mixer.thriftscala.FetchSignalSourcesResult +import com.twitter.cr_mixer.thriftscala.GetTweetsRecommendationsScribe +import com.twitter.cr_mixer.thriftscala.InterleaveResult +import com.twitter.cr_mixer.thriftscala.PerformanceMetrics +import com.twitter.cr_mixer.thriftscala.PreRankFilterResult +import com.twitter.cr_mixer.thriftscala.Product +import com.twitter.cr_mixer.thriftscala.RankResult +import com.twitter.cr_mixer.thriftscala.Result +import com.twitter.cr_mixer.thriftscala.SourceSignal +import com.twitter.cr_mixer.thriftscala.TopLevelApiResult +import com.twitter.cr_mixer.thriftscala.TweetCandidateWithMetadata +import com.twitter.cr_mixer.thriftscala.VITTweetCandidateScribe +import com.twitter.cr_mixer.thriftscala.VITTweetCandidatesScribe +import com.twitter.cr_mixer.model.ModuleNames +import com.twitter.cr_mixer.model.SourceInfo +import com.twitter.cr_mixer.util.CandidateGenerationKeyUtil +import com.twitter.cr_mixer.util.MetricTagUtil +import com.twitter.decider.SimpleRecipient +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.finagle.tracing.Trace +import com.twitter.finatra.kafka.producers.KafkaProducerBase +import com.twitter.logging.Logger +import com.twitter.simclusters_v2.common.UserId +import com.twitter.simclusters_v2.thriftscala.InternalId +import com.twitter.util.Future +import com.twitter.util.Stopwatch +import com.twitter.util.Time + +import javax.inject.Inject +import javax.inject.Named +import javax.inject.Singleton +import scala.util.Random + +@Singleton +case class CrMixerScribeLogger @Inject() ( + decider: CrMixerDecider, + statsReceiver: StatsReceiver, + @Named(ModuleNames.TweetRecsLogger) tweetRecsScribeLogger: Logger, + @Named(ModuleNames.BlueVerifiedTweetRecsLogger) blueVerifiedTweetRecsScribeLogger: Logger, + @Named(ModuleNames.TopLevelApiDdgMetricsLogger) ddgMetricsLogger: Logger, + kafkaProducer: KafkaProducerBase[String, GetTweetsRecommendationsScribe]) { + + import CrMixerScribeLogger._ + + private val scopedStats = statsReceiver.scope("CrMixerScribeLogger") + private val topLevelApiStats = scopedStats.scope("TopLevelApi") + private val upperFunnelsStats = scopedStats.scope("UpperFunnels") + private val kafkaMessagesStats = scopedStats.scope("KafkaMessages") + private val topLevelApiDdgMetricsStats = scopedStats.scope("TopLevelApiDdgMetrics") + private val blueVerifiedTweetCandidatesStats = scopedStats.scope("BlueVerifiedTweetCandidates") + + private val serialization = new ScribeSerialization {} + + def scribeSignalSources( + query: CrCandidateGeneratorQuery, + getResultFn: => Future[(Set[SourceInfo], Map[String, Option[GraphSourceInfo]])] + ): Future[(Set[SourceInfo], Map[String, Option[GraphSourceInfo]])] = { + scribeResultsAndPerformanceMetrics( + ScribeMetadata.from(query), + getResultFn, + convertToResultFn = convertFetchSignalSourcesResult + ) + } + + def scribeInitialCandidates( + query: CrCandidateGeneratorQuery, + getResultFn: => Future[Seq[Seq[InitialCandidate]]] + ): Future[Seq[Seq[InitialCandidate]]] = { + scribeResultsAndPerformanceMetrics( + ScribeMetadata.from(query), + getResultFn, + convertToResultFn = convertFetchCandidatesResult + ) + } + + def scribePreRankFilterCandidates( + query: CrCandidateGeneratorQuery, + getResultFn: => Future[Seq[Seq[InitialCandidate]]] + ): Future[Seq[Seq[InitialCandidate]]] = { + scribeResultsAndPerformanceMetrics( + ScribeMetadata.from(query), + getResultFn, + convertToResultFn = convertPreRankFilterResult + ) + } + + def scribeInterleaveCandidates( + query: CrCandidateGeneratorQuery, + getResultFn: => Future[Seq[BlendedCandidate]] + ): Future[Seq[BlendedCandidate]] = { + scribeResultsAndPerformanceMetrics( + ScribeMetadata.from(query), + getResultFn, + convertToResultFn = convertInterleaveResult, + enableKafkaScribe = true + ) + } + + def scribeRankedCandidates( + query: CrCandidateGeneratorQuery, + getResultFn: => Future[Seq[RankedCandidate]] + ): Future[Seq[RankedCandidate]] = { + scribeResultsAndPerformanceMetrics( + ScribeMetadata.from(query), + getResultFn, + convertToResultFn = convertRankResult + ) + } + + /** + * Scribe Top Level API Request / Response and performance metrics + * for the getTweetRecommendations() endpoint. + */ + def scribeGetTweetRecommendations( + request: CrMixerTweetRequest, + startTime: Long, + scribeMetadata: ScribeMetadata, + getResultFn: => Future[CrMixerTweetResponse] + ): Future[CrMixerTweetResponse] = { + val timer = Stopwatch.start() + getResultFn.onSuccess { response => + val latencyMs = timer().inMilliseconds + val result = convertTopLevelAPIResult(request, response, startTime) + val traceId = Trace.id.traceId.toLong + val scribeMsg = buildScribeMessage(result, scribeMetadata, latencyMs, traceId) + + // We use upperFunnelPerStepScribeRate to cover TopLevelApi scribe logs + if (decider.isAvailableForId( + scribeMetadata.userId, + DeciderConstants.upperFunnelPerStepScribeRate)) { + topLevelApiStats.counter(scribeMetadata.product.originalName).incr() + scribeResult(scribeMsg) + } + if (decider.isAvailableForId( + scribeMetadata.userId, + DeciderConstants.topLevelApiDdgMetricsScribeRate)) { + topLevelApiDdgMetricsStats.counter(scribeMetadata.product.originalName).incr() + val topLevelDdgMetricsMetadata = TopLevelDdgMetricsMetadata.from(request) + publishTopLevelDdgMetrics( + logger = ddgMetricsLogger, + topLevelDdgMetricsMetadata = topLevelDdgMetricsMetadata, + latencyMs = latencyMs, + candidateSize = response.tweets.length) + } + } + } + + /** + * Scribe all of the Blue Verified tweets that are candidates from cr-mixer + * from the getTweetRecommendations() endpoint for stats tracking/debugging purposes. + */ + def scribeGetTweetRecommendationsForBlueVerified( + scribeMetadata: ScribeMetadata, + getResultFn: => Future[Seq[RankedCandidate]] + ): Future[Seq[RankedCandidate]] = { + getResultFn.onSuccess { rankedCandidates => + if (decider.isAvailable(DeciderConstants.enableScribeForBlueVerifiedTweetCandidates)) { + blueVerifiedTweetCandidatesStats.counter("process_request").incr() + + val blueVerifiedTweetCandidates = rankedCandidates.filter { tweet => + tweet.tweetInfo.hasBlueVerifiedAnnotation.contains(true) + } + + val impressedBuckets = getImpressedBuckets(blueVerifiedTweetCandidatesStats).getOrElse(Nil) + + val blueVerifiedCandidateScribes = blueVerifiedTweetCandidates.map { candidate => + blueVerifiedTweetCandidatesStats + .scope(scribeMetadata.product.name).counter( + candidate.tweetInfo.authorId.toString).incr() + VITTweetCandidateScribe( + tweetId = candidate.tweetId, + authorId = candidate.tweetInfo.authorId, + score = candidate.predictionScore, + metricTags = MetricTagUtil.buildMetricTags(candidate) + ) + } + + val blueVerifiedScribe = + VITTweetCandidatesScribe( + uuid = scribeMetadata.requestUUID, + userId = scribeMetadata.userId, + candidates = blueVerifiedCandidateScribes, + product = scribeMetadata.product, + impressedBuckets = impressedBuckets + ) + + publish( + logger = blueVerifiedTweetRecsScribeLogger, + codec = VITTweetCandidatesScribe, + message = blueVerifiedScribe) + } + } + } + + /** + * Scribe Per-step intermediate results and performance metrics + * for each step: fetch signals, fetch candidates, filters, ranker, etc + */ + private[logging] def scribeResultsAndPerformanceMetrics[T]( + scribeMetadata: ScribeMetadata, + getResultFn: => Future[T], + convertToResultFn: (T, UserId) => Result, + enableKafkaScribe: Boolean = false + ): Future[T] = { + val timer = Stopwatch.start() + getResultFn.onSuccess { input => + val latencyMs = timer().inMilliseconds + val result = convertToResultFn(input, scribeMetadata.userId) + val traceId = Trace.id.traceId.toLong + val scribeMsg = buildScribeMessage(result, scribeMetadata, latencyMs, traceId) + + if (decider.isAvailableForId( + scribeMetadata.userId, + DeciderConstants.upperFunnelPerStepScribeRate)) { + upperFunnelsStats.counter(scribeMetadata.product.originalName).incr() + scribeResult(scribeMsg) + } + + // forks the scribe as a Kafka message for async feature hydration + if (enableKafkaScribe && shouldScribeKafkaMessage( + scribeMetadata.userId, + scribeMetadata.product)) { + kafkaMessagesStats.counter(scribeMetadata.product.originalName).incr() + + val batchedKafkaMessages = downsampleKafkaMessage(scribeMsg) + batchedKafkaMessages.foreach { kafkaMessage => + kafkaProducer.send( + topic = ScribeCategories.TweetsRecs.scribeCategory, + key = traceId.toString, + value = kafkaMessage, + timestamp = Time.now.inMilliseconds + ) + } + } + } + } + + private def convertTopLevelAPIResult( + request: CrMixerTweetRequest, + response: CrMixerTweetResponse, + startTime: Long + ): Result = { + Result.TopLevelApiResult( + TopLevelApiResult( + timestamp = startTime, + request = request, + response = response + )) + } + + private def convertFetchSignalSourcesResult( + sourceInfoSetTuple: (Set[SourceInfo], Map[String, Option[GraphSourceInfo]]), + requestUserId: UserId + ): Result = { + val sourceSignals = sourceInfoSetTuple._1.map { sourceInfo => + SourceSignal(id = Some(sourceInfo.internalId)) + } + // For source graphs, we pass in requestUserId as a placeholder + val sourceGraphs = sourceInfoSetTuple._2.map { + case (_, _) => + SourceSignal(id = Some(InternalId.UserId(requestUserId))) + } + Result.FetchSignalSourcesResult( + FetchSignalSourcesResult( + signals = Some(sourceSignals ++ sourceGraphs) + )) + } + + private def convertFetchCandidatesResult( + candidatesSeq: Seq[Seq[InitialCandidate]], + requestUserId: UserId + ): Result = { + val tweetCandidatesWithMetadata = candidatesSeq.flatMap { candidates => + candidates.map { candidate => + TweetCandidateWithMetadata( + tweetId = candidate.tweetId, + candidateGenerationKey = Some( + CandidateGenerationKeyUtil.toThrift(candidate.candidateGenerationInfo, requestUserId)), + score = Some(candidate.getSimilarityScore), + numCandidateGenerationKeys = None // not populated yet + ) + } + } + Result.FetchCandidatesResult(FetchCandidatesResult(Some(tweetCandidatesWithMetadata))) + } + + private def convertPreRankFilterResult( + candidatesSeq: Seq[Seq[InitialCandidate]], + requestUserId: UserId + ): Result = { + val tweetCandidatesWithMetadata = candidatesSeq.flatMap { candidates => + candidates.map { candidate => + TweetCandidateWithMetadata( + tweetId = candidate.tweetId, + candidateGenerationKey = Some( + CandidateGenerationKeyUtil.toThrift(candidate.candidateGenerationInfo, requestUserId)), + score = Some(candidate.getSimilarityScore), + numCandidateGenerationKeys = None // not populated yet + ) + } + } + Result.PreRankFilterResult(PreRankFilterResult(Some(tweetCandidatesWithMetadata))) + } + + // We take InterleaveResult for Unconstrained dataset ML ranker training + private def convertInterleaveResult( + blendedCandidates: Seq[BlendedCandidate], + requestUserId: UserId + ): Result = { + val tweetCandidatesWithMetadata = blendedCandidates.map { blendedCandidate => + val candidateGenerationKey = + CandidateGenerationKeyUtil.toThrift(blendedCandidate.reasonChosen, requestUserId) + TweetCandidateWithMetadata( + tweetId = blendedCandidate.tweetId, + candidateGenerationKey = Some(candidateGenerationKey), + authorId = Some(blendedCandidate.tweetInfo.authorId), // for ML pipeline training + score = Some(blendedCandidate.getSimilarityScore), + numCandidateGenerationKeys = Some(blendedCandidate.potentialReasons.size) + ) // hydrate fields for light ranking training data + } + Result.InterleaveResult(InterleaveResult(Some(tweetCandidatesWithMetadata))) + } + + private def convertRankResult( + rankedCandidates: Seq[RankedCandidate], + requestUserId: UserId + ): Result = { + val tweetCandidatesWithMetadata = rankedCandidates.map { rankedCandidate => + val candidateGenerationKey = + CandidateGenerationKeyUtil.toThrift(rankedCandidate.reasonChosen, requestUserId) + TweetCandidateWithMetadata( + tweetId = rankedCandidate.tweetId, + candidateGenerationKey = Some(candidateGenerationKey), + score = Some(rankedCandidate.getSimilarityScore), + numCandidateGenerationKeys = Some(rankedCandidate.potentialReasons.size) + ) + } + Result.RankResult(RankResult(Some(tweetCandidatesWithMetadata))) + } + + private def buildScribeMessage( + result: Result, + scribeMetadata: ScribeMetadata, + latencyMs: Long, + traceId: Long + ): GetTweetsRecommendationsScribe = { + GetTweetsRecommendationsScribe( + uuid = scribeMetadata.requestUUID, + userId = scribeMetadata.userId, + result = result, + traceId = Some(traceId), + performanceMetrics = Some(PerformanceMetrics(Some(latencyMs))), + impressedBuckets = getImpressedBuckets(scopedStats) + ) + } + + private def scribeResult( + scribeMsg: GetTweetsRecommendationsScribe + ): Unit = { + publish( + logger = tweetRecsScribeLogger, + codec = GetTweetsRecommendationsScribe, + message = scribeMsg) + } + + /** + * Gate for producing messages to Kafka for async feature hydration + */ + private def shouldScribeKafkaMessage( + userId: UserId, + product: Product + ): Boolean = { + val isEligibleUser = decider.isAvailable( + DeciderConstants.kafkaMessageScribeSampleRate, + Some(SimpleRecipient(userId))) + val isHomeProduct = (product == Product.Home) + isEligibleUser && isHomeProduct + } + + /** + * Due to size limits of Strato (see SD-19028), each Kafka message must be downsampled + */ + private[logging] def downsampleKafkaMessage( + scribeMsg: GetTweetsRecommendationsScribe + ): Seq[GetTweetsRecommendationsScribe] = { + val sampledResultSeq: Seq[Result] = scribeMsg.result match { + case Result.InterleaveResult(interleaveResult) => + val sampledTweetsSeq = interleaveResult.tweets + .map { tweets => + Random + .shuffle(tweets).take(KafkaMaxTweetsPerMessage) + .grouped(BatchSize).toSeq + }.getOrElse(Seq.empty) + + sampledTweetsSeq.map { sampledTweets => + Result.InterleaveResult(InterleaveResult(Some(sampledTweets))) + } + + // if it's an unrecognized type, err on the side of sending no candidates + case _ => + kafkaMessagesStats.counter("InvalidKafkaMessageResultType").incr() + Seq(Result.InterleaveResult(InterleaveResult(None))) + } + + sampledResultSeq.map { sampledResult => + GetTweetsRecommendationsScribe( + uuid = scribeMsg.uuid, + userId = scribeMsg.userId, + result = sampledResult, + traceId = scribeMsg.traceId, + performanceMetrics = None, + impressedBuckets = None + ) + } + } + + /** + * Handles client_event serialization to log data into DDG metrics + */ + private[logging] def publishTopLevelDdgMetrics( + logger: Logger, + topLevelDdgMetricsMetadata: TopLevelDdgMetricsMetadata, + candidateSize: Long, + latencyMs: Long, + ): Unit = { + val data = Map[Any, Any]( + "latency_ms" -> latencyMs, + "event_value" -> candidateSize + ) + val label: (String, String) = ("tweetrec", "") + val namespace = getNamespace(topLevelDdgMetricsMetadata, label) + ("action" -> "candidates") + val message = + serialization + .serializeClientEvent(namespace, getClientData(topLevelDdgMetricsMetadata), data) + logger.info(message) + } + + private def getClientData( + topLevelDdgMetricsMetadata: TopLevelDdgMetricsMetadata + ): ClientDataProvider = + MinimalClientDataProvider( + userId = topLevelDdgMetricsMetadata.userId, + guestId = None, + clientApplicationId = topLevelDdgMetricsMetadata.clientApplicationId, + countryCode = topLevelDdgMetricsMetadata.countryCode + ) + + private def getNamespace( + topLevelDdgMetricsMetadata: TopLevelDdgMetricsMetadata, + label: (String, String) + ): Map[String, String] = { + val productName = + CaseFormat.UPPER_CAMEL + .to(CaseFormat.LOWER_UNDERSCORE, topLevelDdgMetricsMetadata.product.originalName) + + Map( + "client" -> ScribingABDeciderUtil.clientForAppId( + topLevelDdgMetricsMetadata.clientApplicationId), + "page" -> "cr-mixer", + "section" -> productName, + "component" -> label._1, + "element" -> label._2 + ) + } +} + +object CrMixerScribeLogger { + val KafkaMaxTweetsPerMessage: Int = 200 + val BatchSize: Int = 20 +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/logging/RelatedTweetScribeLogger.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/logging/RelatedTweetScribeLogger.scala new file mode 100644 index 0000000000..b2b36f43c5 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/logging/RelatedTweetScribeLogger.scala @@ -0,0 +1,193 @@ +package com.twitter.cr_mixer.logging + +import com.twitter.cr_mixer.model.RelatedTweetCandidateGeneratorQuery +import com.twitter.cr_mixer.model.InitialCandidate +import com.twitter.cr_mixer.model.ModuleNames +import com.twitter.cr_mixer.logging.ScribeLoggerUtils._ +import com.twitter.cr_mixer.param.decider.CrMixerDecider +import com.twitter.cr_mixer.param.decider.DeciderConstants +import com.twitter.cr_mixer.thriftscala.FetchCandidatesResult +import com.twitter.cr_mixer.thriftscala.GetRelatedTweetsScribe +import com.twitter.cr_mixer.thriftscala.PerformanceMetrics +import com.twitter.cr_mixer.thriftscala.PreRankFilterResult +import com.twitter.cr_mixer.thriftscala.RelatedTweetRequest +import com.twitter.cr_mixer.thriftscala.RelatedTweetResponse +import com.twitter.cr_mixer.thriftscala.RelatedTweetResult +import com.twitter.cr_mixer.thriftscala.RelatedTweetTopLevelApiResult +import com.twitter.cr_mixer.thriftscala.TweetCandidateWithMetadata +import com.twitter.cr_mixer.util.CandidateGenerationKeyUtil +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.finagle.tracing.Trace +import com.twitter.logging.Logger +import com.twitter.simclusters_v2.common.UserId +import com.twitter.util.Future +import com.twitter.util.Stopwatch +import javax.inject.Inject +import javax.inject.Named +import javax.inject.Singleton + +@Singleton +case class RelatedTweetScribeLogger @Inject() ( + decider: CrMixerDecider, + statsReceiver: StatsReceiver, + @Named(ModuleNames.RelatedTweetsLogger) relatedTweetsScribeLogger: Logger) { + + private val scopedStats = statsReceiver.scope("RelatedTweetsScribeLogger") + private val topLevelApiStats = scopedStats.scope("TopLevelApi") + private val topLevelApiNoUserIdStats = scopedStats.scope("TopLevelApiNoUserId") + private val upperFunnelsStats = scopedStats.scope("UpperFunnels") + private val upperFunnelsNoUserIdStats = scopedStats.scope("UpperFunnelsNoUserId") + + def scribeInitialCandidates( + query: RelatedTweetCandidateGeneratorQuery, + getResultFn: => Future[Seq[Seq[InitialCandidate]]] + ): Future[Seq[Seq[InitialCandidate]]] = { + scribeResultsAndPerformanceMetrics( + RelatedTweetScribeMetadata.from(query), + getResultFn, + convertToResultFn = convertFetchCandidatesResult + ) + } + + def scribePreRankFilterCandidates( + query: RelatedTweetCandidateGeneratorQuery, + getResultFn: => Future[Seq[Seq[InitialCandidate]]] + ): Future[Seq[Seq[InitialCandidate]]] = { + scribeResultsAndPerformanceMetrics( + RelatedTweetScribeMetadata.from(query), + getResultFn, + convertToResultFn = convertPreRankFilterResult + ) + } + + /** + * Scribe Top Level API Request / Response and performance metrics + * for the getRelatedTweets endpoint. + */ + def scribeGetRelatedTweets( + request: RelatedTweetRequest, + startTime: Long, + relatedTweetScribeMetadata: RelatedTweetScribeMetadata, + getResultFn: => Future[RelatedTweetResponse] + ): Future[RelatedTweetResponse] = { + val timer = Stopwatch.start() + getResultFn.onSuccess { response => + relatedTweetScribeMetadata.clientContext.userId match { + case Some(userId) => + if (decider.isAvailableForId(userId, DeciderConstants.upperFunnelPerStepScribeRate)) { + topLevelApiStats.counter(relatedTweetScribeMetadata.product.originalName).incr() + val latencyMs = timer().inMilliseconds + val result = convertTopLevelAPIResult(request, response, startTime) + val traceId = Trace.id.traceId.toLong + val scribeMsg = + buildScribeMessage(result, relatedTweetScribeMetadata, latencyMs, traceId) + + scribeResult(scribeMsg) + } + case _ => + topLevelApiNoUserIdStats.counter(relatedTweetScribeMetadata.product.originalName).incr() + } + } + } + + /** + * Scribe Per-step intermediate results and performance metrics + * for each step: fetch candidates, filters. + */ + private def scribeResultsAndPerformanceMetrics[T]( + relatedTweetScribeMetadata: RelatedTweetScribeMetadata, + getResultFn: => Future[T], + convertToResultFn: (T, UserId) => RelatedTweetResult + ): Future[T] = { + val timer = Stopwatch.start() + getResultFn.onSuccess { input => + relatedTweetScribeMetadata.clientContext.userId match { + case Some(userId) => + if (decider.isAvailableForId(userId, DeciderConstants.upperFunnelPerStepScribeRate)) { + upperFunnelsStats.counter(relatedTweetScribeMetadata.product.originalName).incr() + val latencyMs = timer().inMilliseconds + val result = convertToResultFn(input, userId) + val traceId = Trace.id.traceId.toLong + val scribeMsg = + buildScribeMessage(result, relatedTweetScribeMetadata, latencyMs, traceId) + scribeResult(scribeMsg) + } + case _ => + upperFunnelsNoUserIdStats.counter(relatedTweetScribeMetadata.product.originalName).incr() + } + } + } + + private def convertTopLevelAPIResult( + request: RelatedTweetRequest, + response: RelatedTweetResponse, + startTime: Long + ): RelatedTweetResult = { + RelatedTweetResult.RelatedTweetTopLevelApiResult( + RelatedTweetTopLevelApiResult( + timestamp = startTime, + request = request, + response = response + )) + } + + private def convertFetchCandidatesResult( + candidatesSeq: Seq[Seq[InitialCandidate]], + requestUserId: UserId + ): RelatedTweetResult = { + val tweetCandidatesWithMetadata = candidatesSeq.flatMap { candidates => + candidates.map { candidate => + TweetCandidateWithMetadata( + tweetId = candidate.tweetId, + candidateGenerationKey = None + ) // do not hydrate candidateGenerationKey to save cost + } + } + RelatedTweetResult.FetchCandidatesResult( + FetchCandidatesResult(Some(tweetCandidatesWithMetadata))) + } + + private def convertPreRankFilterResult( + candidatesSeq: Seq[Seq[InitialCandidate]], + requestUserId: UserId + ): RelatedTweetResult = { + val tweetCandidatesWithMetadata = candidatesSeq.flatMap { candidates => + candidates.map { candidate => + val candidateGenerationKey = + CandidateGenerationKeyUtil.toThrift(candidate.candidateGenerationInfo, requestUserId) + TweetCandidateWithMetadata( + tweetId = candidate.tweetId, + candidateGenerationKey = Some(candidateGenerationKey), + authorId = Some(candidate.tweetInfo.authorId), + score = Some(candidate.getSimilarityScore), + numCandidateGenerationKeys = None + ) + } + } + RelatedTweetResult.PreRankFilterResult(PreRankFilterResult(Some(tweetCandidatesWithMetadata))) + } + + private def buildScribeMessage( + relatedTweetResult: RelatedTweetResult, + relatedTweetScribeMetadata: RelatedTweetScribeMetadata, + latencyMs: Long, + traceId: Long + ): GetRelatedTweetsScribe = { + GetRelatedTweetsScribe( + uuid = relatedTweetScribeMetadata.requestUUID, + internalId = relatedTweetScribeMetadata.internalId, + relatedTweetResult = relatedTweetResult, + requesterId = relatedTweetScribeMetadata.clientContext.userId, + guestId = relatedTweetScribeMetadata.clientContext.guestId, + traceId = Some(traceId), + performanceMetrics = Some(PerformanceMetrics(Some(latencyMs))), + impressedBuckets = getImpressedBuckets(scopedStats) + ) + } + + private def scribeResult( + scribeMsg: GetRelatedTweetsScribe + ): Unit = { + publish(logger = relatedTweetsScribeLogger, codec = GetRelatedTweetsScribe, message = scribeMsg) + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/logging/ScribeLoggerUtils.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/logging/ScribeLoggerUtils.scala new file mode 100644 index 0000000000..3b30c3f106 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/logging/ScribeLoggerUtils.scala @@ -0,0 +1,43 @@ +package com.twitter.cr_mixer.logging + +import com.twitter.cr_mixer.featureswitch.CrMixerImpressedBuckets +import com.twitter.cr_mixer.thriftscala.ImpressesedBucketInfo +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.frigate.common.util.StatsUtil +import com.twitter.logging.Logger +import com.twitter.scrooge.BinaryThriftStructSerializer +import com.twitter.scrooge.ThriftStruct +import com.twitter.scrooge.ThriftStructCodec + +object ScribeLoggerUtils { + + /** + * Handles base64-encoding, serialization, and publish. + */ + private[logging] def publish[T <: ThriftStruct]( + logger: Logger, + codec: ThriftStructCodec[T], + message: T + ): Unit = { + logger.info(BinaryThriftStructSerializer(codec).toString(message)) + } + + private[logging] def getImpressedBuckets( + scopedStats: StatsReceiver + ): Option[List[ImpressesedBucketInfo]] = { + StatsUtil.trackNonFutureBlockStats(scopedStats.scope("getImpressedBuckets")) { + CrMixerImpressedBuckets.getAllImpressedBuckets.map { listBuckets => + val listBucketsSet = listBuckets.toSet + scopedStats.stat("impressed_buckets").add(listBucketsSet.size) + listBucketsSet.map { bucket => + ImpressesedBucketInfo( + experimentId = bucket.experiment.settings.experimentId.getOrElse(-1L), + bucketName = bucket.name, + version = bucket.experiment.settings.version, + ) + }.toList + } + } + } + +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/logging/ScribeMetadata.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/logging/ScribeMetadata.scala new file mode 100644 index 0000000000..8c0444e38a --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/logging/ScribeMetadata.scala @@ -0,0 +1,45 @@ +package com.twitter.cr_mixer.logging + +import com.twitter.cr_mixer.model.AdsCandidateGeneratorQuery +import com.twitter.cr_mixer.model.CrCandidateGeneratorQuery +import com.twitter.cr_mixer.model.RelatedTweetCandidateGeneratorQuery +import com.twitter.cr_mixer.model.UtegTweetCandidateGeneratorQuery +import com.twitter.cr_mixer.thriftscala.Product +import com.twitter.product_mixer.core.thriftscala.ClientContext +import com.twitter.simclusters_v2.common.UserId +import com.twitter.simclusters_v2.thriftscala.InternalId + +case class ScribeMetadata( + requestUUID: Long, + userId: UserId, + product: Product) + +object ScribeMetadata { + def from(query: CrCandidateGeneratorQuery): ScribeMetadata = { + ScribeMetadata(query.requestUUID, query.userId, query.product) + } + + def from(query: UtegTweetCandidateGeneratorQuery): ScribeMetadata = { + ScribeMetadata(query.requestUUID, query.userId, query.product) + } + + def from(query: AdsCandidateGeneratorQuery): ScribeMetadata = { + ScribeMetadata(query.requestUUID, query.userId, query.product) + } +} + +case class RelatedTweetScribeMetadata( + requestUUID: Long, + internalId: InternalId, + clientContext: ClientContext, + product: Product) + +object RelatedTweetScribeMetadata { + def from(query: RelatedTweetCandidateGeneratorQuery): RelatedTweetScribeMetadata = { + RelatedTweetScribeMetadata( + query.requestUUID, + query.internalId, + query.clientContext, + query.product) + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/logging/TopLevelDdgMetricsMetadata.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/logging/TopLevelDdgMetricsMetadata.scala new file mode 100644 index 0000000000..3dd07e58e3 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/logging/TopLevelDdgMetricsMetadata.scala @@ -0,0 +1,22 @@ +package com.twitter.cr_mixer +package logging + +import com.twitter.cr_mixer.thriftscala.CrMixerTweetRequest +import com.twitter.cr_mixer.thriftscala.Product + +case class TopLevelDdgMetricsMetadata( + userId: Option[Long], + product: Product, + clientApplicationId: Option[Long], + countryCode: Option[String]) + +object TopLevelDdgMetricsMetadata { + def from(request: CrMixerTweetRequest): TopLevelDdgMetricsMetadata = { + TopLevelDdgMetricsMetadata( + userId = request.clientContext.userId, + product = request.product, + clientApplicationId = request.clientContext.appId, + countryCode = request.clientContext.countryCode + ) + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/logging/UtegTweetScribeLogger.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/logging/UtegTweetScribeLogger.scala new file mode 100644 index 0000000000..fb01a419bf --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/logging/UtegTweetScribeLogger.scala @@ -0,0 +1,147 @@ +package com.twitter.cr_mixer.logging + +import com.twitter.cr_mixer.logging.ScribeLoggerUtils._ +import com.twitter.cr_mixer.model.UtegTweetCandidateGeneratorQuery +import com.twitter.cr_mixer.model.ModuleNames +import com.twitter.cr_mixer.model.TweetWithScoreAndSocialProof +import com.twitter.cr_mixer.param.decider.CrMixerDecider +import com.twitter.cr_mixer.param.decider.DeciderConstants +import com.twitter.cr_mixer.thriftscala.UtegTweetRequest +import com.twitter.cr_mixer.thriftscala.UtegTweetResponse +import com.twitter.cr_mixer.thriftscala.FetchCandidatesResult +import com.twitter.cr_mixer.thriftscala.GetUtegTweetsScribe +import com.twitter.cr_mixer.thriftscala.PerformanceMetrics +import com.twitter.cr_mixer.thriftscala.UtegTweetResult +import com.twitter.cr_mixer.thriftscala.UtegTweetTopLevelApiResult +import com.twitter.cr_mixer.thriftscala.TweetCandidateWithMetadata +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.finagle.tracing.Trace +import com.twitter.logging.Logger +import com.twitter.simclusters_v2.common.UserId +import com.twitter.util.Future +import com.twitter.util.Stopwatch +import javax.inject.Inject +import javax.inject.Named +import javax.inject.Singleton + +@Singleton +case class UtegTweetScribeLogger @Inject() ( + decider: CrMixerDecider, + statsReceiver: StatsReceiver, + @Named(ModuleNames.UtegTweetsLogger) utegTweetScribeLogger: Logger) { + + private val scopedStats = statsReceiver.scope("UtegTweetScribeLogger") + private val topLevelApiStats = scopedStats.scope("TopLevelApi") + private val upperFunnelsStats = scopedStats.scope("UpperFunnels") + + def scribeInitialCandidates( + query: UtegTweetCandidateGeneratorQuery, + getResultFn: => Future[Seq[TweetWithScoreAndSocialProof]] + ): Future[Seq[TweetWithScoreAndSocialProof]] = { + scribeResultsAndPerformanceMetrics( + ScribeMetadata.from(query), + getResultFn, + convertToResultFn = convertFetchCandidatesResult + ) + } + + /** + * Scribe Top Level API Request / Response and performance metrics + * for the GetUtegTweetRecommendations() endpoint. + */ + def scribeGetUtegTweetRecommendations( + request: UtegTweetRequest, + startTime: Long, + scribeMetadata: ScribeMetadata, + getResultFn: => Future[UtegTweetResponse] + ): Future[UtegTweetResponse] = { + val timer = Stopwatch.start() + getResultFn.onSuccess { response => + if (decider.isAvailableForId( + scribeMetadata.userId, + DeciderConstants.upperFunnelPerStepScribeRate)) { + topLevelApiStats.counter(scribeMetadata.product.originalName).incr() + val latencyMs = timer().inMilliseconds + val result = convertTopLevelAPIResult(request, response, startTime) + val traceId = Trace.id.traceId.toLong + val scribeMsg = + buildScribeMessage(result, scribeMetadata, latencyMs, traceId) + + scribeResult(scribeMsg) + } + } + } + + private def convertTopLevelAPIResult( + request: UtegTweetRequest, + response: UtegTweetResponse, + startTime: Long + ): UtegTweetResult = { + UtegTweetResult.UtegTweetTopLevelApiResult( + UtegTweetTopLevelApiResult( + timestamp = startTime, + request = request, + response = response + )) + } + + private def buildScribeMessage( + utegTweetResult: UtegTweetResult, + scribeMetadata: ScribeMetadata, + latencyMs: Long, + traceId: Long + ): GetUtegTweetsScribe = { + GetUtegTweetsScribe( + uuid = scribeMetadata.requestUUID, + userId = scribeMetadata.userId, + utegTweetResult = utegTweetResult, + traceId = Some(traceId), + performanceMetrics = Some(PerformanceMetrics(Some(latencyMs))), + impressedBuckets = getImpressedBuckets(scopedStats) + ) + } + + private def scribeResult( + scribeMsg: GetUtegTweetsScribe + ): Unit = { + publish(logger = utegTweetScribeLogger, codec = GetUtegTweetsScribe, message = scribeMsg) + } + + private def convertFetchCandidatesResult( + candidates: Seq[TweetWithScoreAndSocialProof], + requestUserId: UserId + ): UtegTweetResult = { + val tweetCandidatesWithMetadata = candidates.map { candidate => + TweetCandidateWithMetadata( + tweetId = candidate.tweetId, + candidateGenerationKey = None + ) // do not hydrate candidateGenerationKey to save cost + } + UtegTweetResult.FetchCandidatesResult(FetchCandidatesResult(Some(tweetCandidatesWithMetadata))) + } + + /** + * Scribe Per-step intermediate results and performance metrics + * for each step: fetch candidates, filters. + */ + private def scribeResultsAndPerformanceMetrics[T]( + scribeMetadata: ScribeMetadata, + getResultFn: => Future[T], + convertToResultFn: (T, UserId) => UtegTweetResult + ): Future[T] = { + val timer = Stopwatch.start() + getResultFn.onSuccess { input => + if (decider.isAvailableForId( + scribeMetadata.userId, + DeciderConstants.upperFunnelPerStepScribeRate)) { + upperFunnelsStats.counter(scribeMetadata.product.originalName).incr() + val latencyMs = timer().inMilliseconds + val result = convertToResultFn(input, scribeMetadata.userId) + val traceId = Trace.id.traceId.toLong + val scribeMsg = + buildScribeMessage(result, scribeMetadata, latencyMs, traceId) + scribeResult(scribeMsg) + } + } + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/model/BUILD b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/model/BUILD new file mode 100644 index 0000000000..87c714254e --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/model/BUILD @@ -0,0 +1,16 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + strict_deps = True, + tags = ["bazel-compatible"], + dependencies = [ + "configapi/configapi-core", + "content-recommender/thrift/src/main/thrift:thrift-scala", + "cr-mixer/thrift/src/main/thrift:thrift-scala", + "product-mixer/core/src/main/thrift/com/twitter/product_mixer/core:thrift-scala", + "src/scala/com/twitter/simclusters_v2/common", + "src/thrift/com/twitter/core_workflows/user_model:user_model-scala", + "src/thrift/com/twitter/recos:recos-common-scala", + "src/thrift/com/twitter/simclusters_v2:simclusters_v2-thrift-scala", + ], +) diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/model/Candidate.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/model/Candidate.scala new file mode 100644 index 0000000000..c357c9472c --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/model/Candidate.scala @@ -0,0 +1,200 @@ +package com.twitter.cr_mixer.model + +import com.twitter.contentrecommender.thriftscala.TweetInfo +import com.twitter.cr_mixer.thriftscala.LineItemInfo +import com.twitter.simclusters_v2.common.TweetId + +sealed trait Candidate { + val tweetId: TweetId + + override def hashCode: Int = tweetId.toInt +} + +case class TweetWithCandidateGenerationInfo( + tweetId: TweetId, + candidateGenerationInfo: CandidateGenerationInfo) + extends Candidate { + + def getSimilarityScore: Double = + candidateGenerationInfo.similarityEngineInfo.score.getOrElse(0.0) +} + +case class InitialCandidate( + tweetId: TweetId, + tweetInfo: TweetInfo, + candidateGenerationInfo: CandidateGenerationInfo) + extends Candidate { + + /** * + * Get the Similarity Score of a Tweet from its CG Info. For instance, + * If it is from a UnifiedTweetBasedSimilarityEngine, the score will be the weighted combined score + * And if it is from a SimClustersANNSimilarityEngine, the score will be the SANN score + */ + def getSimilarityScore: Double = + candidateGenerationInfo.similarityEngineInfo.score.getOrElse(0.0) + + /** + * The same candidate can be generated by multiple algorithms. + * During blending, candidate deduping happens. In order to retain the candidateGenerationInfo + * from different algorithms, we attach them to a list of potentialReasons. + */ + def toBlendedCandidate( + potentialReasons: Seq[CandidateGenerationInfo], + ): BlendedCandidate = { + BlendedCandidate( + tweetId, + tweetInfo, + candidateGenerationInfo, + potentialReasons, + ) + } + + // for experimental purposes only when bypassing interleave / ranking + def toRankedCandidate(): RankedCandidate = { + RankedCandidate( + tweetId, + tweetInfo, + 0.0, // prediction score is default to 0.0 to help differentiate that it is a no-op + candidateGenerationInfo, + Seq(candidateGenerationInfo) + ) + } +} + +case class InitialAdsCandidate( + tweetId: TweetId, + lineItemInfo: Seq[LineItemInfo], + candidateGenerationInfo: CandidateGenerationInfo) + extends Candidate { + + /** * + * Get the Similarity Score of a Tweet from its CG Info. For instance, + * If it is from a UnifiedTweetBasedSimilarityEngine, the score will be the weighted combined score + * And if it is from a SimClustersANNSimilarityEngine, the score will be the SANN score + */ + def getSimilarityScore: Double = + candidateGenerationInfo.similarityEngineInfo.score.getOrElse(0.0) + + /** + * The same candidate can be generated by multiple algorithms. + * During blending, candidate deduping happens. In order to retain the candidateGenerationInfo + * from different algorithms, we attach them to a list of potentialReasons. + */ + def toBlendedAdsCandidate( + potentialReasons: Seq[CandidateGenerationInfo], + ): BlendedAdsCandidate = { + BlendedAdsCandidate( + tweetId, + lineItemInfo, + candidateGenerationInfo, + potentialReasons, + ) + } + + // for experimental purposes only when bypassing interleave / ranking + def toRankedAdsCandidate(): RankedAdsCandidate = { + RankedAdsCandidate( + tweetId, + lineItemInfo, + 0.0, // prediction score is default to 0.0 to help differentiate that it is a no-op + candidateGenerationInfo, + Seq(candidateGenerationInfo) + ) + } +} + +case class BlendedCandidate( + tweetId: TweetId, + tweetInfo: TweetInfo, + reasonChosen: CandidateGenerationInfo, + potentialReasons: Seq[CandidateGenerationInfo]) + extends Candidate { + + /** * + * Get the Similarity Score of a Tweet from its CG Info. For instance, + * If it is from a UnifiedTweetBasedSimilarityEngine, the score will be the weighted combined score + * And if it is from a SimClustersANNSimilarityEngine, the score will be the SANN score + */ + def getSimilarityScore: Double = + reasonChosen.similarityEngineInfo.score.getOrElse(0.0) + + assert(potentialReasons.contains(reasonChosen)) + + def toRankedCandidate(predictionScore: Double): RankedCandidate = { + RankedCandidate( + tweetId, + tweetInfo, + predictionScore, + reasonChosen, + potentialReasons + ) + } +} + +case class BlendedAdsCandidate( + tweetId: TweetId, + lineItemInfo: Seq[LineItemInfo], + reasonChosen: CandidateGenerationInfo, + potentialReasons: Seq[CandidateGenerationInfo]) + extends Candidate { + + /** * + * Get the Similarity Score of a Tweet from its CG Info. For instance, + * If it is from a UnifiedTweetBasedSimilarityEngine, the score will be the weighted combined score + * And if it is from a SimClustersANNSimilarityEngine, the score will be the SANN score + */ + def getSimilarityScore: Double = + reasonChosen.similarityEngineInfo.score.getOrElse(0.0) + + assert(potentialReasons.contains(reasonChosen)) + + def toRankedAdsCandidate(predictionScore: Double): RankedAdsCandidate = { + RankedAdsCandidate( + tweetId, + lineItemInfo, + predictionScore, + reasonChosen, + potentialReasons + ) + } +} + +case class RankedCandidate( + tweetId: TweetId, + tweetInfo: TweetInfo, + predictionScore: Double, + reasonChosen: CandidateGenerationInfo, + potentialReasons: Seq[CandidateGenerationInfo]) + extends Candidate { + + /** * + * Get the Similarity Score of a Tweet from its CG Info. For instance, + * If it is from a UnifiedTweetBasedSimilarityEngine, the score will be the weighted combined score + * And if it is from a SimClustersANNSimilarityEngine, the score will be the SANN score + */ + def getSimilarityScore: Double = + reasonChosen.similarityEngineInfo.score.getOrElse(0.0) + + assert(potentialReasons.contains(reasonChosen)) +} + +case class RankedAdsCandidate( + tweetId: TweetId, + lineItemInfo: Seq[LineItemInfo], + predictionScore: Double, + reasonChosen: CandidateGenerationInfo, + potentialReasons: Seq[CandidateGenerationInfo]) + extends Candidate { + + /** * + * Get the Similarity Score of a Tweet from its CG Info. For instance, + * If it is from a UnifiedTweetBasedSimilarityEngine, the score will be the weighted combined score + * And if it is from a SimClustersANNSimilarityEngine, the score will be the SANN score + */ + def getSimilarityScore: Double = + reasonChosen.similarityEngineInfo.score.getOrElse(0.0) + + assert(potentialReasons.contains(reasonChosen)) +} + +case class TripTweetWithScore(tweetId: TweetId, score: Double) extends Candidate diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/model/CandidateGenerationInfo.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/model/CandidateGenerationInfo.scala new file mode 100644 index 0000000000..879c96b667 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/model/CandidateGenerationInfo.scala @@ -0,0 +1,67 @@ +package com.twitter.cr_mixer.model + +import com.twitter.cr_mixer.thriftscala.SimilarityEngineType +import com.twitter.cr_mixer.thriftscala.SourceType +import com.twitter.simclusters_v2.common.UserId +import com.twitter.simclusters_v2.thriftscala.InternalId +import com.twitter.util.Time + +/*** + * Tweet-level attributes. Represents the source used in candidate generation + * Due to legacy reason, SourceType used to represent both SourceType and SimilarityEngineType + * Moving forward, SourceType will be used for SourceType ONLY. eg., TweetFavorite, UserFollow, TwiceUserId + * At the same time, We create a new SimilarityEngineType to separate them. eg., SimClustersANN + * + * Currently, one special case is that we have TwiceUserId as a source, which is not necessarily a "signal" + * @param sourceType, e.g., SourceType.TweetFavorite, SourceType.UserFollow, SourceType.TwiceUserId + * @param internalId, e.g., UserId(0L), TweetId(0L) + */ +case class SourceInfo( + sourceType: SourceType, + internalId: InternalId, + sourceEventTime: Option[Time]) + +/*** + * Tweet-level attributes. Represents the source User Graph used in candidate generation + * It is an intermediate product, and will not be stored, unlike SourceInfo. + * Essentially, CrMixer queries a graph, and the graph returns a list of users to be used as sources. + * For instance, RealGraph, EarlyBird, FRS, Stp, etc. The underlying similarity engines such as + * UTG or UTEG will leverage these sources to build candidates. + * + * We extended the definition of SourceType to cover both "Source Signal" and "Source Graph" + * See [CrMixer] Graph Based Source Fetcher Abstraction Proposal: + * + * consider making both SourceInfo and GraphSourceInfo extends the same trait to + * have a unified interface. + */ +case class GraphSourceInfo( + sourceType: SourceType, + seedWithScores: Map[UserId, Double]) + +/*** + * Tweet-level attributes. Represents the similarity engine (the algorithm) used for + * candidate generation along with their metadata. + * @param similarityEngineType, e.g., SimClustersANN, UserTweetGraph + * @param modelId. e.g., UserTweetGraphConsumerEmbedding_ALL_20210708 + * @param score - a score generated by this sim engine + */ +case class SimilarityEngineInfo( + similarityEngineType: SimilarityEngineType, + modelId: Option[String], // ModelId can be a None. e.g., UTEG, UnifiedTweetBasedSE. etc + score: Option[Double]) + +/**** + * Tweet-level attributes. A combination for both SourceInfo and SimilarityEngineInfo + * SimilarityEngine is a composition, and it can be composed by many leaf Similarity Engines. + * For instance, the TweetBasedUnified SE could be a composition of both UserTweetGraph SE, SimClustersANN SE. + * Note that a SimilarityEngine (Composite) may call other SimilarityEngines (Atomic, Contributing) + * to contribute to its final candidate list. We track these Contributing SEs in the contributingSimilarityEngines list + * + * @param sourceInfoOpt - this is optional as many consumerBased CG does not have a source + * @param similarityEngineInfo - the similarity engine used in Candidate Generation (eg., TweetBasedUnifiedSE). It can be an atomic SE or an composite SE + * @param contributingSimilarityEngines - only composite SE will have it (e.g., SANNN, UTG). Otherwise it is an empty Seq. All contributing SEs mst be atomic + */ +case class CandidateGenerationInfo( + sourceInfoOpt: Option[SourceInfo], + similarityEngineInfo: SimilarityEngineInfo, + contributingSimilarityEngines: Seq[SimilarityEngineInfo]) diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/model/CandidateGeneratorQuery.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/model/CandidateGeneratorQuery.scala new file mode 100644 index 0000000000..084cbb0422 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/model/CandidateGeneratorQuery.scala @@ -0,0 +1,96 @@ +package com.twitter.cr_mixer.model + +import com.twitter.core_workflows.user_model.thriftscala.UserState +import com.twitter.cr_mixer.thriftscala.Product +import com.twitter.product_mixer.core.thriftscala.ClientContext +import com.twitter.simclusters_v2.common.TweetId +import com.twitter.simclusters_v2.common.UserId +import com.twitter.simclusters_v2.thriftscala.InternalId +import com.twitter.simclusters_v2.thriftscala.TopicId +import com.twitter.timelines.configapi.Params + +sealed trait CandidateGeneratorQuery { + val product: Product + val maxNumResults: Int + val impressedTweetList: Set[TweetId] + val params: Params + val requestUUID: Long +} + +sealed trait HasUserId { + val userId: UserId +} + +case class CrCandidateGeneratorQuery( + userId: UserId, + product: Product, + userState: UserState, + maxNumResults: Int, + impressedTweetList: Set[TweetId], + params: Params, + requestUUID: Long, + languageCode: Option[String] = None) + extends CandidateGeneratorQuery + with HasUserId + +case class UtegTweetCandidateGeneratorQuery( + userId: UserId, + product: Product, + userState: UserState, + maxNumResults: Int, + impressedTweetList: Set[TweetId], + params: Params, + requestUUID: Long) + extends CandidateGeneratorQuery + with HasUserId + +case class RelatedTweetCandidateGeneratorQuery( + internalId: InternalId, + clientContext: ClientContext, // To scribe LogIn/LogOut requests + product: Product, + maxNumResults: Int, + impressedTweetList: Set[TweetId], + params: Params, + requestUUID: Long) + extends CandidateGeneratorQuery + +case class RelatedVideoTweetCandidateGeneratorQuery( + internalId: InternalId, + clientContext: ClientContext, // To scribe LogIn/LogOut requests + product: Product, + maxNumResults: Int, + impressedTweetList: Set[TweetId], + params: Params, + requestUUID: Long) + extends CandidateGeneratorQuery + +case class FrsTweetCandidateGeneratorQuery( + userId: UserId, + product: Product, + maxNumResults: Int, + impressedUserList: Set[UserId], + impressedTweetList: Set[TweetId], + params: Params, + languageCodeOpt: Option[String] = None, + countryCodeOpt: Option[String] = None, + requestUUID: Long) + extends CandidateGeneratorQuery + +case class AdsCandidateGeneratorQuery( + userId: UserId, + product: Product, + userState: UserState, + maxNumResults: Int, + params: Params, + requestUUID: Long) + +case class TopicTweetCandidateGeneratorQuery( + userId: UserId, + topicIds: Set[TopicId], + product: Product, + maxNumResults: Int, + impressedTweetList: Set[TweetId], + params: Params, + requestUUID: Long, + isVideoOnly: Boolean) + extends CandidateGeneratorQuery diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/model/EarlybirdSimilarityEngineType.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/model/EarlybirdSimilarityEngineType.scala new file mode 100644 index 0000000000..aa30403738 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/model/EarlybirdSimilarityEngineType.scala @@ -0,0 +1,6 @@ +package com.twitter.cr_mixer.model + +sealed trait EarlybirdSimilarityEngineType +object EarlybirdSimilarityEngineType_RecencyBased extends EarlybirdSimilarityEngineType +object EarlybirdSimilarityEngineType_ModelBased extends EarlybirdSimilarityEngineType +object EarlybirdSimilarityEngineType_TensorflowBased extends EarlybirdSimilarityEngineType diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/model/HealthThreshold.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/model/HealthThreshold.scala new file mode 100644 index 0000000000..0249798bda --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/model/HealthThreshold.scala @@ -0,0 +1,11 @@ +package com.twitter.cr_mixer.model + +object HealthThreshold { + object Enum extends Enumeration { + val Off: Value = Value(1) + val Moderate: Value = Value(2) + val Strict: Value = Value(3) + val Stricter: Value = Value(4) + val StricterPlus: Value = Value(5) + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/model/ModelConfig.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/model/ModelConfig.scala new file mode 100644 index 0000000000..26db7898b1 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/model/ModelConfig.scala @@ -0,0 +1,77 @@ +package com.twitter.cr_mixer.model + +/** + * A Configuration class for all Model Based Candidate Sources. + * + * The Model Name Guideline. Please your modelId as "Algorithm_Product_Date" + * If your model is used for multiple product surfaces, name it as all + * Don't name your algorithm as MBCG. All the algorithms here are MBCG =.= + * + * Don't forgot to add your new models into allHnswANNSimilarityEngineModelIds list. + */ +object ModelConfig { + // Offline SimClusters CG Experiment related Model Ids + val OfflineInterestedInFromKnownFor2020: String = "OfflineIIKF_ALL_20220414" + val OfflineInterestedInFromKnownFor2020Hl0El15: String = "OfflineIIKF_ALL_20220414_Hl0_El15" + val OfflineInterestedInFromKnownFor2020Hl2El15: String = "OfflineIIKF_ALL_20220414_Hl2_El15" + val OfflineInterestedInFromKnownFor2020Hl2El50: String = "OfflineIIKF_ALL_20220414_Hl2_El50" + val OfflineInterestedInFromKnownFor2020Hl8El50: String = "OfflineIIKF_ALL_20220414_Hl8_El50" + val OfflineMTSConsumerEmbeddingsFav90P20M: String = + "OfflineMTSConsumerEmbeddingsFav90P20M_ALL_20220414" + + // Twhin Model Ids + val ConsumerBasedTwHINRegularUpdateAll20221024: String = + "ConsumerBasedTwHINRegularUpdate_All_20221024" + + // Averaged Twhin Model Ids + val TweetBasedTwHINRegularUpdateAll20221024: String = + "TweetBasedTwHINRegularUpdate_All_20221024" + + // Collaborative Filtering Twhin Model Ids + val TwhinCollabFilterForFollow: String = + "TwhinCollabFilterForFollow" + val TwhinCollabFilterForEngagement: String = + "TwhinCollabFilterForEngagement" + val TwhinMultiClusterForFollow: String = + "TwhinMultiClusterForFollow" + val TwhinMultiClusterForEngagement: String = + "TwhinMultiClusterForEngagement" + + // Two Tower model Ids + val TwoTowerFavALL20220808: String = + "TwoTowerFav_ALL_20220808" + + // Debugger Demo-Only Model Ids + val DebuggerDemo: String = "DebuggerDemo" + + // ColdStartLookalike - this is not really a model name, it is as a placeholder to + // indicate ColdStartLookalike candidate source, which is currently being pluged into + // CustomizedRetrievalCandidateGeneration temporarily. + val ColdStartLookalikeModelName: String = "ConsumersBasedUtgColdStartLookalike20220707" + + // consumersBasedUTG-RealGraphOon Model Id + val ConsumersBasedUtgRealGraphOon20220705: String = "ConsumersBasedUtgRealGraphOon_All_20220705" + // consumersBasedUAG-RealGraphOon Model Id + val ConsumersBasedUagRealGraphOon20221205: String = "ConsumersBasedUagRealGraphOon_All_20221205" + + // FTR + val OfflineFavDecayedSum: String = "OfflineFavDecayedSum" + val OfflineFtrAt5Pop1000RnkDcy11: String = "OfflineFtrAt5Pop1000RnkDcy11" + val OfflineFtrAt5Pop10000RnkDcy11: String = "OfflineFtrAt5Pop10000RnkDcy11" + + // All Model Ids of HnswANNSimilarityEngines + val allHnswANNSimilarityEngineModelIds = Seq( + ConsumerBasedTwHINRegularUpdateAll20221024, + TwoTowerFavALL20220808, + DebuggerDemo + ) + + val ConsumerLogFavBasedInterestedInEmbedding: String = + "ConsumerLogFavBasedInterestedIn_ALL_20221228" + val ConsumerFollowBasedInterestedInEmbedding: String = + "ConsumerFollowBasedInterestedIn_ALL_20221228" + + val RetweetBasedDiffusion: String = + "RetweetBasedDiffusion" + +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/model/ModuleNames.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/model/ModuleNames.scala new file mode 100644 index 0000000000..6aec7b052a --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/model/ModuleNames.scala @@ -0,0 +1,122 @@ +package com.twitter.cr_mixer.model + +/** + * Define name annotated module names here + */ +object ModuleNames { + + final val FrsStore = "FrsStore" + final val UssStore = "UssStore" + final val UssStratoColumn = "UssStratoColumn" + final val RsxStore = "RsxStore" + final val RmsTweetLogFavLongestL2EmbeddingStore = "RmsTweetLogFavLongestL2EmbeddingStore" + final val RmsUserFavBasedProducerEmbeddingStore = "RmsUserFavBasedProducerEmbeddingStore" + final val RmsUserLogFavInterestedInEmbeddingStore = "RmsUserLogFavInterestedInEmbeddingStore" + final val RmsUserFollowInterestedInEmbeddingStore = "RmsUserFollowInterestedInEmbeddingStore" + final val StpStore = "StpStore" + final val TwiceClustersMembersStore = "TwiceClustersMembersStore" + final val TripCandidateStore = "TripCandidateStore" + + final val ConsumerEmbeddingBasedTripSimilarityEngine = + "ConsumerEmbeddingBasedTripSimilarityEngine" + final val ConsumerEmbeddingBasedTwHINANNSimilarityEngine = + "ConsumerEmbeddingBasedTwHINANNSimilarityEngine" + final val ConsumerEmbeddingBasedTwoTowerANNSimilarityEngine = + "ConsumerEmbeddingBasedTwoTowerANNSimilarityEngine" + final val ConsumersBasedUserAdGraphSimilarityEngine = + "ConsumersBasedUserAdGraphSimilarityEngine" + final val ConsumersBasedUserVideoGraphSimilarityEngine = + "ConsumersBasedUserVideoGraphSimilarityEngine" + + final val ConsumerBasedWalsSimilarityEngine = "ConsumerBasedWalsSimilarityEngine" + + final val TweetBasedTwHINANNSimilarityEngine = "TweetBasedTwHINANNSimilarityEngine" + + final val SimClustersANNSimilarityEngine = "SimClustersANNSimilarityEngine" + + final val ProdSimClustersANNServiceClientName = "ProdSimClustersANNServiceClient" + final val ExperimentalSimClustersANNServiceClientName = "ExperimentalSimClustersANNServiceClient" + final val SimClustersANNServiceClientName1 = "SimClustersANNServiceClient1" + final val SimClustersANNServiceClientName2 = "SimClustersANNServiceClient2" + final val SimClustersANNServiceClientName3 = "SimClustersANNServiceClient3" + final val SimClustersANNServiceClientName5 = "SimClustersANNServiceClient5" + final val SimClustersANNServiceClientName4 = "SimClustersANNServiceClient4" + final val UnifiedCache = "unifiedCache" + final val MLScoreCache = "mlScoreCache" + final val TweetRecommendationResultsCache = "tweetRecommendationResultsCache" + final val EarlybirdTweetsCache = "earlybirdTweetsCache" + final val EarlybirdRecencyBasedWithoutRetweetsRepliesTweetsCache = + "earlybirdTweetsWithoutRetweetsRepliesCacheStore" + final val EarlybirdRecencyBasedWithRetweetsRepliesTweetsCache = + "earlybirdTweetsWithRetweetsRepliesCacheStore" + + final val AbDeciderLogger = "abDeciderLogger" + final val TopLevelApiDdgMetricsLogger = "topLevelApiDdgMetricsLogger" + final val TweetRecsLogger = "tweetRecsLogger" + final val BlueVerifiedTweetRecsLogger = "blueVerifiedTweetRecsLogger" + final val RelatedTweetsLogger = "relatedTweetsLogger" + final val UtegTweetsLogger = "utegTweetsLogger" + final val AdsRecommendationsLogger = "adsRecommendationLogger" + + final val OfflineSimClustersANNInterestedInSimilarityEngine = + "OfflineSimClustersANNInterestedInSimilarityEngine" + + final val RealGraphOonStore = "RealGraphOonStore" + final val RealGraphInStore = "RealGraphInStore" + + final val OfflineTweet2020CandidateStore = "OfflineTweet2020CandidateStore" + final val OfflineTweet2020Hl0El15CandidateStore = "OfflineTweet2020Hl0El15CandidateStore" + final val OfflineTweet2020Hl2El15CandidateStore = "OfflineTweet2020Hl2El15CandidateStore" + final val OfflineTweet2020Hl2El50CandidateStore = "OfflineTweet2020Hl2El50CandidateStore" + final val OfflineTweet2020Hl8El50CandidateStore = "OfflineTweet2020Hl8El50CandidateStore" + final val OfflineTweetMTSCandidateStore = "OfflineTweetMTSCandidateStore" + + final val OfflineFavDecayedSumCandidateStore = "OfflineFavDecayedSumCandidateStore" + final val OfflineFtrAt5Pop1000RankDecay11CandidateStore = + "OfflineFtrAt5Pop1000RankDecay11CandidateStore" + final val OfflineFtrAt5Pop10000RankDecay11CandidateStore = + "OfflineFtrAt5Pop10000RankDecay11CandidateStore" + + final val TwhinCollabFilterStratoStoreForFollow = "TwhinCollabFilterStratoStoreForFollow" + final val TwhinCollabFilterStratoStoreForEngagement = "TwhinCollabFilterStratoStoreForEngagement" + final val TwhinMultiClusterStratoStoreForFollow = "TwhinMultiClusterStratoStoreForFollow" + final val TwhinMultiClusterStratoStoreForEngagement = "TwhinMultiClusterStratoStoreForEngagement" + + final val ProducerBasedUserAdGraphSimilarityEngine = + "ProducerBasedUserAdGraphSimilarityEngine" + final val ProducerBasedUserTweetGraphSimilarityEngine = + "ProducerBasedUserTweetGraphSimilarityEngine" + final val ProducerBasedUnifiedSimilarityEngine = "ProducerBasedUnifiedSimilarityEngine" + + final val TweetBasedUserAdGraphSimilarityEngine = "TweetBasedUserAdGraphSimilarityEngine" + final val TweetBasedUserTweetGraphSimilarityEngine = "TweetBasedUserTweetGraphSimilarityEngine" + final val TweetBasedUserVideoGraphSimilarityEngine = "TweetBasedUserVideoGraphSimilarityEngine" + final val TweetBasedQigSimilarityEngine = "TweetBasedQigSimilarityEngine" + final val TweetBasedUnifiedSimilarityEngine = "TweetBasedUnifiedSimilarityEngine" + + final val TwhinCollabFilterSimilarityEngine = "TwhinCollabFilterSimilarityEngine" + + final val ConsumerBasedUserTweetGraphStore = "ConsumerBasedUserTweetGraphStore" + final val ConsumerBasedUserVideoGraphStore = "ConsumerBasedUserVideoGraphStore" + final val ConsumerBasedUserAdGraphStore = "ConsumerBasedUserAdGraphStore" + + final val UserTweetEntityGraphSimilarityEngine = + "UserTweetEntityGraphSimilarityEngine" + + final val CertoTopicTweetSimilarityEngine = "CertoTopicTweetSimilarityEngine" + final val CertoStratoStoreName = "CertoStratoStore" + + final val SkitTopicTweetSimilarityEngine = "SkitTopicTweetSimilarityEngine" + final val SkitHighPrecisionTopicTweetSimilarityEngine = + "SkitHighPrecisionTopicTweetSimilarityEngine" + final val SkitStratoStoreName = "SkitStratoStore" + + final val HomeNaviGRPCClient = "HomeNaviGRPCClient" + final val AdsFavedNaviGRPCClient = "AdsFavedNaviGRPCClient" + final val AdsMonetizableNaviGRPCClient = "AdsMonetizableNaviGRPCClient" + + final val RetweetBasedDiffusionRecsMhStore = "RetweetBasedDiffusionRecsMhStore" + final val DiffusionBasedSimilarityEngine = "DiffusionBasedSimilarityEngine" + + final val BlueVerifiedAnnotationStore = "BlueVerifiedAnnotationStore" +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/model/TopicTweetWithScore.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/model/TopicTweetWithScore.scala new file mode 100644 index 0000000000..e9a0cf173c --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/model/TopicTweetWithScore.scala @@ -0,0 +1,13 @@ +package com.twitter.cr_mixer.model + +import com.twitter.cr_mixer.thriftscala.SimilarityEngineType +import com.twitter.simclusters_v2.common.TweetId + +/*** + * Bind a tweetId with a raw score generated from one single Similarity Engine + * @param similarityEngineType, which underlying topic source the topic tweet is from + */ +case class TopicTweetWithScore( + tweetId: TweetId, + score: Double, + similarityEngineType: SimilarityEngineType) diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/model/TweetWithAuthor.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/model/TweetWithAuthor.scala new file mode 100644 index 0000000000..16a506a4cb --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/model/TweetWithAuthor.scala @@ -0,0 +1,6 @@ +package com.twitter.cr_mixer.model + +import com.twitter.simclusters_v2.common.TweetId +import com.twitter.simclusters_v2.common.UserId + +case class TweetWithAuthor(tweetId: TweetId, authorId: UserId) diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/model/TweetWithScore.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/model/TweetWithScore.scala new file mode 100644 index 0000000000..ad88669120 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/model/TweetWithScore.scala @@ -0,0 +1,8 @@ +package com.twitter.cr_mixer.model + +import com.twitter.simclusters_v2.common.TweetId + +/*** + * Bind a tweetId with a raw score generated from one single Similarity Engine + */ +case class TweetWithScore(tweetId: TweetId, score: Double) diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/model/TweetWithScoreAndSocialProof.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/model/TweetWithScoreAndSocialProof.scala new file mode 100644 index 0000000000..94e430d8e5 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/model/TweetWithScoreAndSocialProof.scala @@ -0,0 +1,12 @@ +package com.twitter.cr_mixer.model + +import com.twitter.simclusters_v2.common.TweetId +import com.twitter.recos.recos_common.thriftscala.SocialProofType + +/*** + * Bind a tweetId with a raw score and social proofs by type + */ +case class TweetWithScoreAndSocialProof( + tweetId: TweetId, + score: Double, + socialProofByType: Map[SocialProofType, Seq[Long]]) diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/ActivePromotedTweetStoreModule.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/ActivePromotedTweetStoreModule.scala new file mode 100644 index 0000000000..d6529531a2 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/ActivePromotedTweetStoreModule.scala @@ -0,0 +1,135 @@ +package com.twitter.cr_mixer.module + +import com.google.inject.Provides +import com.google.inject.Singleton +import com.twitter.bijection.thrift.CompactThriftCodec +import com.twitter.ads.entities.db.thriftscala.LineItemObjective +import com.twitter.bijection.Injection +import com.twitter.conversions.DurationOps._ +import com.twitter.cr_mixer.model.ModuleNames +import com.twitter.cr_mixer.thriftscala.LineItemInfo +import com.twitter.finagle.memcached.{Client => MemcachedClient} +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.hermit.store.common.ObservedCachedReadableStore +import com.twitter.hermit.store.common.ObservedMemcachedReadableStore +import com.twitter.inject.TwitterModule +import com.twitter.ml.api.DataRecord +import com.twitter.ml.api.DataType +import com.twitter.ml.api.Feature +import com.twitter.ml.api.GeneralTensor +import com.twitter.ml.api.RichDataRecord +import com.twitter.relevance_platform.common.injection.LZ4Injection +import com.twitter.relevance_platform.common.injection.SeqObjectInjection +import com.twitter.simclusters_v2.common.TweetId +import com.twitter.storage.client.manhattan.kv.ManhattanKVClientMtlsParams +import com.twitter.storehaus.ReadableStore +import com.twitter.storehaus_internal.manhattan.ManhattanRO +import com.twitter.storehaus_internal.manhattan.ManhattanROConfig +import com.twitter.storehaus_internal.manhattan.Revenue +import com.twitter.storehaus_internal.util.ApplicationID +import com.twitter.storehaus_internal.util.DatasetName +import com.twitter.storehaus_internal.util.HDFSPath +import com.twitter.util.Future +import javax.inject.Named +import scala.collection.JavaConverters._ + +object ActivePromotedTweetStoreModule extends TwitterModule { + + case class ActivePromotedTweetStore( + activePromotedTweetMHStore: ReadableStore[String, DataRecord], + statsReceiver: StatsReceiver) + extends ReadableStore[TweetId, Seq[LineItemInfo]] { + override def get(tweetId: TweetId): Future[Option[Seq[LineItemInfo]]] = { + activePromotedTweetMHStore.get(tweetId.toString).map { + _.map { dataRecord => + val richDataRecord = new RichDataRecord(dataRecord) + val lineItemIdsFeature: Feature[GeneralTensor] = + new Feature.Tensor("active_promoted_tweets.line_item_ids", DataType.INT64) + + val lineItemObjectivesFeature: Feature[GeneralTensor] = + new Feature.Tensor("active_promoted_tweets.line_item_objectives", DataType.INT64) + + val lineItemIdsTensor: GeneralTensor = richDataRecord.getFeatureValue(lineItemIdsFeature) + val lineItemObjectivesTensor: GeneralTensor = + richDataRecord.getFeatureValue(lineItemObjectivesFeature) + + val lineItemIds: Seq[Long] = + if (lineItemIdsTensor.getSetField == GeneralTensor._Fields.INT64_TENSOR && lineItemIdsTensor.getInt64Tensor.isSetLongs) { + lineItemIdsTensor.getInt64Tensor.getLongs.asScala.map(_.toLong) + } else Seq.empty + + val lineItemObjectives: Seq[LineItemObjective] = + if (lineItemObjectivesTensor.getSetField == GeneralTensor._Fields.INT64_TENSOR && lineItemObjectivesTensor.getInt64Tensor.isSetLongs) { + lineItemObjectivesTensor.getInt64Tensor.getLongs.asScala.map(objective => + LineItemObjective(objective.toInt)) + } else Seq.empty + + val lineItemInfo = + if (lineItemIds.size == lineItemObjectives.size) { + lineItemIds.zipWithIndex.map { + case (lineItemId, index) => + LineItemInfo( + lineItemId = lineItemId, + lineItemObjective = lineItemObjectives(index) + ) + } + } else Seq.empty + + lineItemInfo + } + } + } + } + + @Provides + @Singleton + def providesActivePromotedTweetStore( + manhattanKVClientMtlsParams: ManhattanKVClientMtlsParams, + @Named(ModuleNames.UnifiedCache) crMixerUnifiedCacheClient: MemcachedClient, + crMixerStatsReceiver: StatsReceiver + ): ReadableStore[TweetId, Seq[LineItemInfo]] = { + + val mhConfig = new ManhattanROConfig { + val hdfsPath = HDFSPath("") + val applicationID = ApplicationID("ads_bigquery_features") + val datasetName = DatasetName("active_promoted_tweets") + val cluster = Revenue + + override def statsReceiver: StatsReceiver = + crMixerStatsReceiver.scope("active_promoted_tweets_mh") + } + val mhStore: ReadableStore[String, DataRecord] = + ManhattanRO + .getReadableStoreWithMtls[String, DataRecord]( + mhConfig, + manhattanKVClientMtlsParams + )( + implicitly[Injection[String, Array[Byte]]], + CompactThriftCodec[DataRecord] + ) + + val underlyingStore = + ActivePromotedTweetStore(mhStore, crMixerStatsReceiver.scope("ActivePromotedTweetStore")) + val memcachedStore = ObservedMemcachedReadableStore.fromCacheClient( + backingStore = underlyingStore, + cacheClient = crMixerUnifiedCacheClient, + ttl = 60.minutes, + asyncUpdate = false + )( + valueInjection = LZ4Injection.compose(SeqObjectInjection[LineItemInfo]()), + statsReceiver = crMixerStatsReceiver.scope("memCachedActivePromotedTweetStore"), + keyToString = { k: TweetId => s"apt/$k" } + ) + + ObservedCachedReadableStore.from( + memcachedStore, + ttl = 30.minutes, + maxKeys = 250000, // size of promoted tweet is around 200,000 + windowSize = 10000L, + cacheName = "active_promoted_tweet_cache", + maxMultiGetSize = 20 + )(crMixerStatsReceiver.scope("inMemoryCachedActivePromotedTweetStore")) + + } + +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/BUILD.bazel b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/BUILD.bazel new file mode 100644 index 0000000000..6773b526c3 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/BUILD.bazel @@ -0,0 +1,130 @@ +scala_library( + sources = [ + "*.scala", + "core/*.scala", + "grpc_client/*.scala", + "similarity_engine/*.scala", + "source_signal/*.scala", + "thrift_client/*.scala", + ], + compiler_option_sets = ["fatal_warnings"], + strict_deps = True, + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/com/twitter/bijection:core", + "3rdparty/jvm/com/twitter/bijection:scrooge", + "3rdparty/jvm/com/twitter/storehaus:core", + "3rdparty/jvm/com/twitter/storehaus:memcache", + "3rdparty/jvm/io/grpc:grpc-api", + "3rdparty/jvm/io/grpc:grpc-auth", + "3rdparty/jvm/io/grpc:grpc-core", + "3rdparty/jvm/io/grpc:grpc-netty", + "3rdparty/jvm/io/grpc:grpc-protobuf", + "3rdparty/jvm/io/grpc:grpc-stub", + "3rdparty/jvm/javax/inject:javax.inject", + "3rdparty/jvm/org/scalanlp:breeze", + "3rdparty/src/jvm/com/twitter/storehaus:core", + "abdecider/src/main/scala", + "ann/src/main/thrift/com/twitter/ann/common:ann-common-scala", + "configapi/configapi-abdecider", + "configapi/configapi-core", + "configapi/configapi-featureswitches:v2", + "content-recommender/server/src/main/scala/com/twitter/contentrecommender:cr-mixer-deps", + "content-recommender/thrift/src/main/thrift:thrift-scala", + "cr-mixer/server/src/main/scala/com/twitter/cr_mixer/candidate_generation", + "cr-mixer/server/src/main/scala/com/twitter/cr_mixer/config", + "cr-mixer/server/src/main/scala/com/twitter/cr_mixer/featureswitch", + "cr-mixer/server/src/main/scala/com/twitter/cr_mixer/model", + "cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param", + "cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/decider", + "cr-mixer/server/src/main/scala/com/twitter/cr_mixer/ranker", + "cr-mixer/server/src/main/scala/com/twitter/cr_mixer/scribe", + "cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine", + "cr-mixer/server/src/main/scala/com/twitter/cr_mixer/source_signal", + "cr-mixer/server/src/main/scala/com/twitter/cr_mixer/util", + "cr-mixer/thrift/src/main/thrift:thrift-scala", + "decider/src/main/scala", + "discovery-common/src/main/scala/com/twitter/discovery/common/configapi", + "featureswitches/featureswitches-core", + "featureswitches/featureswitches-core/src/main/scala/com/twitter/featureswitches/v2/builder", + "finagle-internal/finagle-grpc/src/main/scala", + "finagle-internal/mtls/src/main/scala/com/twitter/finagle/mtls/authentication", + "finatra-internal/kafka/src/main/scala/com/twitter/finatra/kafka/consumers", + "finatra-internal/mtls-thriftmux/src/main/scala", + "finatra/inject/inject-core/src/main/scala", + "finatra/inject/inject-modules/src/main/scala", + "finatra/inject/inject-thrift-client", + "follow-recommendations-service/thrift/src/main/thrift:thrift-scala", + "frigate/frigate-common:util", + "frigate/frigate-common/src/main/scala/com/twitter/frigate/common/base", + "frigate/frigate-common/src/main/scala/com/twitter/frigate/common/candidate", + "frigate/frigate-common/src/main/scala/com/twitter/frigate/common/store", + "frigate/frigate-common/src/main/scala/com/twitter/frigate/common/store/health", + "frigate/frigate-common/src/main/scala/com/twitter/frigate/common/store/interests", + "frigate/frigate-common/src/main/scala/com/twitter/frigate/common/store/strato", + "hermit/hermit-core/src/main/scala/com/twitter/hermit/store/common", + "hydra/partition/thrift/src/main/thrift:thrift-scala", + "hydra/root/thrift/src/main/thrift:thrift-scala", + "mediaservices/commons/src/main/scala:futuretracker", + "product-mixer/core/src/main/thrift/com/twitter/product_mixer/core:thrift-scala", + "qig-ranker/thrift/src/main/thrift:thrift-scala", + "relevance-platform/src/main/scala/com/twitter/relevance_platform/common/health_store", + "relevance-platform/src/main/scala/com/twitter/relevance_platform/common/injection", + "relevance-platform/thrift/src/main/thrift:thrift-scala", + "representation-manager/client/src/main/scala/com/twitter/representation_manager", + "representation-manager/client/src/main/scala/com/twitter/representation_manager/config", + "representation-manager/server/src/main/scala/com/twitter/representation_manager/migration", + "representation-manager/server/src/main/thrift:thrift-scala", + "representation-scorer/server/src/main/thrift:thrift-scala", + "servo/decider", + "servo/util/src/main/scala", + "simclusters-ann/thrift/src/main/thrift:thrift-scala", + "snowflake/src/main/scala/com/twitter/snowflake/id", + "src/java/com/twitter/ml/api:api-base", + "src/java/com/twitter/search/queryparser/query:core-query-nodes", + "src/java/com/twitter/search/queryparser/query/search:search-query-nodes", + "src/scala/com/twitter/algebird_internal/injection", + "src/scala/com/twitter/cortex/ml/embeddings/common:Helpers", + "src/scala/com/twitter/ml/api/embedding", + "src/scala/com/twitter/ml/featurestore/lib", + "src/scala/com/twitter/scalding_internal/multiformat/format", + "src/scala/com/twitter/simclusters_v2/candidate_source", + "src/scala/com/twitter/simclusters_v2/common", + "src/scala/com/twitter/storehaus_internal/manhattan", + "src/scala/com/twitter/storehaus_internal/manhattan/config", + "src/scala/com/twitter/storehaus_internal/memcache", + "src/scala/com/twitter/storehaus_internal/memcache/config", + "src/scala/com/twitter/storehaus_internal/offline", + "src/scala/com/twitter/storehaus_internal/util", + "src/scala/com/twitter/topic_recos/stores", + "src/thrift/com/twitter/core_workflows/user_model:user_model-scala", + "src/thrift/com/twitter/frigate:frigate-common-thrift-scala", + "src/thrift/com/twitter/frigate:frigate-thrift-scala", + "src/thrift/com/twitter/frigate/data_pipeline/scalding:blue_verified_annotations-scala", + "src/thrift/com/twitter/hermit/stp:hermit-stp-scala", + "src/thrift/com/twitter/ml/api:data-java", + "src/thrift/com/twitter/ml/api:embedding-scala", + "src/thrift/com/twitter/ml/featurestore:ml-feature-store-embedding-scala", + "src/thrift/com/twitter/onboarding/relevance/coldstart_lookalike:coldstartlookalike-thrift-scala", + "src/thrift/com/twitter/recos:recos-common-scala", + "src/thrift/com/twitter/recos/user_ad_graph:user_ad_graph-scala", + "src/thrift/com/twitter/recos/user_tweet_entity_graph:user_tweet_entity_graph-scala", + "src/thrift/com/twitter/recos/user_tweet_graph:user_tweet_graph-scala", + "src/thrift/com/twitter/recos/user_tweet_graph_plus:user_tweet_graph_plus-scala", + "src/thrift/com/twitter/recos/user_video_graph:user_video_graph-scala", + "src/thrift/com/twitter/search:earlybird-scala", + "src/thrift/com/twitter/search/query_interaction_graph/service:qig-service-scala", + "src/thrift/com/twitter/simclusters_v2:simclusters_v2-thrift-scala", + "src/thrift/com/twitter/topic_recos:topic_recos-thrift-scala", + "src/thrift/com/twitter/trends/trip_v1:trip-tweets-thrift-scala", + "src/thrift/com/twitter/tweetypie:service-scala", + "src/thrift/com/twitter/twistly:twistly-scala", + "src/thrift/com/twitter/wtf/candidate:wtf-candidate-scala", + "stitch/stitch-storehaus", + "stitch/stitch-tweetypie/src/main/scala", + "strato/src/main/scala/com/twitter/strato/client", + "user-signal-service/thrift/src/main/thrift:thrift-scala", + "util-internal/scribe/src/main/scala/com/twitter/logging", + "util/util-hashing", + ], +) diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/BlueVerifiedAnnotationStoreModule.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/BlueVerifiedAnnotationStoreModule.scala new file mode 100644 index 0000000000..21769d3fa0 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/BlueVerifiedAnnotationStoreModule.scala @@ -0,0 +1,52 @@ +package com.twitter.cr_mixer.module + +import com.google.inject.Provides +import com.google.inject.Singleton +import com.google.inject.name.Named +import com.twitter.inject.TwitterModule +import com.twitter.conversions.DurationOps._ +import com.twitter.cr_mixer.model.ModuleNames +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.frigate.data_pipeline.scalding.thriftscala.BlueVerifiedAnnotationsV2 +import com.twitter.storage.client.manhattan.kv.ManhattanKVClientMtlsParams +import com.twitter.storehaus.ReadableStore +import com.twitter.storehaus_internal.manhattan.Athena +import com.twitter.storehaus_internal.manhattan.ManhattanRO +import com.twitter.storehaus_internal.manhattan.ManhattanROConfig +import com.twitter.storehaus_internal.util.ApplicationID +import com.twitter.storehaus_internal.util.DatasetName +import com.twitter.storehaus_internal.util.HDFSPath +import com.twitter.bijection.scrooge.BinaryScalaCodec +import com.twitter.hermit.store.common.ObservedCachedReadableStore + +object BlueVerifiedAnnotationStoreModule extends TwitterModule { + + @Provides + @Singleton + @Named(ModuleNames.BlueVerifiedAnnotationStore) + def providesBlueVerifiedAnnotationStore( + statsReceiver: StatsReceiver, + manhattanKVClientMtlsParams: ManhattanKVClientMtlsParams, + ): ReadableStore[String, BlueVerifiedAnnotationsV2] = { + + implicit val valueCodec = new BinaryScalaCodec(BlueVerifiedAnnotationsV2) + + val underlyingStore = ManhattanRO + .getReadableStoreWithMtls[String, BlueVerifiedAnnotationsV2]( + ManhattanROConfig( + HDFSPath(""), + ApplicationID("content_recommender_athena"), + DatasetName("blue_verified_annotations"), + Athena), + manhattanKVClientMtlsParams + ) + + ObservedCachedReadableStore.from( + underlyingStore, + ttl = 24.hours, + maxKeys = 100000, + windowSize = 10000L, + cacheName = "blue_verified_annotation_cache" + )(statsReceiver.scope("inMemoryCachedBlueVerifiedAnnotationStore")) + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/CertoStratoStoreModule.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/CertoStratoStoreModule.scala new file mode 100644 index 0000000000..9908aa7028 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/CertoStratoStoreModule.scala @@ -0,0 +1,57 @@ +package com.twitter.cr_mixer.module + +import com.google.inject.Provides +import com.google.inject.Singleton +import com.google.inject.name.Named +import com.twitter.conversions.DurationOps._ +import com.twitter.cr_mixer.model.ModuleNames +import com.twitter.cr_mixer.similarity_engine.SimilarityEngine.keyHasher +import com.twitter.finagle.memcached.{Client => MemcachedClient} +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.hermit.store.common.ObservedCachedReadableStore +import com.twitter.hermit.store.common.ObservedMemcachedReadableStore +import com.twitter.hermit.store.common.ObservedReadableStore +import com.twitter.inject.TwitterModule +import com.twitter.relevance_platform.common.injection.LZ4Injection +import com.twitter.relevance_platform.common.injection.SeqObjectInjection +import com.twitter.simclusters_v2.thriftscala.TopicId +import com.twitter.storehaus.ReadableStore +import com.twitter.strato.client.Client +import com.twitter.topic_recos.stores.CertoTopicTopKTweetsStore +import com.twitter.topic_recos.thriftscala.TweetWithScores + +object CertoStratoStoreModule extends TwitterModule { + + @Provides + @Singleton + @Named(ModuleNames.CertoStratoStoreName) + def providesCertoStratoStore( + @Named(ModuleNames.UnifiedCache) crMixerUnifiedCacheClient: MemcachedClient, + stratoClient: Client, + statsReceiver: StatsReceiver + ): ReadableStore[TopicId, Seq[TweetWithScores]] = { + val certoStore = ObservedReadableStore(CertoTopicTopKTweetsStore.prodStore(stratoClient))( + statsReceiver.scope(ModuleNames.CertoStratoStoreName)).mapValues { topKTweetsWithScores => + topKTweetsWithScores.topTweetsByFollowerL2NormalizedCosineSimilarityScore + } + + val memCachedStore = ObservedMemcachedReadableStore + .fromCacheClient( + backingStore = certoStore, + cacheClient = crMixerUnifiedCacheClient, + ttl = 10.minutes + )( + valueInjection = LZ4Injection.compose(SeqObjectInjection[TweetWithScores]()), + statsReceiver = statsReceiver.scope("memcached_certo_store"), + keyToString = { k => s"certo:${keyHasher.hashKey(k.toString.getBytes)}" } + ) + + ObservedCachedReadableStore.from[TopicId, Seq[TweetWithScores]]( + memCachedStore, + ttl = 5.minutes, + maxKeys = 100000, // ~150MB max + cacheName = "certo_in_memory_cache", + windowSize = 10000L + )(statsReceiver.scope("certo_in_memory_cache")) + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/ConsumersBasedUserAdGraphStoreModule.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/ConsumersBasedUserAdGraphStoreModule.scala new file mode 100644 index 0000000000..33a0d33fc6 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/ConsumersBasedUserAdGraphStoreModule.scala @@ -0,0 +1,30 @@ +package com.twitter.cr_mixer.module + +import com.google.inject.Provides +import com.twitter.cr_mixer.model.ModuleNames +import com.twitter.inject.TwitterModule +import com.twitter.recos.user_ad_graph.thriftscala.ConsumersBasedRelatedAdRequest +import com.twitter.recos.user_ad_graph.thriftscala.RelatedAdResponse +import com.twitter.recos.user_ad_graph.thriftscala.UserAdGraph +import com.twitter.storehaus.ReadableStore +import com.twitter.util.Future +import javax.inject.Named +import javax.inject.Singleton + +object ConsumersBasedUserAdGraphStoreModule extends TwitterModule { + + @Provides + @Singleton + @Named(ModuleNames.ConsumerBasedUserAdGraphStore) + def providesConsumerBasedUserAdGraphStore( + userAdGraphService: UserAdGraph.MethodPerEndpoint + ): ReadableStore[ConsumersBasedRelatedAdRequest, RelatedAdResponse] = { + new ReadableStore[ConsumersBasedRelatedAdRequest, RelatedAdResponse] { + override def get( + k: ConsumersBasedRelatedAdRequest + ): Future[Option[RelatedAdResponse]] = { + userAdGraphService.consumersBasedRelatedAds(k).map(Some(_)) + } + } + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/ConsumersBasedUserTweetGraphStoreModule.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/ConsumersBasedUserTweetGraphStoreModule.scala new file mode 100644 index 0000000000..ab027744af --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/ConsumersBasedUserTweetGraphStoreModule.scala @@ -0,0 +1,30 @@ +package com.twitter.cr_mixer.module + +import com.google.inject.Provides +import com.twitter.cr_mixer.model.ModuleNames +import com.twitter.inject.TwitterModule +import com.twitter.recos.user_tweet_graph.thriftscala.ConsumersBasedRelatedTweetRequest +import com.twitter.recos.user_tweet_graph.thriftscala.RelatedTweetResponse +import com.twitter.recos.user_tweet_graph.thriftscala.UserTweetGraph +import com.twitter.storehaus.ReadableStore +import com.twitter.util.Future +import javax.inject.Named +import javax.inject.Singleton + +object ConsumersBasedUserTweetGraphStoreModule extends TwitterModule { + + @Provides + @Singleton + @Named(ModuleNames.ConsumerBasedUserTweetGraphStore) + def providesConsumerBasedUserTweetGraphStore( + userTweetGraphService: UserTweetGraph.MethodPerEndpoint + ): ReadableStore[ConsumersBasedRelatedTweetRequest, RelatedTweetResponse] = { + new ReadableStore[ConsumersBasedRelatedTweetRequest, RelatedTweetResponse] { + override def get( + k: ConsumersBasedRelatedTweetRequest + ): Future[Option[RelatedTweetResponse]] = { + userTweetGraphService.consumersBasedRelatedTweets(k).map(Some(_)) + } + } + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/ConsumersBasedUserVideoGraphStoreModule.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/ConsumersBasedUserVideoGraphStoreModule.scala new file mode 100644 index 0000000000..05cf496d89 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/ConsumersBasedUserVideoGraphStoreModule.scala @@ -0,0 +1,30 @@ +package com.twitter.cr_mixer.module + +import com.google.inject.Provides +import com.twitter.cr_mixer.model.ModuleNames +import com.twitter.inject.TwitterModule +import com.twitter.recos.user_video_graph.thriftscala.ConsumersBasedRelatedTweetRequest +import com.twitter.recos.user_video_graph.thriftscala.RelatedTweetResponse +import com.twitter.recos.user_video_graph.thriftscala.UserVideoGraph +import com.twitter.storehaus.ReadableStore +import com.twitter.util.Future +import javax.inject.Named +import javax.inject.Singleton + +object ConsumersBasedUserVideoGraphStoreModule extends TwitterModule { + + @Provides + @Singleton + @Named(ModuleNames.ConsumerBasedUserVideoGraphStore) + def providesConsumerBasedUserVideoGraphStore( + userVideoGraphService: UserVideoGraph.MethodPerEndpoint + ): ReadableStore[ConsumersBasedRelatedTweetRequest, RelatedTweetResponse] = { + new ReadableStore[ConsumersBasedRelatedTweetRequest, RelatedTweetResponse] { + override def get( + k: ConsumersBasedRelatedTweetRequest + ): Future[Option[RelatedTweetResponse]] = { + userVideoGraphService.consumersBasedRelatedTweets(k).map(Some(_)) + } + } + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/CrMixerParamConfigModule.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/CrMixerParamConfigModule.scala new file mode 100644 index 0000000000..baece79471 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/CrMixerParamConfigModule.scala @@ -0,0 +1,16 @@ +package com.twitter.cr_mixer.module + +import com.google.inject.Provides +import com.twitter.timelines.configapi.Config +import com.twitter.cr_mixer.param.CrMixerParamConfig +import com.twitter.inject.TwitterModule +import javax.inject.Singleton + +object CrMixerParamConfigModule extends TwitterModule { + + @Provides + @Singleton + def provideConfig(): Config = { + CrMixerParamConfig.config + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/DiffusionStoreModule.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/DiffusionStoreModule.scala new file mode 100644 index 0000000000..dc95f07f5f --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/DiffusionStoreModule.scala @@ -0,0 +1,54 @@ +package com.twitter.cr_mixer.module + +import com.google.inject.Provides +import com.twitter.bijection.Injection +import com.twitter.bijection.scrooge.BinaryScalaCodec +import com.twitter.cr_mixer.model.ModuleNames +import com.twitter.finagle.mtls.authentication.ServiceIdentifier +import com.twitter.inject.TwitterModule +import com.twitter.simclusters_v2.thriftscala.TweetsWithScore +import com.twitter.storage.client.manhattan.kv.ManhattanKVClientMtlsParams +import com.twitter.storehaus.ReadableStore +import com.twitter.storehaus_internal.manhattan.Apollo +import com.twitter.storehaus_internal.manhattan.ManhattanRO +import com.twitter.storehaus_internal.manhattan.ManhattanROConfig +import com.twitter.storehaus_internal.util.ApplicationID +import com.twitter.storehaus_internal.util.DatasetName +import com.twitter.storehaus_internal.util.HDFSPath +import javax.inject.Named +import javax.inject.Singleton + +object DiffusionStoreModule extends TwitterModule { + type UserId = Long + implicit val longCodec = implicitly[Injection[Long, Array[Byte]]] + implicit val tweetRecsInjection: Injection[TweetsWithScore, Array[Byte]] = + BinaryScalaCodec(TweetsWithScore) + + @Provides + @Singleton + @Named(ModuleNames.RetweetBasedDiffusionRecsMhStore) + def retweetBasedDiffusionRecsMhStore( + serviceIdentifier: ServiceIdentifier + ): ReadableStore[Long, TweetsWithScore] = { + val manhattanROConfig = ManhattanROConfig( + HDFSPath(""), // not needed + ApplicationID("cr_mixer_apollo"), + DatasetName("diffusion_retweet_tweet_recs"), + Apollo + ) + + buildTweetRecsStore(serviceIdentifier, manhattanROConfig) + } + + private def buildTweetRecsStore( + serviceIdentifier: ServiceIdentifier, + manhattanROConfig: ManhattanROConfig + ): ReadableStore[Long, TweetsWithScore] = { + + ManhattanRO + .getReadableStoreWithMtls[Long, TweetsWithScore]( + manhattanROConfig, + ManhattanKVClientMtlsParams(serviceIdentifier) + )(longCodec, tweetRecsInjection) + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/EarlybirdRecencyBasedCandidateStoreModule.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/EarlybirdRecencyBasedCandidateStoreModule.scala new file mode 100644 index 0000000000..c0fe025f0e --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/EarlybirdRecencyBasedCandidateStoreModule.scala @@ -0,0 +1,189 @@ +package com.twitter.cr_mixer.module + +import com.google.inject.Provides +import com.google.inject.Singleton +import com.twitter.cr_mixer.config.TimeoutConfig +import com.twitter.cr_mixer.model.ModuleNames +import com.twitter.cr_mixer.util.EarlybirdSearchUtil.EarlybirdClientId +import com.twitter.cr_mixer.util.EarlybirdSearchUtil.FacetsToFetch +import com.twitter.cr_mixer.util.EarlybirdSearchUtil.GetCollectorTerminationParams +import com.twitter.cr_mixer.util.EarlybirdSearchUtil.GetEarlybirdQuery +import com.twitter.cr_mixer.util.EarlybirdSearchUtil.MetadataOptions +import com.twitter.finagle.memcached.{Client => MemcachedClient} +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.frigate.common.util.SeqLongInjection +import com.twitter.hashing.KeyHasher +import com.twitter.hermit.store.common.ObservedMemcachedReadableStore +import com.twitter.inject.TwitterModule +import com.twitter.search.common.query.thriftjava.thriftscala.CollectorParams +import com.twitter.search.earlybird.thriftscala.EarlybirdRequest +import com.twitter.search.earlybird.thriftscala.EarlybirdResponseCode +import com.twitter.search.earlybird.thriftscala.EarlybirdService +import com.twitter.search.earlybird.thriftscala.ThriftSearchQuery +import com.twitter.search.earlybird.thriftscala.ThriftSearchRankingMode +import com.twitter.simclusters_v2.common.TweetId +import com.twitter.simclusters_v2.common.UserId +import com.twitter.storehaus.ReadableStore +import com.twitter.util.Duration +import com.twitter.util.Future +import javax.inject.Named + +object EarlybirdRecencyBasedCandidateStoreModule extends TwitterModule { + + @Provides + @Singleton + @Named(ModuleNames.EarlybirdRecencyBasedWithoutRetweetsRepliesTweetsCache) + def providesEarlybirdRecencyBasedWithoutRetweetsRepliesCandidateStore( + statsReceiver: StatsReceiver, + earlybirdSearchClient: EarlybirdService.MethodPerEndpoint, + @Named(ModuleNames.EarlybirdTweetsCache) earlybirdRecencyBasedTweetsCache: MemcachedClient, + timeoutConfig: TimeoutConfig + ): ReadableStore[UserId, Seq[TweetId]] = { + val stats = statsReceiver.scope("EarlybirdRecencyBasedWithoutRetweetsRepliesCandidateStore") + val underlyingStore = new ReadableStore[UserId, Seq[TweetId]] { + override def get(userId: UserId): Future[Option[Seq[TweetId]]] = { + // Home based EB filters out retweets and replies + val earlybirdRequest = + buildEarlybirdRequest( + userId, + FilterOutRetweetsAndReplies, + DefaultMaxNumTweetPerUser, + timeoutConfig.earlybirdServerTimeout) + getEarlybirdSearchResult(earlybirdSearchClient, earlybirdRequest, stats) + } + } + ObservedMemcachedReadableStore.fromCacheClient( + backingStore = underlyingStore, + cacheClient = earlybirdRecencyBasedTweetsCache, + ttl = MemcacheKeyTimeToLiveDuration, + asyncUpdate = true + )( + valueInjection = SeqLongInjection, + statsReceiver = statsReceiver.scope("earlybird_recency_based_tweets_home_memcache"), + keyToString = { k => + f"uEBRBHM:${keyHasher.hashKey(k.toString.getBytes)}%X" // prefix = EarlyBirdRecencyBasedHoMe + } + ) + } + + @Provides + @Singleton + @Named(ModuleNames.EarlybirdRecencyBasedWithRetweetsRepliesTweetsCache) + def providesEarlybirdRecencyBasedWithRetweetsRepliesCandidateStore( + statsReceiver: StatsReceiver, + earlybirdSearchClient: EarlybirdService.MethodPerEndpoint, + @Named(ModuleNames.EarlybirdTweetsCache) earlybirdRecencyBasedTweetsCache: MemcachedClient, + timeoutConfig: TimeoutConfig + ): ReadableStore[UserId, Seq[TweetId]] = { + val stats = statsReceiver.scope("EarlybirdRecencyBasedWithRetweetsRepliesCandidateStore") + val underlyingStore = new ReadableStore[UserId, Seq[TweetId]] { + override def get(userId: UserId): Future[Option[Seq[TweetId]]] = { + val earlybirdRequest = buildEarlybirdRequest( + userId, + // Notifications based EB keeps retweets and replies + NotFilterOutRetweetsAndReplies, + DefaultMaxNumTweetPerUser, + processingTimeout = timeoutConfig.earlybirdServerTimeout + ) + getEarlybirdSearchResult(earlybirdSearchClient, earlybirdRequest, stats) + } + } + ObservedMemcachedReadableStore.fromCacheClient( + backingStore = underlyingStore, + cacheClient = earlybirdRecencyBasedTweetsCache, + ttl = MemcacheKeyTimeToLiveDuration, + asyncUpdate = true + )( + valueInjection = SeqLongInjection, + statsReceiver = statsReceiver.scope("earlybird_recency_based_tweets_notifications_memcache"), + keyToString = { k => + f"uEBRBN:${keyHasher.hashKey(k.toString.getBytes)}%X" // prefix = EarlyBirdRecencyBasedNotifications + } + ) + } + + private val keyHasher: KeyHasher = KeyHasher.FNV1A_64 + + /** + * Note the DefaultMaxNumTweetPerUser is used to adjust the result size per cache entry. + * If the value changes, it will increase the size of the memcache. + */ + private val DefaultMaxNumTweetPerUser: Int = 100 + private val FilterOutRetweetsAndReplies = true + private val NotFilterOutRetweetsAndReplies = false + private val MemcacheKeyTimeToLiveDuration: Duration = Duration.fromMinutes(15) + + private def buildEarlybirdRequest( + seedUserId: UserId, + filterOutRetweetsAndReplies: Boolean, + maxNumTweetsPerSeedUser: Int, + processingTimeout: Duration + ): EarlybirdRequest = + EarlybirdRequest( + searchQuery = getThriftSearchQuery( + seedUserId = seedUserId, + filterOutRetweetsAndReplies = filterOutRetweetsAndReplies, + maxNumTweetsPerSeedUser = maxNumTweetsPerSeedUser, + processingTimeout = processingTimeout + ), + clientId = Some(EarlybirdClientId), + timeoutMs = processingTimeout.inMilliseconds.intValue(), + getOlderResults = Some(false), + adjustedProtectedRequestParams = None, + adjustedFullArchiveRequestParams = None, + getProtectedTweetsOnly = Some(false), + skipVeryRecentTweets = true, + ) + + private def getThriftSearchQuery( + seedUserId: UserId, + filterOutRetweetsAndReplies: Boolean, + maxNumTweetsPerSeedUser: Int, + processingTimeout: Duration + ): ThriftSearchQuery = ThriftSearchQuery( + serializedQuery = GetEarlybirdQuery( + None, + None, + Set.empty, + filterOutRetweetsAndReplies + ).map(_.serialize), + fromUserIDFilter64 = Some(Seq(seedUserId)), + numResults = maxNumTweetsPerSeedUser, + rankingMode = ThriftSearchRankingMode.Recency, + collectorParams = Some( + CollectorParams( + // numResultsToReturn defines how many results each EB shard will return to search root + numResultsToReturn = maxNumTweetsPerSeedUser, + // terminationParams.maxHitsToProcess is used for early terminating per shard results fetching. + terminationParams = + GetCollectorTerminationParams(maxNumTweetsPerSeedUser, processingTimeout) + )), + facetFieldNames = Some(FacetsToFetch), + resultMetadataOptions = Some(MetadataOptions), + searchStatusIds = None + ) + + private def getEarlybirdSearchResult( + earlybirdSearchClient: EarlybirdService.MethodPerEndpoint, + request: EarlybirdRequest, + statsReceiver: StatsReceiver + ): Future[Option[Seq[TweetId]]] = earlybirdSearchClient + .search(request) + .map { response => + response.responseCode match { + case EarlybirdResponseCode.Success => + val earlybirdSearchResult = + response.searchResults + .map { + _.results + .map(searchResult => searchResult.id) + } + statsReceiver.scope("result").stat("size").add(earlybirdSearchResult.size) + earlybirdSearchResult + case e => + statsReceiver.scope("failures").counter(e.getClass.getSimpleName).incr() + Some(Seq.empty) + } + } + +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/EmbeddingStoreModule.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/EmbeddingStoreModule.scala new file mode 100644 index 0000000000..26d9f8ad11 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/EmbeddingStoreModule.scala @@ -0,0 +1,195 @@ +package com.twitter.cr_mixer.module + +import com.google.inject.Provides +import com.twitter.bijection.Injection +import com.twitter.bijection.scrooge.BinaryScalaCodec +import com.twitter.bijection.scrooge.CompactScalaCodec +import com.twitter.finagle.mtls.authentication.ServiceIdentifier +import com.twitter.inject.TwitterModule +import com.twitter.ml.api.{thriftscala => api} +import com.twitter.simclusters_v2.thriftscala.CandidateTweetsList +import com.twitter.simclusters_v2.common.TweetId +import com.twitter.simclusters_v2.thriftscala.InternalId +import com.twitter.storage.client.manhattan.kv.ManhattanKVClientMtlsParams +import com.twitter.storehaus.ReadableStore +import com.twitter.storehaus_internal.manhattan.Apollo +import com.twitter.storehaus_internal.manhattan.ManhattanRO +import com.twitter.storehaus_internal.manhattan.ManhattanROConfig +import com.twitter.storehaus_internal.util.ApplicationID +import com.twitter.storehaus_internal.util.DatasetName +import com.twitter.storehaus_internal.util.HDFSPath +import javax.inject.Named +import javax.inject.Singleton + +object EmbeddingStoreModule extends TwitterModule { + type UserId = Long + implicit val mbcgUserEmbeddingInjection: Injection[api.Embedding, Array[Byte]] = + CompactScalaCodec(api.Embedding) + implicit val tweetCandidatesInjection: Injection[CandidateTweetsList, Array[Byte]] = + CompactScalaCodec(CandidateTweetsList) + + final val TwHINEmbeddingRegularUpdateMhStoreName = "TwHINEmbeddingRegularUpdateMhStore" + @Provides + @Singleton + @Named(TwHINEmbeddingRegularUpdateMhStoreName) + def twHINEmbeddingRegularUpdateMhStore( + serviceIdentifier: ServiceIdentifier + ): ReadableStore[InternalId, api.Embedding] = { + val binaryEmbeddingInjection: Injection[api.Embedding, Array[Byte]] = + BinaryScalaCodec(api.Embedding) + + val longCodec = implicitly[Injection[Long, Array[Byte]]] + + ManhattanRO + .getReadableStoreWithMtls[TweetId, api.Embedding]( + ManhattanROConfig( + HDFSPath(""), // not needed + ApplicationID("cr_mixer_apollo"), + DatasetName("twhin_regular_update_tweet_embedding_apollo"), + Apollo + ), + ManhattanKVClientMtlsParams(serviceIdentifier) + )(longCodec, binaryEmbeddingInjection).composeKeyMapping[InternalId] { + case InternalId.TweetId(tweetId) => + tweetId + case _ => + throw new UnsupportedOperationException("Invalid Internal Id") + } + } + + final val ConsumerBasedTwHINEmbeddingRegularUpdateMhStoreName = + "ConsumerBasedTwHINEmbeddingRegularUpdateMhStore" + @Provides + @Singleton + @Named(ConsumerBasedTwHINEmbeddingRegularUpdateMhStoreName) + def consumerBasedTwHINEmbeddingRegularUpdateMhStore( + serviceIdentifier: ServiceIdentifier + ): ReadableStore[InternalId, api.Embedding] = { + val binaryEmbeddingInjection: Injection[api.Embedding, Array[Byte]] = + BinaryScalaCodec(api.Embedding) + + val longCodec = implicitly[Injection[Long, Array[Byte]]] + + ManhattanRO + .getReadableStoreWithMtls[UserId, api.Embedding]( + ManhattanROConfig( + HDFSPath(""), // not needed + ApplicationID("cr_mixer_apollo"), + DatasetName("twhin_user_embedding_regular_update_apollo"), + Apollo + ), + ManhattanKVClientMtlsParams(serviceIdentifier) + )(longCodec, binaryEmbeddingInjection).composeKeyMapping[InternalId] { + case InternalId.UserId(userId) => + userId + case _ => + throw new UnsupportedOperationException("Invalid Internal Id") + } + } + + final val TwoTowerFavConsumerEmbeddingMhStoreName = "TwoTowerFavConsumerEmbeddingMhStore" + @Provides + @Singleton + @Named(TwoTowerFavConsumerEmbeddingMhStoreName) + def twoTowerFavConsumerEmbeddingMhStore( + serviceIdentifier: ServiceIdentifier + ): ReadableStore[InternalId, api.Embedding] = { + val binaryEmbeddingInjection: Injection[api.Embedding, Array[Byte]] = + BinaryScalaCodec(api.Embedding) + + val longCodec = implicitly[Injection[Long, Array[Byte]]] + + ManhattanRO + .getReadableStoreWithMtls[UserId, api.Embedding]( + ManhattanROConfig( + HDFSPath(""), // not needed + ApplicationID("cr_mixer_apollo"), + DatasetName("two_tower_fav_user_embedding_apollo"), + Apollo + ), + ManhattanKVClientMtlsParams(serviceIdentifier) + )(longCodec, binaryEmbeddingInjection).composeKeyMapping[InternalId] { + case InternalId.UserId(userId) => + userId + case _ => + throw new UnsupportedOperationException("Invalid Internal Id") + } + } + + final val DebuggerDemoUserEmbeddingMhStoreName = "DebuggerDemoUserEmbeddingMhStoreName" + @Provides + @Singleton + @Named(DebuggerDemoUserEmbeddingMhStoreName) + def debuggerDemoUserEmbeddingStore( + serviceIdentifier: ServiceIdentifier + ): ReadableStore[InternalId, api.Embedding] = { + // This dataset is from src/scala/com/twitter/wtf/beam/bq_embedding_export/sql/MlfExperimentalUserEmbeddingScalaDataset.sql + // Change the above sql if you want to use a diff embedding + val manhattanROConfig = ManhattanROConfig( + HDFSPath(""), // not needed + ApplicationID("cr_mixer_apollo"), + DatasetName("experimental_user_embedding"), + Apollo + ) + buildUserEmbeddingStore(serviceIdentifier, manhattanROConfig) + } + + final val DebuggerDemoTweetEmbeddingMhStoreName = "DebuggerDemoTweetEmbeddingMhStore" + @Provides + @Singleton + @Named(DebuggerDemoTweetEmbeddingMhStoreName) + def debuggerDemoTweetEmbeddingStore( + serviceIdentifier: ServiceIdentifier + ): ReadableStore[InternalId, api.Embedding] = { + // This dataset is from src/scala/com/twitter/wtf/beam/bq_embedding_export/sql/MlfExperimentalTweetEmbeddingScalaDataset.sql + // Change the above sql if you want to use a diff embedding + val manhattanROConfig = ManhattanROConfig( + HDFSPath(""), // not needed + ApplicationID("cr_mixer_apollo"), + DatasetName("experimental_tweet_embedding"), + Apollo + ) + buildTweetEmbeddingStore(serviceIdentifier, manhattanROConfig) + } + + private def buildUserEmbeddingStore( + serviceIdentifier: ServiceIdentifier, + manhattanROConfig: ManhattanROConfig + ): ReadableStore[InternalId, api.Embedding] = { + val binaryEmbeddingInjection: Injection[api.Embedding, Array[Byte]] = + BinaryScalaCodec(api.Embedding) + + val longCodec = implicitly[Injection[Long, Array[Byte]]] + ManhattanRO + .getReadableStoreWithMtls[UserId, api.Embedding]( + manhattanROConfig, + ManhattanKVClientMtlsParams(serviceIdentifier) + )(longCodec, binaryEmbeddingInjection).composeKeyMapping[InternalId] { + case InternalId.UserId(userId) => + userId + case _ => + throw new UnsupportedOperationException("Invalid Internal Id") + } + } + + private def buildTweetEmbeddingStore( + serviceIdentifier: ServiceIdentifier, + manhattanROConfig: ManhattanROConfig + ): ReadableStore[InternalId, api.Embedding] = { + val binaryEmbeddingInjection: Injection[api.Embedding, Array[Byte]] = + BinaryScalaCodec(api.Embedding) + + val longCodec = implicitly[Injection[Long, Array[Byte]]] + + ManhattanRO + .getReadableStoreWithMtls[TweetId, api.Embedding]( + manhattanROConfig, + ManhattanKVClientMtlsParams(serviceIdentifier) + )(longCodec, binaryEmbeddingInjection).composeKeyMapping[InternalId] { + case InternalId.TweetId(tweetId) => + tweetId + case _ => + throw new UnsupportedOperationException("Invalid Internal Id") + } + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/FrsStoreModule.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/FrsStoreModule.scala new file mode 100644 index 0000000000..cfe044afd4 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/FrsStoreModule.scala @@ -0,0 +1,29 @@ +package com.twitter.cr_mixer.module + +import com.google.inject.Provides +import com.google.inject.Singleton +import com.twitter.cr_mixer.model.ModuleNames +import com.twitter.cr_mixer.param.decider.CrMixerDecider +import com.twitter.cr_mixer.source_signal.FrsStore +import com.twitter.cr_mixer.source_signal.FrsStore.FrsQueryResult +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.inject.TwitterModule +import com.twitter.follow_recommendations.thriftscala.FollowRecommendationsThriftService +import com.twitter.hermit.store.common.ObservedReadableStore +import com.twitter.storehaus.ReadableStore +import javax.inject.Named + +object FrsStoreModule extends TwitterModule { + + @Provides + @Singleton + @Named(ModuleNames.FrsStore) + def providesFrsStore( + frsClient: FollowRecommendationsThriftService.MethodPerEndpoint, + statsReceiver: StatsReceiver, + decider: CrMixerDecider + ): ReadableStore[FrsStore.Query, Seq[FrsQueryResult]] = { + ObservedReadableStore(FrsStore(frsClient, statsReceiver, decider))( + statsReceiver.scope("follow_recommendations_store")) + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/MHMtlsParamsModule.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/MHMtlsParamsModule.scala new file mode 100644 index 0000000000..339d0330ab --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/MHMtlsParamsModule.scala @@ -0,0 +1,17 @@ +package com.twitter.cr_mixer.module + +import com.google.inject.Provides +import com.twitter.finagle.mtls.authentication.ServiceIdentifier +import com.twitter.inject.TwitterModule +import com.twitter.storage.client.manhattan.kv.ManhattanKVClientMtlsParams +import javax.inject.Singleton + +object MHMtlsParamsModule extends TwitterModule { + @Singleton + @Provides + def providesManhattanMtlsParams( + serviceIdentifier: ServiceIdentifier + ): ManhattanKVClientMtlsParams = { + ManhattanKVClientMtlsParams(serviceIdentifier) + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/OfflineCandidateStoreModule.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/OfflineCandidateStoreModule.scala new file mode 100644 index 0000000000..db4a3fa5e1 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/OfflineCandidateStoreModule.scala @@ -0,0 +1,150 @@ +package com.twitter.cr_mixer.module + +import com.google.inject.Provides +import com.twitter.bijection.Injection +import com.twitter.bijection.scrooge.CompactScalaCodec +import com.twitter.cr_mixer.model.ModuleNames +import com.twitter.finagle.mtls.authentication.ServiceIdentifier +import com.twitter.inject.TwitterModule +import com.twitter.simclusters_v2.thriftscala.CandidateTweetsList +import com.twitter.storage.client.manhattan.kv.ManhattanKVClientMtlsParams +import com.twitter.storehaus.ReadableStore +import com.twitter.storehaus_internal.manhattan.Apollo +import com.twitter.storehaus_internal.manhattan.ManhattanRO +import com.twitter.storehaus_internal.manhattan.ManhattanROConfig +import com.twitter.storehaus_internal.util.ApplicationID +import com.twitter.storehaus_internal.util.DatasetName +import com.twitter.storehaus_internal.util.HDFSPath +import javax.inject.Named +import javax.inject.Singleton + +object OfflineCandidateStoreModule extends TwitterModule { + type UserId = Long + implicit val tweetCandidatesInjection: Injection[CandidateTweetsList, Array[Byte]] = + CompactScalaCodec(CandidateTweetsList) + + @Provides + @Singleton + @Named(ModuleNames.OfflineTweet2020CandidateStore) + def offlineTweet2020CandidateMhStore( + serviceIdentifier: ServiceIdentifier + ): ReadableStore[UserId, CandidateTweetsList] = { + buildOfflineCandidateStore( + serviceIdentifier, + datasetName = "offline_tweet_recommendations_from_interestedin_2020" + ) + } + + @Provides + @Singleton + @Named(ModuleNames.OfflineTweet2020Hl0El15CandidateStore) + def offlineTweet2020Hl0El15CandidateMhStore( + serviceIdentifier: ServiceIdentifier + ): ReadableStore[UserId, CandidateTweetsList] = { + buildOfflineCandidateStore( + serviceIdentifier, + datasetName = "offline_tweet_recommendations_from_interestedin_2020_hl_0_el_15" + ) + } + + @Provides + @Singleton + @Named(ModuleNames.OfflineTweet2020Hl2El15CandidateStore) + def offlineTweet2020Hl2El15CandidateMhStore( + serviceIdentifier: ServiceIdentifier + ): ReadableStore[UserId, CandidateTweetsList] = { + buildOfflineCandidateStore( + serviceIdentifier, + datasetName = "offline_tweet_recommendations_from_interestedin_2020_hl_2_el_15" + ) + } + + @Provides + @Singleton + @Named(ModuleNames.OfflineTweet2020Hl2El50CandidateStore) + def offlineTweet2020Hl2El50CandidateMhStore( + serviceIdentifier: ServiceIdentifier + ): ReadableStore[UserId, CandidateTweetsList] = { + buildOfflineCandidateStore( + serviceIdentifier, + datasetName = "offline_tweet_recommendations_from_interestedin_2020_hl_2_el_50" + ) + } + + @Provides + @Singleton + @Named(ModuleNames.OfflineTweet2020Hl8El50CandidateStore) + def offlineTweet2020Hl8El50CandidateMhStore( + serviceIdentifier: ServiceIdentifier + ): ReadableStore[UserId, CandidateTweetsList] = { + buildOfflineCandidateStore( + serviceIdentifier, + datasetName = "offline_tweet_recommendations_from_interestedin_2020_hl_8_el_50" + ) + } + + @Provides + @Singleton + @Named(ModuleNames.OfflineTweetMTSCandidateStore) + def offlineTweetMTSCandidateMhStore( + serviceIdentifier: ServiceIdentifier + ): ReadableStore[UserId, CandidateTweetsList] = { + buildOfflineCandidateStore( + serviceIdentifier, + datasetName = "offline_tweet_recommendations_from_mts_consumer_embeddings" + ) + } + + @Provides + @Singleton + @Named(ModuleNames.OfflineFavDecayedSumCandidateStore) + def offlineFavDecayedSumCandidateStore( + serviceIdentifier: ServiceIdentifier + ): ReadableStore[UserId, CandidateTweetsList] = { + buildOfflineCandidateStore( + serviceIdentifier, + datasetName = "offline_tweet_recommendations_from_decayed_sum" + ) + } + + @Provides + @Singleton + @Named(ModuleNames.OfflineFtrAt5Pop1000RankDecay11CandidateStore) + def offlineFtrAt5Pop1000RankDecay11CandidateStore( + serviceIdentifier: ServiceIdentifier + ): ReadableStore[UserId, CandidateTweetsList] = { + buildOfflineCandidateStore( + serviceIdentifier, + datasetName = "offline_tweet_recommendations_from_ftrat5_pop1000_rank_decay_1_1" + ) + } + + @Provides + @Singleton + @Named(ModuleNames.OfflineFtrAt5Pop10000RankDecay11CandidateStore) + def offlineFtrAt5Pop10000RankDecay11CandidateStore( + serviceIdentifier: ServiceIdentifier + ): ReadableStore[UserId, CandidateTweetsList] = { + buildOfflineCandidateStore( + serviceIdentifier, + datasetName = "offline_tweet_recommendations_from_ftrat5_pop10000_rank_decay_1_1" + ) + } + + private def buildOfflineCandidateStore( + serviceIdentifier: ServiceIdentifier, + datasetName: String + ): ReadableStore[UserId, CandidateTweetsList] = { + ManhattanRO + .getReadableStoreWithMtls[Long, CandidateTweetsList]( + ManhattanROConfig( + HDFSPath(""), // not needed + ApplicationID("multi_type_simclusters"), + DatasetName(datasetName), + Apollo + ), + ManhattanKVClientMtlsParams(serviceIdentifier) + ) + } + +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/RealGraphOonStoreModule.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/RealGraphOonStoreModule.scala new file mode 100644 index 0000000000..3d9a71a1c4 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/RealGraphOonStoreModule.scala @@ -0,0 +1,39 @@ +package com.twitter.cr_mixer.module + +import com.google.inject.Provides +import com.twitter.app.Flag +import com.twitter.cr_mixer.model.ModuleNames +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.frigate.common.store.strato.StratoFetchableStore +import com.twitter.hermit.store.common.ObservedReadableStore +import com.twitter.inject.TwitterModule +import com.twitter.simclusters_v2.common.UserId +import com.twitter.storehaus.ReadableStore +import javax.inject.Named +import javax.inject.Singleton +import com.twitter.strato.client.{Client => StratoClient} +import com.twitter.wtf.candidate.thriftscala.CandidateSeq + +object RealGraphOonStoreModule extends TwitterModule { + + private val userRealGraphOonColumnPath: Flag[String] = flag[String]( + name = "crMixer.userRealGraphOonColumnPath", + default = "recommendations/twistly/userRealgraphOon", + help = "Strato column path for user real graph OON Store" + ) + + @Provides + @Singleton + @Named(ModuleNames.RealGraphOonStore) + def providesRealGraphOonStore( + stratoClient: StratoClient, + statsReceiver: StatsReceiver + ): ReadableStore[UserId, CandidateSeq] = { + val realGraphOonStratoFetchableStore = StratoFetchableStore + .withUnitView[UserId, CandidateSeq](stratoClient, userRealGraphOonColumnPath()) + + ObservedReadableStore( + realGraphOonStratoFetchableStore + )(statsReceiver.scope("user_real_graph_oon_store")) + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/RealGraphStoreMhModule.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/RealGraphStoreMhModule.scala new file mode 100644 index 0000000000..0cd1a3ad74 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/RealGraphStoreMhModule.scala @@ -0,0 +1,67 @@ +package com.twitter.cr_mixer.module + +import com.google.inject.Provides +import com.google.inject.Singleton +import com.google.inject.name.Named +import com.twitter.inject.TwitterModule +import com.twitter.simclusters_v2.common.UserId +import com.twitter.conversions.DurationOps._ +import com.twitter.cr_mixer.model.ModuleNames +import com.twitter.cr_mixer.param.decider.CrMixerDecider +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.finagle.memcached.{Client => MemcachedClient} +import com.twitter.storage.client.manhattan.kv.ManhattanKVClientMtlsParams +import com.twitter.storehaus.ReadableStore +import com.twitter.storehaus_internal.manhattan.Apollo +import com.twitter.storehaus_internal.manhattan.ManhattanRO +import com.twitter.storehaus_internal.manhattan.ManhattanROConfig +import com.twitter.storehaus_internal.util.ApplicationID +import com.twitter.storehaus_internal.util.DatasetName +import com.twitter.storehaus_internal.util.HDFSPath +import com.twitter.bijection.scrooge.BinaryScalaCodec +import com.twitter.cr_mixer.param.decider.DeciderKey +import com.twitter.hermit.store.common.DeciderableReadableStore +import com.twitter.hermit.store.common.ObservedMemcachedReadableStore +import com.twitter.wtf.candidate.thriftscala.CandidateSeq + +object RealGraphStoreMhModule extends TwitterModule { + + @Provides + @Singleton + @Named(ModuleNames.RealGraphInStore) + def providesRealGraphStoreMh( + decider: CrMixerDecider, + statsReceiver: StatsReceiver, + manhattanKVClientMtlsParams: ManhattanKVClientMtlsParams, + @Named(ModuleNames.UnifiedCache) crMixerUnifiedCacheClient: MemcachedClient, + ): ReadableStore[UserId, CandidateSeq] = { + + implicit val valueCodec = new BinaryScalaCodec(CandidateSeq) + val underlyingStore = ManhattanRO + .getReadableStoreWithMtls[UserId, CandidateSeq]( + ManhattanROConfig( + HDFSPath(""), + ApplicationID("cr_mixer_apollo"), + DatasetName("real_graph_scores_apollo"), + Apollo), + manhattanKVClientMtlsParams + ) + + val memCachedStore = ObservedMemcachedReadableStore + .fromCacheClient( + backingStore = underlyingStore, + cacheClient = crMixerUnifiedCacheClient, + ttl = 24.hours, + )( + valueInjection = valueCodec, + statsReceiver = statsReceiver.scope("memCachedUserRealGraphMh"), + keyToString = { k: UserId => s"uRGraph/$k" } + ) + + DeciderableReadableStore( + memCachedStore, + decider.deciderGateBuilder.idGate(DeciderKey.enableRealGraphMhStoreDeciderKey), + statsReceiver.scope("RealGraphMh") + ) + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/RepresentationManagerModule.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/RepresentationManagerModule.scala new file mode 100644 index 0000000000..227e5fff30 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/RepresentationManagerModule.scala @@ -0,0 +1,107 @@ +package com.twitter.cr_mixer.module + +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.inject.TwitterModule +import com.twitter.simclusters_v2.common.SimClustersEmbedding +import com.twitter.simclusters_v2.common.TweetId +import com.twitter.simclusters_v2.common.UserId +import com.twitter.storehaus.ReadableStore +import com.twitter.strato.client.{Client => StratoClient} +import com.twitter.representation_manager.thriftscala.SimClustersEmbeddingView +import com.twitter.simclusters_v2.thriftscala.EmbeddingType +import com.twitter.simclusters_v2.thriftscala.ModelVersion +import com.google.inject.Provides +import com.google.inject.Singleton +import javax.inject.Named +import com.twitter.cr_mixer.model.ModuleNames +import com.twitter.frigate.common.store.strato.StratoFetchableStore +import com.twitter.hermit.store.common.ObservedReadableStore +import com.twitter.simclusters_v2.thriftscala.{SimClustersEmbedding => ThriftSimClustersEmbedding} + +object RepresentationManagerModule extends TwitterModule { + private val ColPathPrefix = "recommendations/representation_manager/" + private val SimclustersTweetColPath = ColPathPrefix + "simClustersEmbedding.Tweet" + private val SimclustersUserColPath = ColPathPrefix + "simClustersEmbedding.User" + + @Provides + @Singleton + @Named(ModuleNames.RmsTweetLogFavLongestL2EmbeddingStore) + def providesRepresentationManagerTweetStore( + statsReceiver: StatsReceiver, + stratoClient: StratoClient, + ): ReadableStore[TweetId, SimClustersEmbedding] = { + ObservedReadableStore( + StratoFetchableStore + .withView[Long, SimClustersEmbeddingView, ThriftSimClustersEmbedding]( + stratoClient, + SimclustersTweetColPath, + SimClustersEmbeddingView( + EmbeddingType.LogFavLongestL2EmbeddingTweet, + ModelVersion.Model20m145k2020)) + .mapValues(SimClustersEmbedding(_)))( + statsReceiver.scope("rms_tweet_log_fav_longest_l2_store")) + } + + @Provides + @Singleton + @Named(ModuleNames.RmsUserFavBasedProducerEmbeddingStore) + def providesRepresentationManagerUserFavBasedProducerEmbeddingStore( + statsReceiver: StatsReceiver, + stratoClient: StratoClient, + ): ReadableStore[UserId, SimClustersEmbedding] = { + ObservedReadableStore( + StratoFetchableStore + .withView[Long, SimClustersEmbeddingView, ThriftSimClustersEmbedding]( + stratoClient, + SimclustersUserColPath, + SimClustersEmbeddingView( + EmbeddingType.FavBasedProducer, + ModelVersion.Model20m145k2020 + ) + ) + .mapValues(SimClustersEmbedding(_)))( + statsReceiver.scope("rms_user_fav_based_producer_store")) + } + + @Provides + @Singleton + @Named(ModuleNames.RmsUserLogFavInterestedInEmbeddingStore) + def providesRepresentationManagerUserLogFavConsumerEmbeddingStore( + statsReceiver: StatsReceiver, + stratoClient: StratoClient, + ): ReadableStore[UserId, SimClustersEmbedding] = { + ObservedReadableStore( + StratoFetchableStore + .withView[Long, SimClustersEmbeddingView, ThriftSimClustersEmbedding]( + stratoClient, + SimclustersUserColPath, + SimClustersEmbeddingView( + EmbeddingType.LogFavBasedUserInterestedIn, + ModelVersion.Model20m145k2020 + ) + ) + .mapValues(SimClustersEmbedding(_)))( + statsReceiver.scope("rms_user_log_fav_interestedin_store")) + } + + @Provides + @Singleton + @Named(ModuleNames.RmsUserFollowInterestedInEmbeddingStore) + def providesRepresentationManagerUserFollowInterestedInEmbeddingStore( + statsReceiver: StatsReceiver, + stratoClient: StratoClient, + ): ReadableStore[UserId, SimClustersEmbedding] = { + ObservedReadableStore( + StratoFetchableStore + .withView[Long, SimClustersEmbeddingView, ThriftSimClustersEmbedding]( + stratoClient, + SimclustersUserColPath, + SimClustersEmbeddingView( + EmbeddingType.FollowBasedUserInterestedIn, + ModelVersion.Model20m145k2020 + ) + ) + .mapValues(SimClustersEmbedding(_)))( + statsReceiver.scope("rms_user_follow_interestedin_store")) + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/RepresentationScorerModule.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/RepresentationScorerModule.scala new file mode 100644 index 0000000000..7db6474cc2 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/RepresentationScorerModule.scala @@ -0,0 +1,56 @@ +package com.twitter.cr_mixer.module + +import com.twitter.inject.TwitterModule +import com.twitter.simclusters_v2.thriftscala.EmbeddingType +import com.twitter.simclusters_v2.thriftscala.InternalId +import com.twitter.simclusters_v2.thriftscala.ModelVersion +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.frigate.common.store.strato.StratoFetchableStore +import com.twitter.simclusters_v2.common.UserId +import com.twitter.simclusters_v2.common.TweetId +import com.twitter.strato.client.{Client => StratoClient} +import com.twitter.storehaus.ReadableStore +import com.twitter.simclusters_v2.thriftscala.ScoringAlgorithm +import com.google.inject.Provides +import com.google.inject.Singleton +import com.twitter.hermit.store.common.ObservedReadableStore +import javax.inject.Named +import com.twitter.cr_mixer.model.ModuleNames +import com.twitter.representationscorer.thriftscala.ListScoreId + +object RepresentationScorerModule extends TwitterModule { + + private val rsxColumnPath = "recommendations/representation_scorer/listScore" + + private final val SimClusterModelVersion = ModelVersion.Model20m145k2020 + private final val TweetEmbeddingType = EmbeddingType.LogFavBasedTweet + + @Provides + @Singleton + @Named(ModuleNames.RsxStore) + def providesRepresentationScorerStore( + statsReceiver: StatsReceiver, + stratoClient: StratoClient, + ): ReadableStore[(UserId, TweetId), Double] = { + ObservedReadableStore( + StratoFetchableStore + .withUnitView[ListScoreId, Double](stratoClient, rsxColumnPath).composeKeyMapping[( + UserId, + TweetId + )] { key => + representationScorerStoreKeyMapping(key._1, key._2) + } + )(statsReceiver.scope("rsx_store")) + } + + private def representationScorerStoreKeyMapping(t1: TweetId, t2: TweetId): ListScoreId = { + ListScoreId( + algorithm = ScoringAlgorithm.PairEmbeddingLogCosineSimilarity, + modelVersion = SimClusterModelVersion, + targetEmbeddingType = TweetEmbeddingType, + targetId = InternalId.TweetId(t1), + candidateEmbeddingType = TweetEmbeddingType, + candidateIds = Seq(InternalId.TweetId(t2)) + ) + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/SampleSimilarityEngineModule.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/SampleSimilarityEngineModule.scala new file mode 100644 index 0000000000..98c3f4af6b --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/SampleSimilarityEngineModule.scala @@ -0,0 +1,90 @@ +package com.twitter.cr_mixer.module + +import com.google.inject.Provides +import com.twitter.cr_mixer.config.TimeoutConfig +import com.twitter.cr_mixer.similarity_engine.StandardSimilarityEngine +import com.twitter.cr_mixer.similarity_engine.LookupSimilarityEngine +import com.twitter.cr_mixer.similarity_engine.SimilarityEngine.GatingConfig +import com.twitter.cr_mixer.similarity_engine.SimilarityEngine.SimilarityEngineConfig +import com.twitter.cr_mixer.thriftscala.SimilarityEngineType +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.inject.TwitterModule +import com.twitter.simclusters_v2.common.TweetId +import com.twitter.simclusters_v2.common.UserId +import com.twitter.storehaus.ReadableStore +import javax.inject.Singleton + +/** + * In this example we build a [[StandardSimilarityEngine]] to wrap a dummy store + */ +object SimpleSimilarityEngineModule extends TwitterModule { + @Provides + @Singleton + def providesSimpleSimilarityEngine( + timeoutConfig: TimeoutConfig, + globalStats: StatsReceiver + ): StandardSimilarityEngine[UserId, (TweetId, Double)] = { + // Inject your readableStore implementation here + val dummyStore = ReadableStore.fromMap( + Map( + 1L -> Seq((100L, 1.0), (101L, 1.0)), + 2L -> Seq((200L, 2.0), (201L, 2.0)), + 3L -> Seq((300L, 3.0), (301L, 3.0)) + )) + + new StandardSimilarityEngine[UserId, (TweetId, Double)]( + implementingStore = dummyStore, + identifier = SimilarityEngineType.EnumUnknownSimilarityEngineType(9997), + globalStats = globalStats, + engineConfig = SimilarityEngineConfig( + timeout = timeoutConfig.similarityEngineTimeout, + gatingConfig = GatingConfig( + deciderConfig = None, + enableFeatureSwitch = None + ) + ) + ) + } +} + +/** + * In this example we build a [[LookupSimilarityEngine]] to wrap a dummy store with 2 versions + */ +object LookupSimilarityEngineModule extends TwitterModule { + @Provides + @Singleton + def providesLookupSimilarityEngine( + timeoutConfig: TimeoutConfig, + globalStats: StatsReceiver + ): LookupSimilarityEngine[UserId, (TweetId, Double)] = { + // Inject your readableStore implementation here + val dummyStoreV1 = ReadableStore.fromMap( + Map( + 1L -> Seq((100L, 1.0), (101L, 1.0)), + 2L -> Seq((200L, 2.0), (201L, 2.0)), + )) + + val dummyStoreV2 = ReadableStore.fromMap( + Map( + 1L -> Seq((100L, 1.0), (101L, 1.0)), + 2L -> Seq((200L, 2.0), (201L, 2.0)), + )) + + new LookupSimilarityEngine[UserId, (TweetId, Double)]( + versionedStoreMap = Map( + "V1" -> dummyStoreV1, + "V2" -> dummyStoreV2 + ), + identifier = SimilarityEngineType.EnumUnknownSimilarityEngineType(9998), + globalStats = globalStats, + engineConfig = SimilarityEngineConfig( + timeout = timeoutConfig.similarityEngineTimeout, + gatingConfig = GatingConfig( + deciderConfig = None, + enableFeatureSwitch = None + ) + ) + ) + } + +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/SimClustersANNServiceNameToClientMapper.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/SimClustersANNServiceNameToClientMapper.scala new file mode 100644 index 0000000000..305839816d --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/SimClustersANNServiceNameToClientMapper.scala @@ -0,0 +1,33 @@ +package com.twitter.cr_mixer.module + +import com.google.inject.Provides +import com.google.inject.Singleton +import com.twitter.cr_mixer.model.ModuleNames +import com.twitter.inject.TwitterModule +import com.twitter.simclustersann.thriftscala.SimClustersANNService +import javax.inject.Named + +object SimClustersANNServiceNameToClientMapper extends TwitterModule { + + @Provides + @Singleton + def providesSimClustersANNServiceNameToClientMapping( + @Named(ModuleNames.ProdSimClustersANNServiceClientName) simClustersANNServiceProd: SimClustersANNService.MethodPerEndpoint, + @Named(ModuleNames.ExperimentalSimClustersANNServiceClientName) simClustersANNServiceExperimental: SimClustersANNService.MethodPerEndpoint, + @Named(ModuleNames.SimClustersANNServiceClientName1) simClustersANNService1: SimClustersANNService.MethodPerEndpoint, + @Named(ModuleNames.SimClustersANNServiceClientName2) simClustersANNService2: SimClustersANNService.MethodPerEndpoint, + @Named(ModuleNames.SimClustersANNServiceClientName3) simClustersANNService3: SimClustersANNService.MethodPerEndpoint, + @Named(ModuleNames.SimClustersANNServiceClientName5) simClustersANNService5: SimClustersANNService.MethodPerEndpoint, + @Named(ModuleNames.SimClustersANNServiceClientName4) simClustersANNService4: SimClustersANNService.MethodPerEndpoint + ): Map[String, SimClustersANNService.MethodPerEndpoint] = { + Map[String, SimClustersANNService.MethodPerEndpoint]( + "simclusters-ann" -> simClustersANNServiceProd, + "simclusters-ann-experimental" -> simClustersANNServiceExperimental, + "simclusters-ann-1" -> simClustersANNService1, + "simclusters-ann-2" -> simClustersANNService2, + "simclusters-ann-3" -> simClustersANNService3, + "simclusters-ann-5" -> simClustersANNService5, + "simclusters-ann-4" -> simClustersANNService4 + ) + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/SkitStratoStoreModule.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/SkitStratoStoreModule.scala new file mode 100644 index 0000000000..318c2ed00b --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/SkitStratoStoreModule.scala @@ -0,0 +1,65 @@ +package com.twitter.cr_mixer.module + +import com.google.inject.Provides +import com.google.inject.Singleton +import com.google.inject.name.Named +import com.twitter.conversions.DurationOps._ +import com.twitter.cr_mixer.model.ModuleNames +import com.twitter.cr_mixer.similarity_engine.SimilarityEngine.keyHasher +import com.twitter.finagle.memcached.{Client => MemcachedClient} +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.frigate.common.store.strato.StratoFetchableStore +import com.twitter.hermit.store.common.ObservedCachedReadableStore +import com.twitter.hermit.store.common.ObservedMemcachedReadableStore +import com.twitter.hermit.store.common.ObservedReadableStore +import com.twitter.inject.TwitterModule +import com.twitter.relevance_platform.common.injection.LZ4Injection +import com.twitter.relevance_platform.common.injection.SeqObjectInjection +import com.twitter.storehaus.ReadableStore +import com.twitter.strato.client.Client +import com.twitter.topic_recos.thriftscala.TopicTopTweets +import com.twitter.topic_recos.thriftscala.TopicTweet +import com.twitter.topic_recos.thriftscala.TopicTweetPartitionFlatKey + +/** + * Strato store that wraps the topic top tweets pipeline indexed from a Summingbird job + */ +object SkitStratoStoreModule extends TwitterModule { + + val column = "recommendations/topic_recos/topicTopTweets" + + @Provides + @Singleton + @Named(ModuleNames.SkitStratoStoreName) + def providesSkitStratoStore( + @Named(ModuleNames.UnifiedCache) crMixerUnifiedCacheClient: MemcachedClient, + stratoClient: Client, + statsReceiver: StatsReceiver + ): ReadableStore[TopicTweetPartitionFlatKey, Seq[TopicTweet]] = { + val skitStore = ObservedReadableStore( + StratoFetchableStore + .withUnitView[TopicTweetPartitionFlatKey, TopicTopTweets](stratoClient, column))( + statsReceiver.scope(ModuleNames.SkitStratoStoreName)).mapValues { topicTopTweets => + topicTopTweets.topTweets + } + + val memCachedStore = ObservedMemcachedReadableStore + .fromCacheClient( + backingStore = skitStore, + cacheClient = crMixerUnifiedCacheClient, + ttl = 10.minutes + )( + valueInjection = LZ4Injection.compose(SeqObjectInjection[TopicTweet]()), + statsReceiver = statsReceiver.scope("memcached_skit_store"), + keyToString = { k => s"skit:${keyHasher.hashKey(k.toString.getBytes)}" } + ) + + ObservedCachedReadableStore.from[TopicTweetPartitionFlatKey, Seq[TopicTweet]]( + memCachedStore, + ttl = 5.minutes, + maxKeys = 100000, // ~150MB max + cacheName = "skit_in_memory_cache", + windowSize = 10000L + )(statsReceiver.scope("skit_in_memory_cache")) + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/StrongTiePredictionStoreModule.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/StrongTiePredictionStoreModule.scala new file mode 100644 index 0000000000..51d5560771 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/StrongTiePredictionStoreModule.scala @@ -0,0 +1,39 @@ +package com.twitter.cr_mixer.module + +import com.google.inject.Provides +import com.google.inject.Singleton +import com.twitter.app.Flag +import com.twitter.cr_mixer.model.ModuleNames +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.frigate.common.store.strato.StratoFetchableStore +import com.twitter.hermit.store.common.ObservedReadableStore +import com.twitter.inject.TwitterModule +import com.twitter.simclusters_v2.common.UserId +import com.twitter.hermit.stp.thriftscala.STPResult +import com.twitter.storehaus.ReadableStore +import com.twitter.strato.client.{Client => StratoClient} +import javax.inject.Named + +object StrongTiePredictionStoreModule extends TwitterModule { + + private val strongTiePredictionColumnPath: Flag[String] = flag[String]( + name = "crMixer.strongTiePredictionColumnPath", + default = "onboarding/userrecs/strong_tie_prediction_big", + help = "Strato column path for StrongTiePredictionStore" + ) + + @Provides + @Singleton + @Named(ModuleNames.StpStore) + def providesStrongTiePredictionStore( + statsReceiver: StatsReceiver, + stratoClient: StratoClient, + ): ReadableStore[UserId, STPResult] = { + val strongTiePredictionStratoFetchableStore = StratoFetchableStore + .withUnitView[UserId, STPResult](stratoClient, strongTiePredictionColumnPath()) + + ObservedReadableStore( + strongTiePredictionStratoFetchableStore + )(statsReceiver.scope("strong_tie_prediction_big_store")) + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/TripCandidateStoreModule.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/TripCandidateStoreModule.scala new file mode 100644 index 0000000000..802d5c9869 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/TripCandidateStoreModule.scala @@ -0,0 +1,34 @@ +package com.twitter.cr_mixer.module + +import com.google.inject.Provides +import com.twitter.cr_mixer.model.ModuleNames +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.frigate.common.store.strato.StratoFetchableStore +import com.twitter.hermit.store.common.ObservedReadableStore +import com.twitter.inject.TwitterModule +import com.twitter.storehaus.ReadableStore +import com.twitter.strato.client.{Client => StratoClient} +import com.twitter.trends.trip_v1.trip_tweets.thriftscala.TripTweet +import com.twitter.trends.trip_v1.trip_tweets.thriftscala.TripTweets +import com.twitter.trends.trip_v1.trip_tweets.thriftscala.TripDomain +import javax.inject.Named + +object TripCandidateStoreModule extends TwitterModule { + private val stratoColumn = "trends/trip/tripTweetsDataflowProd" + + @Provides + @Named(ModuleNames.TripCandidateStore) + def providesSimClustersTripCandidateStore( + statsReceiver: StatsReceiver, + stratoClient: StratoClient + ): ReadableStore[TripDomain, Seq[TripTweet]] = { + val tripCandidateStratoFetchableStore = + StratoFetchableStore + .withUnitView[TripDomain, TripTweets](stratoClient, stratoColumn) + .mapValues(_.tweets) + + ObservedReadableStore( + tripCandidateStratoFetchableStore + )(statsReceiver.scope("simclusters_trip_candidate_store")) + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/TweetInfoStoreModule.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/TweetInfoStoreModule.scala new file mode 100644 index 0000000000..a3a794e8e5 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/TweetInfoStoreModule.scala @@ -0,0 +1,205 @@ +package com.twitter.cr_mixer.module + +import com.google.inject.Module +import com.google.inject.Provides +import com.google.inject.Singleton +import com.twitter.bijection.scrooge.BinaryScalaCodec +import com.twitter.contentrecommender.thriftscala.TweetInfo +import com.twitter.conversions.DurationOps._ +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.finagle.memcached.{Client => MemcachedClient} +import com.twitter.finagle.mtls.authentication.ServiceIdentifier +import com.twitter.frigate.common.store.health.TweetHealthModelStore +import com.twitter.frigate.common.store.health.TweetHealthModelStore.TweetHealthModelStoreConfig +import com.twitter.frigate.common.store.health.UserHealthModelStore +import com.twitter.frigate.thriftscala.TweetHealthScores +import com.twitter.frigate.thriftscala.UserAgathaScores +import com.twitter.hermit.store.common.DeciderableReadableStore +import com.twitter.hermit.store.common.ObservedCachedReadableStore +import com.twitter.hermit.store.common.ObservedMemcachedReadableStore +import com.twitter.hermit.store.common.ObservedReadableStore +import com.twitter.inject.TwitterModule +import com.twitter.simclusters_v2.common.TweetId +import com.twitter.simclusters_v2.common.UserId +import com.twitter.storehaus.ReadableStore +import com.twitter.strato.client.{Client => StratoClient} +import com.twitter.contentrecommender.store.TweetInfoStore +import com.twitter.contentrecommender.store.TweetyPieFieldsStore +import com.twitter.cr_mixer.model.ModuleNames +import com.twitter.cr_mixer.param.decider.CrMixerDecider +import com.twitter.cr_mixer.param.decider.DeciderKey +import com.twitter.frigate.data_pipeline.scalding.thriftscala.BlueVerifiedAnnotationsV2 +import com.twitter.recos.user_tweet_graph_plus.thriftscala.UserTweetGraphPlus +import com.twitter.recos.user_tweet_graph_plus.thriftscala.TweetEngagementScores +import com.twitter.relevance_platform.common.health_store.UserMediaRepresentationHealthStore +import com.twitter.relevance_platform.common.health_store.MagicRecsRealTimeAggregatesStore +import com.twitter.relevance_platform.thriftscala.MagicRecsRealTimeAggregatesScores +import com.twitter.relevance_platform.thriftscala.UserMediaRepresentationScores +import com.twitter.storage.client.manhattan.kv.ManhattanKVClientMtlsParams +import com.twitter.tweetypie.thriftscala.TweetService +import com.twitter.util.Future +import com.twitter.util.JavaTimer +import com.twitter.util.Timer + +import javax.inject.Named + +object TweetInfoStoreModule extends TwitterModule { + implicit val timer: Timer = new JavaTimer(true) + override def modules: Seq[Module] = Seq(UnifiedCacheClient) + + @Provides + @Singleton + def providesTweetInfoStore( + statsReceiver: StatsReceiver, + serviceIdentifier: ServiceIdentifier, + stratoClient: StratoClient, + @Named(ModuleNames.UnifiedCache) crMixerUnifiedCacheClient: MemcachedClient, + manhattanKVClientMtlsParams: ManhattanKVClientMtlsParams, + tweetyPieService: TweetService.MethodPerEndpoint, + userTweetGraphPlusService: UserTweetGraphPlus.MethodPerEndpoint, + @Named(ModuleNames.BlueVerifiedAnnotationStore) blueVerifiedAnnotationStore: ReadableStore[ + String, + BlueVerifiedAnnotationsV2 + ], + decider: CrMixerDecider + ): ReadableStore[TweetId, TweetInfo] = { + + val tweetEngagementScoreStore: ReadableStore[TweetId, TweetEngagementScores] = { + val underlyingStore = + ObservedReadableStore(new ReadableStore[TweetId, TweetEngagementScores] { + override def get( + k: TweetId + ): Future[Option[TweetEngagementScores]] = { + userTweetGraphPlusService.tweetEngagementScore(k).map { + Some(_) + } + } + })(statsReceiver.scope("UserTweetGraphTweetEngagementScoreStore")) + + DeciderableReadableStore( + underlyingStore, + decider.deciderGateBuilder.idGate( + DeciderKey.enableUtgRealTimeTweetEngagementScoreDeciderKey), + statsReceiver.scope("UserTweetGraphTweetEngagementScoreStore") + ) + + } + + val tweetHealthModelStore: ReadableStore[TweetId, TweetHealthScores] = { + val underlyingStore = TweetHealthModelStore.buildReadableStore( + stratoClient, + Some( + TweetHealthModelStoreConfig( + enablePBlock = true, + enableToxicity = true, + enablePSpammy = true, + enablePReported = true, + enableSpammyTweetContent = true, + enablePNegMultimodal = true, + )) + )(statsReceiver.scope("UnderlyingTweetHealthModelStore")) + + DeciderableReadableStore( + ObservedMemcachedReadableStore.fromCacheClient( + backingStore = underlyingStore, + cacheClient = crMixerUnifiedCacheClient, + ttl = 2.hours + )( + valueInjection = BinaryScalaCodec(TweetHealthScores), + statsReceiver = statsReceiver.scope("memCachedTweetHealthModelStore"), + keyToString = { k: TweetId => s"tHMS/$k" } + ), + decider.deciderGateBuilder.idGate(DeciderKey.enableHealthSignalsScoreDeciderKey), + statsReceiver.scope("TweetHealthModelStore") + ) // use s"tHMS/$k" instead of s"tweetHealthModelStore/$k" to differentiate from CR cache + } + + val userHealthModelStore: ReadableStore[UserId, UserAgathaScores] = { + val underlyingStore = UserHealthModelStore.buildReadableStore(stratoClient)( + statsReceiver.scope("UnderlyingUserHealthModelStore")) + DeciderableReadableStore( + ObservedMemcachedReadableStore.fromCacheClient( + backingStore = underlyingStore, + cacheClient = crMixerUnifiedCacheClient, + ttl = 18.hours + )( + valueInjection = BinaryScalaCodec(UserAgathaScores), + statsReceiver = statsReceiver.scope("memCachedUserHealthModelStore"), + keyToString = { k: UserId => s"uHMS/$k" } + ), + decider.deciderGateBuilder.idGate(DeciderKey.enableUserAgathaScoreDeciderKey), + statsReceiver.scope("UserHealthModelStore") + ) + } + + val userMediaRepresentationHealthStore: ReadableStore[UserId, UserMediaRepresentationScores] = { + val underlyingStore = + UserMediaRepresentationHealthStore.buildReadableStore( + manhattanKVClientMtlsParams, + statsReceiver.scope("UnderlyingUserMediaRepresentationHealthStore") + ) + DeciderableReadableStore( + ObservedMemcachedReadableStore.fromCacheClient( + backingStore = underlyingStore, + cacheClient = crMixerUnifiedCacheClient, + ttl = 12.hours + )( + valueInjection = BinaryScalaCodec(UserMediaRepresentationScores), + statsReceiver = statsReceiver.scope("memCacheUserMediaRepresentationHealthStore"), + keyToString = { k: UserId => s"uMRHS/$k" } + ), + decider.deciderGateBuilder.idGate(DeciderKey.enableUserMediaRepresentationStoreDeciderKey), + statsReceiver.scope("UserMediaRepresentationHealthStore") + ) + } + + val magicRecsRealTimeAggregatesStore: ReadableStore[ + TweetId, + MagicRecsRealTimeAggregatesScores + ] = { + val underlyingStore = + MagicRecsRealTimeAggregatesStore.buildReadableStore( + serviceIdentifier, + statsReceiver.scope("UnderlyingMagicRecsRealTimeAggregatesScores") + ) + DeciderableReadableStore( + underlyingStore, + decider.deciderGateBuilder.idGate(DeciderKey.enableMagicRecsRealTimeAggregatesStore), + statsReceiver.scope("MagicRecsRealTimeAggregatesStore") + ) + } + + val tweetInfoStore: ReadableStore[TweetId, TweetInfo] = { + val underlyingStore = TweetInfoStore( + TweetyPieFieldsStore.getStoreFromTweetyPie(tweetyPieService), + userMediaRepresentationHealthStore, + magicRecsRealTimeAggregatesStore, + tweetEngagementScoreStore, + blueVerifiedAnnotationStore + )(statsReceiver.scope("tweetInfoStore")) + + val memcachedStore = ObservedMemcachedReadableStore.fromCacheClient( + backingStore = underlyingStore, + cacheClient = crMixerUnifiedCacheClient, + ttl = 15.minutes, + // Hydrating tweetInfo is now a required step for all candidates, + // hence we needed to tune these thresholds. + asyncUpdate = serviceIdentifier.environment == "prod" + )( + valueInjection = BinaryScalaCodec(TweetInfo), + statsReceiver = statsReceiver.scope("memCachedTweetInfoStore"), + keyToString = { k: TweetId => s"tIS/$k" } + ) + + ObservedCachedReadableStore.from( + memcachedStore, + ttl = 15.minutes, + maxKeys = 8388607, // Check TweetInfo definition. size~92b. Around 736 MB + windowSize = 10000L, + cacheName = "tweet_info_cache", + maxMultiGetSize = 20 + )(statsReceiver.scope("inMemoryCachedTweetInfoStore")) + } + tweetInfoStore + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/TweetRecentEngagedUserStoreModule.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/TweetRecentEngagedUserStoreModule.scala new file mode 100644 index 0000000000..2e379e545f --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/TweetRecentEngagedUserStoreModule.scala @@ -0,0 +1,42 @@ +package com.twitter.cr_mixer.module + +import com.google.inject.Provides +import com.google.inject.Singleton +import com.twitter.app.Flag +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.frigate.common.store.strato.StratoFetchableStore +import com.twitter.hermit.store.common.ObservedReadableStore +import com.twitter.inject.TwitterModule +import com.twitter.simclusters_v2.common.TweetId +import com.twitter.storehaus.ReadableStore +import com.twitter.strato.client.{Client => StratoClient} +import com.twitter.twistly.thriftscala.TweetRecentEngagedUsers + +object TweetRecentEngagedUserStoreModule extends TwitterModule { + + private val tweetRecentEngagedUsersStoreDefaultVersion = + 0 // DefaultVersion for tweetEngagedUsersStore, whose key = (tweetId, DefaultVersion) + private val tweetRecentEngagedUsersColumnPath: Flag[String] = flag[String]( + name = "crMixer.tweetRecentEngagedUsersColumnPath", + default = "recommendations/twistly/tweetRecentEngagedUsers", + help = "Strato column path for TweetRecentEngagedUsersStore" + ) + private type Version = Long + + @Provides + @Singleton + def providesTweetRecentEngagedUserStore( + statsReceiver: StatsReceiver, + stratoClient: StratoClient, + ): ReadableStore[TweetId, TweetRecentEngagedUsers] = { + val tweetRecentEngagedUsersStratoFetchableStore = StratoFetchableStore + .withUnitView[(TweetId, Version), TweetRecentEngagedUsers]( + stratoClient, + tweetRecentEngagedUsersColumnPath()).composeKeyMapping[TweetId](tweetId => + (tweetId, tweetRecentEngagedUsersStoreDefaultVersion)) + + ObservedReadableStore( + tweetRecentEngagedUsersStratoFetchableStore + )(statsReceiver.scope("tweet_recent_engaged_users_store")) + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/TweetRecommendationResultsStoreModule.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/TweetRecommendationResultsStoreModule.scala new file mode 100644 index 0000000000..04c03eda68 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/TweetRecommendationResultsStoreModule.scala @@ -0,0 +1,32 @@ +package com.twitter.cr_mixer.module + +import com.google.inject.Provides +import com.google.inject.Singleton +import com.twitter.bijection.scrooge.BinaryScalaCodec +import com.twitter.conversions.DurationOps._ +import com.twitter.cr_mixer.model.ModuleNames +import com.twitter.cr_mixer.thriftscala.CrMixerTweetResponse +import com.twitter.finagle.memcached.{Client => MemcachedClient} +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.inject.TwitterModule +import com.twitter.hermit.store.common.ReadableWritableStore +import com.twitter.hermit.store.common.ObservedReadableWritableMemcacheStore +import com.twitter.simclusters_v2.common.UserId +import javax.inject.Named + +object TweetRecommendationResultsStoreModule extends TwitterModule { + @Provides + @Singleton + def providesTweetRecommendationResultsStore( + @Named(ModuleNames.TweetRecommendationResultsCache) tweetRecommendationResultsCacheClient: MemcachedClient, + statsReceiver: StatsReceiver + ): ReadableWritableStore[UserId, CrMixerTweetResponse] = { + ObservedReadableWritableMemcacheStore.fromCacheClient( + cacheClient = tweetRecommendationResultsCacheClient, + ttl = 24.hours)( + valueInjection = BinaryScalaCodec(CrMixerTweetResponse), + statsReceiver = statsReceiver.scope("TweetRecommendationResultsMemcacheStore"), + keyToString = { k: UserId => k.toString } + ) + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/TwhinCollabFilterStratoStoreModule.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/TwhinCollabFilterStratoStoreModule.scala new file mode 100644 index 0000000000..4275ad2a81 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/TwhinCollabFilterStratoStoreModule.scala @@ -0,0 +1,67 @@ +package com.twitter.cr_mixer.module + +import com.google.inject.Provides +import com.google.inject.Singleton +import com.twitter.inject.TwitterModule +import com.twitter.cr_mixer.model.ModuleNames +import com.twitter.frigate.common.store.strato.StratoFetchableStore +import com.twitter.cr_mixer.similarity_engine.TwhinCollabFilterSimilarityEngine.TwhinCollabFilterView +import com.twitter.strato.client.{Client => StratoClient} +import com.twitter.simclusters_v2.common.TweetId +import com.twitter.storehaus.ReadableStore +import javax.inject.Named + +object TwhinCollabFilterStratoStoreModule extends TwitterModule { + + val stratoColumnPath: String = "cuad/twhin/getCollabFilterTweetCandidatesProd.User" + + @Provides + @Singleton + @Named(ModuleNames.TwhinCollabFilterStratoStoreForFollow) + def providesTwhinCollabFilterStratoStoreForFollow( + stratoClient: StratoClient + ): ReadableStore[Long, Seq[TweetId]] = { + StratoFetchableStore.withView[Long, TwhinCollabFilterView, Seq[TweetId]]( + stratoClient, + column = stratoColumnPath, + view = TwhinCollabFilterView("follow_2022_03_10_c_500K") + ) + } + + @Provides + @Singleton + @Named(ModuleNames.TwhinCollabFilterStratoStoreForEngagement) + def providesTwhinCollabFilterStratoStoreForEngagement( + stratoClient: StratoClient + ): ReadableStore[Long, Seq[TweetId]] = { + StratoFetchableStore.withView[Long, TwhinCollabFilterView, Seq[TweetId]]( + stratoClient, + column = stratoColumnPath, + view = TwhinCollabFilterView("engagement_2022_04_10_c_500K")) + } + + @Provides + @Singleton + @Named(ModuleNames.TwhinMultiClusterStratoStoreForFollow) + def providesTwhinMultiClusterStratoStoreForFollow( + stratoClient: StratoClient + ): ReadableStore[Long, Seq[TweetId]] = { + StratoFetchableStore.withView[Long, TwhinCollabFilterView, Seq[TweetId]]( + stratoClient, + column = stratoColumnPath, + view = TwhinCollabFilterView("multiclusterFollow20220921") + ) + } + + @Provides + @Singleton + @Named(ModuleNames.TwhinMultiClusterStratoStoreForEngagement) + def providesTwhinMultiClusterStratoStoreForEngagement( + stratoClient: StratoClient + ): ReadableStore[Long, Seq[TweetId]] = { + StratoFetchableStore.withView[Long, TwhinCollabFilterView, Seq[TweetId]]( + stratoClient, + column = stratoColumnPath, + view = TwhinCollabFilterView("multiclusterEng20220921")) + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/TwiceClustersMembersStoreModule.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/TwiceClustersMembersStoreModule.scala new file mode 100644 index 0000000000..a15e2549a7 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/TwiceClustersMembersStoreModule.scala @@ -0,0 +1,42 @@ +package com.twitter.cr_mixer.module + +import com.google.inject.Provides +import com.google.inject.Singleton +import com.twitter.app.Flag +import com.twitter.cr_mixer.model.ModuleNames +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.frigate.common.store.strato.StratoFetchableStore +import com.twitter.hermit.store.common.ObservedReadableStore +import com.twitter.inject.TwitterModule +import com.twitter.simclusters_v2.common.UserId +import com.twitter.storehaus.ReadableStore +import com.twitter.strato.client.{Client => StratoClient} +import com.twitter.simclusters_v2.thriftscala.OrderedClustersAndMembers +import javax.inject.Named + +object TwiceClustersMembersStoreModule extends TwitterModule { + + private val twiceClustersMembersColumnPath: Flag[String] = flag[String]( + name = "crMixer.twiceClustersMembersColumnPath", + default = + "recommendations/simclusters_v2/embeddings/TwiceClustersMembersLargestDimApeSimilarity", + help = "Strato column path for TweetRecentEngagedUsersStore" + ) + + @Provides + @Singleton + @Named(ModuleNames.TwiceClustersMembersStore) + def providesTweetRecentEngagedUserStore( + statsReceiver: StatsReceiver, + stratoClient: StratoClient, + ): ReadableStore[UserId, OrderedClustersAndMembers] = { + val twiceClustersMembersStratoFetchableStore = StratoFetchableStore + .withUnitView[UserId, OrderedClustersAndMembers]( + stratoClient, + twiceClustersMembersColumnPath()) + + ObservedReadableStore( + twiceClustersMembersStratoFetchableStore + )(statsReceiver.scope("twice_clusters_members_largestDimApe_similarity_store")) + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/UnifiedCacheClient.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/UnifiedCacheClient.scala new file mode 100644 index 0000000000..3b48f4c027 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/UnifiedCacheClient.scala @@ -0,0 +1,83 @@ +package com.twitter.cr_mixer.module + +import com.google.inject.Provides +import com.google.inject.Singleton +import com.twitter.app.Flag +import com.twitter.conversions.DurationOps._ +import com.twitter.cr_mixer.model.ModuleNames +import com.twitter.finagle.memcached.Client +import com.twitter.finagle.mtls.authentication.ServiceIdentifier +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.inject.TwitterModule +import com.twitter.storehaus_internal.memcache.MemcacheStore +import com.twitter.storehaus_internal.util.ClientName +import com.twitter.storehaus_internal.util.ZkEndPoint +import javax.inject.Named + +object UnifiedCacheClient extends TwitterModule { + + private val TIME_OUT = 20.milliseconds + + val crMixerUnifiedCacheDest: Flag[String] = flag[String]( + name = "crMixer.unifiedCacheDest", + default = "/s/cache/content_recommender_unified_v2", + help = "Wily path to Content Recommender unified cache" + ) + + val tweetRecommendationResultsCacheDest: Flag[String] = flag[String]( + name = "tweetRecommendationResults.CacheDest", + default = "/s/cache/tweet_recommendation_results", + help = "Wily path to CrMixer getTweetRecommendations() results cache" + ) + + val earlybirdTweetsCacheDest: Flag[String] = flag[String]( + name = "earlybirdTweets.CacheDest", + default = "/s/cache/crmixer_earlybird_tweets", + help = "Wily path to CrMixer Earlybird Recency Based Similarity Engine result cache" + ) + + @Provides + @Singleton + @Named(ModuleNames.UnifiedCache) + def provideUnifiedCacheClient( + serviceIdentifier: ServiceIdentifier, + statsReceiver: StatsReceiver, + ): Client = + MemcacheStore.memcachedClient( + name = ClientName("memcache-content-recommender-unified"), + dest = ZkEndPoint(crMixerUnifiedCacheDest()), + statsReceiver = statsReceiver.scope("cache_client"), + serviceIdentifier = serviceIdentifier, + timeout = TIME_OUT + ) + + @Provides + @Singleton + @Named(ModuleNames.TweetRecommendationResultsCache) + def providesTweetRecommendationResultsCache( + serviceIdentifier: ServiceIdentifier, + statsReceiver: StatsReceiver, + ): Client = + MemcacheStore.memcachedClient( + name = ClientName("memcache-tweet-recommendation-results"), + dest = ZkEndPoint(tweetRecommendationResultsCacheDest()), + statsReceiver = statsReceiver.scope("cache_client"), + serviceIdentifier = serviceIdentifier, + timeout = TIME_OUT + ) + + @Provides + @Singleton + @Named(ModuleNames.EarlybirdTweetsCache) + def providesEarlybirdTweetsCache( + serviceIdentifier: ServiceIdentifier, + statsReceiver: StatsReceiver, + ): Client = + MemcacheStore.memcachedClient( + name = ClientName("memcache-crmixer-earlybird-tweets"), + dest = ZkEndPoint(earlybirdTweetsCacheDest()), + statsReceiver = statsReceiver.scope("cache_client"), + serviceIdentifier = serviceIdentifier, + timeout = TIME_OUT + ) +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/UserSignalServiceColumnModule.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/UserSignalServiceColumnModule.scala new file mode 100644 index 0000000000..b15ebe0feb --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/UserSignalServiceColumnModule.scala @@ -0,0 +1,30 @@ +package com.twitter.cr_mixer.module +import com.google.inject.Provides +import com.google.inject.Singleton +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.inject.TwitterModule +import com.twitter.storehaus.ReadableStore +import com.twitter.strato.client.{Client => StratoClient} +import com.twitter.cr_mixer.model.ModuleNames +import com.twitter.frigate.common.store.strato.StratoFetchableStore +import com.twitter.hermit.store.common.ObservedReadableStore +import com.twitter.usersignalservice.thriftscala.BatchSignalRequest +import com.twitter.usersignalservice.thriftscala.BatchSignalResponse +import javax.inject.Named + +object UserSignalServiceColumnModule extends TwitterModule { + private val UssColumnPath = "recommendations/user-signal-service/signals" + + @Provides + @Singleton + @Named(ModuleNames.UssStratoColumn) + def providesUserSignalServiceStore( + statsReceiver: StatsReceiver, + stratoClient: StratoClient, + ): ReadableStore[BatchSignalRequest, BatchSignalResponse] = { + ObservedReadableStore( + StratoFetchableStore + .withUnitView[BatchSignalRequest, BatchSignalResponse](stratoClient, UssColumnPath))( + statsReceiver.scope("user_signal_service_store")) + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/UserSignalServiceStoreModule.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/UserSignalServiceStoreModule.scala new file mode 100644 index 0000000000..cc55f0e9a3 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/UserSignalServiceStoreModule.scala @@ -0,0 +1,37 @@ +package com.twitter.cr_mixer.module + +import com.google.inject.Provides +import com.google.inject.Singleton +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.inject.TwitterModule +import com.twitter.storehaus.ReadableStore +import com.twitter.strato.client.{Client => StratoClient} +import com.twitter.cr_mixer.model.ModuleNames +import com.twitter.cr_mixer.source_signal.UssStore +import com.twitter.cr_mixer.source_signal.UssStore.Query +import com.twitter.frigate.common.store.strato.StratoFetchableStore +import com.twitter.hermit.store.common.ObservedReadableStore +import com.twitter.usersignalservice.thriftscala.BatchSignalRequest +import com.twitter.usersignalservice.thriftscala.BatchSignalResponse +import com.twitter.usersignalservice.thriftscala.SignalType +import com.twitter.usersignalservice.thriftscala.{Signal => UssSignal} +import javax.inject.Named + +object UserSignalServiceStoreModule extends TwitterModule { + + private val UssColumnPath = "recommendations/user-signal-service/signals" + + @Provides + @Singleton + @Named(ModuleNames.UssStore) + def providesUserSignalServiceStore( + statsReceiver: StatsReceiver, + stratoClient: StratoClient, + ): ReadableStore[Query, Seq[(SignalType, Seq[UssSignal])]] = { + ObservedReadableStore( + UssStore( + StratoFetchableStore + .withUnitView[BatchSignalRequest, BatchSignalResponse](stratoClient, UssColumnPath), + statsReceiver))(statsReceiver.scope("user_signal_service_store")) + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/UserStateStoreModule.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/UserStateStoreModule.scala new file mode 100644 index 0000000000..6db2c38fd1 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/UserStateStoreModule.scala @@ -0,0 +1,113 @@ +package com.twitter.cr_mixer.module + +import com.google.inject.Provides +import com.google.inject.Singleton +import com.twitter.bijection.Bufferable +import com.twitter.bijection.Injection +import com.twitter.bijection.scrooge.BinaryScalaCodec +import com.twitter.cr_mixer.model.ModuleNames +import com.twitter.conversions.DurationOps._ +import com.twitter.finagle.memcached.{Client => MemcachedClient} +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.hermit.store.common.ObservedMemcachedReadableStore +import com.twitter.inject.TwitterModule +import com.twitter.simclusters_v2.common.UserId +import com.twitter.snowflake.id.SnowflakeId +import com.twitter.storage.client.manhattan.kv.ManhattanKVClientMtlsParams +import com.twitter.storehaus.ReadableStore +import com.twitter.storehaus_internal.manhattan.ManhattanRO +import com.twitter.storehaus_internal.manhattan.ManhattanROConfig +import com.twitter.storehaus_internal.util.HDFSPath +import com.twitter.core_workflows.user_model.thriftscala.UserState +import com.twitter.core_workflows.user_model.thriftscala.CondensedUserState +import com.twitter.cr_mixer.config.TimeoutConfig +import com.twitter.cr_mixer.param.decider.CrMixerDecider +import com.twitter.cr_mixer.param.decider.DeciderKey +import com.twitter.hermit.store.common.DeciderableReadableStore +import com.twitter.storehaus_internal.manhattan.Apollo +import com.twitter.storehaus_internal.util.ApplicationID +import com.twitter.storehaus_internal.util.DatasetName +import com.twitter.util.Duration +import com.twitter.util.Future +import com.twitter.util.JavaTimer +import com.twitter.util.Time +import com.twitter.util.TimeoutException +import com.twitter.util.Timer +import javax.inject.Named + +object UserStateStoreModule extends TwitterModule { + implicit val timer: Timer = new JavaTimer(true) + final val NewUserCreateDaysThreshold = 7 + final val DefaultUnknownUserStateValue = 100 + + // Convert CondensedUserState to UserState Enum + // If CondensedUserState is None, back fill by checking whether the user is new user + class UserStateStore( + userStateStore: ReadableStore[UserId, CondensedUserState], + timeout: Duration, + statsReceiver: StatsReceiver) + extends ReadableStore[UserId, UserState] { + override def get(userId: UserId): Future[Option[UserState]] = { + userStateStore + .get(userId).map(_.flatMap(_.userState)).map { + case Some(userState) => Some(userState) + case None => + val isNewUser = SnowflakeId.timeFromIdOpt(userId).exists { userCreateTime => + Time.now - userCreateTime < Duration.fromDays(NewUserCreateDaysThreshold) + } + if (isNewUser) Some(UserState.New) + else Some(UserState.EnumUnknownUserState(DefaultUnknownUserStateValue)) + + }.raiseWithin(timeout)(timer).rescue { + case _: TimeoutException => + statsReceiver.counter("TimeoutException").incr() + Future.None + } + } + } + + @Provides + @Singleton + def providesUserStateStore( + crMixerDecider: CrMixerDecider, + statsReceiver: StatsReceiver, + manhattanKVClientMtlsParams: ManhattanKVClientMtlsParams, + @Named(ModuleNames.UnifiedCache) crMixerUnifiedCacheClient: MemcachedClient, + timeoutConfig: TimeoutConfig + ): ReadableStore[UserId, UserState] = { + + val underlyingStore = new UserStateStore( + ManhattanRO + .getReadableStoreWithMtls[UserId, CondensedUserState]( + ManhattanROConfig( + HDFSPath(""), + ApplicationID("cr_mixer_apollo"), + DatasetName("condensed_user_state"), + Apollo), + manhattanKVClientMtlsParams + )( + implicitly[Injection[Long, Array[Byte]]], + BinaryScalaCodec(CondensedUserState) + ), + timeoutConfig.userStateStoreTimeout, + statsReceiver.scope("UserStateStore") + ).mapValues(_.value) // Read the value of Enum so that we only caches the Int + + val memCachedStore = ObservedMemcachedReadableStore + .fromCacheClient( + backingStore = underlyingStore, + cacheClient = crMixerUnifiedCacheClient, + ttl = 24.hours, + )( + valueInjection = Bufferable.injectionOf[Int], // Cache Value is Enum Value for UserState + statsReceiver = statsReceiver.scope("memCachedUserStateStore"), + keyToString = { k: UserId => s"uState/$k" } + ).mapValues(value => UserState.getOrUnknown(value)) + + DeciderableReadableStore( + memCachedStore, + crMixerDecider.deciderGateBuilder.idGate(DeciderKey.enableUserStateStoreDeciderKey), + statsReceiver.scope("UserStateStore") + ) + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/core/ABDeciderModule.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/core/ABDeciderModule.scala new file mode 100644 index 0000000000..9d981f4f35 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/core/ABDeciderModule.scala @@ -0,0 +1,33 @@ +package com.twitter.cr_mixer.module.core + +import com.google.inject.Provides +import com.google.inject.name.Named +import com.twitter.abdecider.ABDeciderFactory +import com.twitter.abdecider.LoggingABDecider +import com.twitter.cr_mixer.model.ModuleNames +import com.twitter.inject.TwitterModule +import com.twitter.inject.annotations.Flag +import com.twitter.logging.Logger +import javax.inject.Singleton + +object ABDeciderModule extends TwitterModule { + + flag( + name = "abdecider.path", + default = "/usr/local/config/abdecider/abdecider.yml", + help = "path to the abdecider Yml file location" + ) + + @Provides + @Singleton + def provideABDecider( + @Flag("abdecider.path") abDeciderYmlPath: String, + @Named(ModuleNames.AbDeciderLogger) scribeLogger: Logger + ): LoggingABDecider = { + ABDeciderFactory( + abDeciderYmlPath = abDeciderYmlPath, + scribeLogger = Some(scribeLogger), + environment = Some("production") + ).buildWithLogging() + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/core/CrMixerFlagModule.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/core/CrMixerFlagModule.scala new file mode 100644 index 0000000000..9e7b9938a8 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/core/CrMixerFlagModule.scala @@ -0,0 +1,20 @@ +package com.twitter.cr_mixer.module.core + +import com.twitter.inject.TwitterModule + +object CrMixerFlagName { + val SERVICE_FLAG = "cr_mixer.flag" + val DarkTrafficFilterDeciderKey = "thrift.dark.traffic.filter.decider_key" +} + +object CrMixerFlagModule extends TwitterModule { + import CrMixerFlagName._ + + flag[Boolean](name = SERVICE_FLAG, default = false, help = "This is a CR Mixer flag") + + flag[String]( + name = DarkTrafficFilterDeciderKey, + default = "dark_traffic_filter", + help = "Dark traffic filter decider key" + ) +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/core/CrMixerLoggingABDeciderModule.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/core/CrMixerLoggingABDeciderModule.scala new file mode 100644 index 0000000000..6b674495f1 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/core/CrMixerLoggingABDeciderModule.scala @@ -0,0 +1,20 @@ +package com.twitter.cr_mixer.module.core + +import com.google.inject.Provides +import com.twitter.abdecider.LoggingABDecider +import com.twitter.cr_mixer.featureswitch.CrMixerLoggingABDecider +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.inject.TwitterModule +import javax.inject.Singleton + +object CrMixerLoggingABDeciderModule extends TwitterModule { + + @Provides + @Singleton + def provideABDecider( + loggingABDecider: LoggingABDecider, + statsReceiver: StatsReceiver + ): CrMixerLoggingABDecider = { + CrMixerLoggingABDecider(loggingABDecider, statsReceiver) + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/core/FeatureContextBuilderModule.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/core/FeatureContextBuilderModule.scala new file mode 100644 index 0000000000..18d262c545 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/core/FeatureContextBuilderModule.scala @@ -0,0 +1,16 @@ +package com.twitter.cr_mixer.module.core + +import com.google.inject.Provides +import com.twitter.discovery.common.configapi.FeatureContextBuilder +import com.twitter.featureswitches.v2.FeatureSwitches +import com.twitter.inject.TwitterModule +import javax.inject.Singleton + +object FeatureContextBuilderModule extends TwitterModule { + + @Provides + @Singleton + def providesFeatureContextBuilder(featureSwitches: FeatureSwitches): FeatureContextBuilder = { + FeatureContextBuilder(featureSwitches) + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/core/FeatureSwitchesModule.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/core/FeatureSwitchesModule.scala new file mode 100644 index 0000000000..a87d1f54bd --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/core/FeatureSwitchesModule.scala @@ -0,0 +1,74 @@ +package com.twitter.cr_mixer.module.core + +import com.google.inject.Provides +import com.twitter.cr_mixer.featureswitch.CrMixerLoggingABDecider +import com.twitter.featureswitches.v2.FeatureSwitches +import com.twitter.featureswitches.v2.builder.FeatureSwitchesBuilder +import com.twitter.featureswitches.v2.experimentation.NullBucketImpressor +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.inject.TwitterModule +import com.twitter.inject.annotations.Flag +import com.twitter.util.Duration +import javax.inject.Singleton + +object FeatureSwitchesModule extends TwitterModule { + + flag( + name = "featureswitches.path", + default = "/features/cr-mixer/main", + help = "path to the featureswitch configuration directory" + ) + flag( + "use_config_repo_mirror.bool", + false, + "If true, read config from a different directory, to facilitate testing.") + + val DefaultFastRefresh: Boolean = false + val AddServiceDetailsFromAurora: Boolean = true + val ImpressExperiments: Boolean = true + + @Provides + @Singleton + def providesFeatureSwitches( + @Flag("featureswitches.path") featureSwitchDirectory: String, + @Flag("use_config_repo_mirror.bool") useConfigRepoMirrorFlag: Boolean, + abDecider: CrMixerLoggingABDecider, + statsReceiver: StatsReceiver + ): FeatureSwitches = { + val configRepoAbsPath = + getConfigRepoAbsPath(useConfigRepoMirrorFlag) + val fastRefresh = + shouldFastRefresh(useConfigRepoMirrorFlag) + + val featureSwitches = FeatureSwitchesBuilder() + .abDecider(abDecider) + .statsReceiver(statsReceiver.scope("featureswitches-v2")) + .configRepoAbsPath(configRepoAbsPath) + .featuresDirectory(featureSwitchDirectory) + .limitToReferencedExperiments(shouldLimit = true) + .experimentImpressionStatsEnabled(true) + + if (!ImpressExperiments) featureSwitches.experimentBucketImpressor(NullBucketImpressor) + if (AddServiceDetailsFromAurora) featureSwitches.serviceDetailsFromAurora() + if (fastRefresh) featureSwitches.refreshPeriod(Duration.fromSeconds(10)) + + featureSwitches.build() + } + + private def getConfigRepoAbsPath( + useConfigRepoMirrorFlag: Boolean + ): String = { + if (useConfigRepoMirrorFlag) + "config_repo_mirror/" + else "/usr/local/config" + } + + private def shouldFastRefresh( + useConfigRepoMirrorFlag: Boolean + ): Boolean = { + if (useConfigRepoMirrorFlag) + true + else DefaultFastRefresh + } + +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/core/KafkaProducerModule.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/core/KafkaProducerModule.scala new file mode 100644 index 0000000000..770ad1e7e4 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/core/KafkaProducerModule.scala @@ -0,0 +1,70 @@ +package com.twitter.cr_mixer.module.core + +import com.google.inject.Provides +import com.twitter.cr_mixer.thriftscala.GetTweetsRecommendationsScribe +import com.twitter.finagle.mtls.authentication.ServiceIdentifier +import com.twitter.finatra.kafka.producers.FinagleKafkaProducerBuilder +import com.twitter.finatra.kafka.producers.KafkaProducerBase +import com.twitter.finatra.kafka.producers.NullKafkaProducer +import com.twitter.finatra.kafka.serde.ScalaSerdes +import com.twitter.inject.TwitterModule +import javax.inject.Singleton +import org.apache.kafka.clients.CommonClientConfigs +import org.apache.kafka.common.config.SaslConfigs +import org.apache.kafka.common.config.SslConfigs +import org.apache.kafka.common.record.CompressionType +import org.apache.kafka.common.security.auth.SecurityProtocol +import org.apache.kafka.common.serialization.Serdes + +object KafkaProducerModule extends TwitterModule { + + @Provides + @Singleton + def provideTweetRecsLoggerFactory( + serviceIdentifier: ServiceIdentifier, + ): KafkaProducerBase[String, GetTweetsRecommendationsScribe] = { + KafkaProducerFactory.getKafkaProducer(serviceIdentifier.environment) + } +} + +object KafkaProducerFactory { + private val jaasConfig = + """com.sun.security.auth.module.Krb5LoginModule + |required + |principal="cr-mixer@TWITTER.BIZ" + |debug=true + |useKeyTab=true + |storeKey=true + |keyTab="/var/lib/tss/keys/fluffy/keytabs/client/cr-mixer.keytab" + |doNotPrompt=true; + """.stripMargin.replaceAll("\n", " ") + + private val trustStoreLocation = "/etc/tw_truststore/messaging/kafka/client.truststore.jks" + + def getKafkaProducer( + environment: String + ): KafkaProducerBase[String, GetTweetsRecommendationsScribe] = { + if (environment == "prod") { + FinagleKafkaProducerBuilder() + .dest("/s/kafka/recommendations:kafka-tls") + // kerberos params + .withConfig(SaslConfigs.SASL_JAAS_CONFIG, jaasConfig) + .withConfig( + CommonClientConfigs.SECURITY_PROTOCOL_CONFIG, + SecurityProtocol.SASL_SSL.toString) + .withConfig(SslConfigs.SSL_TRUSTSTORE_LOCATION_CONFIG, trustStoreLocation) + .withConfig(SaslConfigs.SASL_MECHANISM, SaslConfigs.GSSAPI_MECHANISM) + .withConfig(SaslConfigs.SASL_KERBEROS_SERVICE_NAME, "kafka") + .withConfig(SaslConfigs.SASL_KERBEROS_SERVER_NAME, "kafka") + // Kafka params + .keySerializer(Serdes.String.serializer) + .valueSerializer(ScalaSerdes.CompactThrift[GetTweetsRecommendationsScribe].serializer()) + .clientId("cr-mixer") + .enableIdempotence(true) + .compressionType(CompressionType.LZ4) + .build() + } else { + new NullKafkaProducer[String, GetTweetsRecommendationsScribe] + } + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/core/LoggerFactoryModule.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/core/LoggerFactoryModule.scala new file mode 100644 index 0000000000..877ed4bb25 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/core/LoggerFactoryModule.scala @@ -0,0 +1,155 @@ +package com.twitter.cr_mixer.module.core + +import com.google.inject.Provides +import com.twitter.cr_mixer.model.ModuleNames +import com.twitter.cr_mixer.scribe.ScribeCategories +import com.twitter.cr_mixer.scribe.ScribeCategory +import com.twitter.finagle.mtls.authentication.ServiceIdentifier +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.inject.TwitterModule +import com.twitter.logging.BareFormatter +import com.twitter.logging.Level +import com.twitter.logging.Logger +import com.twitter.logging.NullHandler +import com.twitter.logging.QueueingHandler +import com.twitter.logging.ScribeHandler +import com.twitter.logging.{LoggerFactory => TwitterLoggerFactory} +import javax.inject.Named +import javax.inject.Singleton + +object LoggerFactoryModule extends TwitterModule { + + private val DefaultQueueSize = 10000 + + @Provides + @Singleton + @Named(ModuleNames.AbDeciderLogger) + def provideAbDeciderLogger( + serviceIdentifier: ServiceIdentifier, + statsReceiver: StatsReceiver + ): Logger = { + buildLoggerFactory( + ScribeCategories.AbDecider, + serviceIdentifier.environment, + statsReceiver.scope("ScribeLogger")) + .apply() + } + + @Provides + @Singleton + @Named(ModuleNames.TopLevelApiDdgMetricsLogger) + def provideTopLevelApiDdgMetricsLogger( + serviceIdentifier: ServiceIdentifier, + statsReceiver: StatsReceiver + ): Logger = { + buildLoggerFactory( + ScribeCategories.TopLevelApiDdgMetrics, + serviceIdentifier.environment, + statsReceiver.scope("ScribeLogger")) + .apply() + } + + @Provides + @Singleton + @Named(ModuleNames.TweetRecsLogger) + def provideTweetRecsLogger( + serviceIdentifier: ServiceIdentifier, + statsReceiver: StatsReceiver + ): Logger = { + buildLoggerFactory( + ScribeCategories.TweetsRecs, + serviceIdentifier.environment, + statsReceiver.scope("ScribeLogger")) + .apply() + } + + @Provides + @Singleton + @Named(ModuleNames.BlueVerifiedTweetRecsLogger) + def provideVITTweetRecsLogger( + serviceIdentifier: ServiceIdentifier, + statsReceiver: StatsReceiver + ): Logger = { + buildLoggerFactory( + ScribeCategories.VITTweetsRecs, + serviceIdentifier.environment, + statsReceiver.scope("ScribeLogger")) + .apply() + } + + @Provides + @Singleton + @Named(ModuleNames.RelatedTweetsLogger) + def provideRelatedTweetsLogger( + serviceIdentifier: ServiceIdentifier, + statsReceiver: StatsReceiver + ): Logger = { + buildLoggerFactory( + ScribeCategories.RelatedTweets, + serviceIdentifier.environment, + statsReceiver.scope("ScribeLogger")) + .apply() + } + + @Provides + @Singleton + @Named(ModuleNames.UtegTweetsLogger) + def provideUtegTweetsLogger( + serviceIdentifier: ServiceIdentifier, + statsReceiver: StatsReceiver + ): Logger = { + buildLoggerFactory( + ScribeCategories.UtegTweets, + serviceIdentifier.environment, + statsReceiver.scope("ScribeLogger")) + .apply() + } + + @Provides + @Singleton + @Named(ModuleNames.AdsRecommendationsLogger) + def provideAdsRecommendationsLogger( + serviceIdentifier: ServiceIdentifier, + statsReceiver: StatsReceiver + ): Logger = { + buildLoggerFactory( + ScribeCategories.AdsRecommendations, + serviceIdentifier.environment, + statsReceiver.scope("ScribeLogger")) + .apply() + } + + private def buildLoggerFactory( + category: ScribeCategory, + environment: String, + statsReceiver: StatsReceiver + ): TwitterLoggerFactory = { + environment match { + case "prod" => + TwitterLoggerFactory( + node = category.getProdLoggerFactoryNode, + level = Some(Level.INFO), + useParents = false, + handlers = List( + QueueingHandler( + maxQueueSize = DefaultQueueSize, + handler = ScribeHandler( + category = category.scribeCategory, + formatter = BareFormatter, + statsReceiver = statsReceiver.scope(category.getProdLoggerFactoryNode) + ) + ) + ) + ) + case _ => + TwitterLoggerFactory( + node = category.getStagingLoggerFactoryNode, + level = Some(Level.DEBUG), + useParents = false, + handlers = List( + { () => NullHandler } + ) + ) + } + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/core/MemoizingStatsReceiverModule.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/core/MemoizingStatsReceiverModule.scala new file mode 100644 index 0000000000..ee94cf1662 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/core/MemoizingStatsReceiverModule.scala @@ -0,0 +1,12 @@ +package com.twitter.cr_mixer.module.core + +import com.twitter.finagle.stats.LoadedStatsReceiver +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.inject.TwitterModule +import com.twitter.servo.util.MemoizingStatsReceiver + +object MemoizingStatsReceiverModule extends TwitterModule { + override def configure(): Unit = { + bind[StatsReceiver].toInstance(new MemoizingStatsReceiver(LoadedStatsReceiver)) + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/core/TimeoutConfigModule.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/core/TimeoutConfigModule.scala new file mode 100644 index 0000000000..1b62008126 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/core/TimeoutConfigModule.scala @@ -0,0 +1,104 @@ +package com.twitter.cr_mixer.module.core + +import com.twitter.inject.TwitterModule +import com.google.inject.Provides +import javax.inject.Singleton +import com.twitter.util.Duration +import com.twitter.app.Flag +import com.twitter.cr_mixer.config.TimeoutConfig + +/** + * All timeout settings in CrMixer. + * Timeout numbers are defined in source/cr-mixer/server/config/deploy.aurora + */ +object TimeoutConfigModule extends TwitterModule { + + /** + * Flag names for client timeout + * These are used in modules extending ThriftMethodBuilderClientModule + * which cannot accept injection of TimeoutConfig + */ + val EarlybirdClientTimeoutFlagName = "earlybird.client.timeout" + val FrsClientTimeoutFlagName = "frsSignalFetch.client.timeout" + val QigRankerClientTimeoutFlagName = "qigRanker.client.timeout" + val TweetypieClientTimeoutFlagName = "tweetypie.client.timeout" + val UserTweetGraphClientTimeoutFlagName = "userTweetGraph.client.timeout" + val UserTweetGraphPlusClientTimeoutFlagName = "userTweetGraphPlus.client.timeout" + val UserAdGraphClientTimeoutFlagName = "userAdGraph.client.timeout" + val UserVideoGraphClientTimeoutFlagName = "userVideoGraph.client.timeout" + val UtegClientTimeoutFlagName = "uteg.client.timeout" + val NaviRequestTimeoutFlagName = "navi.client.request.timeout" + + /** + * Flags for timeouts + * These are defined and initialized only in this file + */ + // timeout for the service + private val serviceTimeout: Flag[Duration] = + flag("service.timeout", "service total timeout") + + // timeout for signal fetch + private val signalFetchTimeout: Flag[Duration] = + flag[Duration]("signalFetch.timeout", "signal fetch timeout") + + // timeout for similarity engine + private val similarityEngineTimeout: Flag[Duration] = + flag[Duration]("similarityEngine.timeout", "similarity engine timeout") + private val annServiceClientTimeout: Flag[Duration] = + flag[Duration]("annService.client.timeout", "annQueryService client timeout") + + // timeout for user affinities fetcher + private val userStateUnderlyingStoreTimeout: Flag[Duration] = + flag[Duration]("userStateUnderlyingStore.timeout", "user state underlying store timeout") + + private val userStateStoreTimeout: Flag[Duration] = + flag[Duration]("userStateStore.timeout", "user state store timeout") + + private val utegSimilarityEngineTimeout: Flag[Duration] = + flag[Duration]("uteg.similarityEngine.timeout", "uteg similarity engine timeout") + + private val earlybirdServerTimeout: Flag[Duration] = + flag[Duration]("earlybird.server.timeout", "earlybird server timeout") + + private val earlybirdSimilarityEngineTimeout: Flag[Duration] = + flag[Duration]("earlybird.similarityEngine.timeout", "Earlybird similarity engine timeout") + + private val frsBasedTweetEndpointTimeout: Flag[Duration] = + flag[Duration]( + "frsBasedTweet.endpoint.timeout", + "frsBasedTweet endpoint timeout" + ) + + private val topicTweetEndpointTimeout: Flag[Duration] = + flag[Duration]( + "topicTweet.endpoint.timeout", + "topicTweet endpoint timeout" + ) + + // timeout for Navi client + private val naviRequestTimeout: Flag[Duration] = + flag[Duration]( + NaviRequestTimeoutFlagName, + Duration.fromMilliseconds(2000), + "Request timeout for a single RPC Call", + ) + + @Provides + @Singleton + def provideTimeoutBudget(): TimeoutConfig = + TimeoutConfig( + serviceTimeout = serviceTimeout(), + signalFetchTimeout = signalFetchTimeout(), + similarityEngineTimeout = similarityEngineTimeout(), + annServiceClientTimeout = annServiceClientTimeout(), + utegSimilarityEngineTimeout = utegSimilarityEngineTimeout(), + userStateUnderlyingStoreTimeout = userStateUnderlyingStoreTimeout(), + userStateStoreTimeout = userStateStoreTimeout(), + earlybirdServerTimeout = earlybirdServerTimeout(), + earlybirdSimilarityEngineTimeout = earlybirdSimilarityEngineTimeout(), + frsBasedTweetEndpointTimeout = frsBasedTweetEndpointTimeout(), + topicTweetEndpointTimeout = topicTweetEndpointTimeout(), + naviRequestTimeout = naviRequestTimeout() + ) + +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/grpc_client/NaviGRPCClientModule.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/grpc_client/NaviGRPCClientModule.scala new file mode 100644 index 0000000000..418f447479 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/grpc_client/NaviGRPCClientModule.scala @@ -0,0 +1,90 @@ +package com.twitter.cr_mixer.module.grpc_client + +import com.google.inject.Provides +import com.twitter.cr_mixer.config.TimeoutConfig +import com.twitter.cr_mixer.model.ModuleNames +import com.twitter.finagle.Http +import com.twitter.finagle.grpc.FinagleChannelBuilder +import com.twitter.finagle.mtls.authentication.ServiceIdentifier +import com.twitter.finagle.mtls.client.MtlsStackClient.MtlsStackClientSyntax +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.inject.TwitterModule +import com.twitter.util.Duration +import io.grpc.ManagedChannel +import javax.inject.Named +import javax.inject.Singleton + +object NaviGRPCClientModule extends TwitterModule { + + val maxRetryAttempts = 3 + + @Provides + @Singleton + @Named(ModuleNames.HomeNaviGRPCClient) + def providesHomeNaviGRPCClient( + serviceIdentifier: ServiceIdentifier, + timeoutConfig: TimeoutConfig, + statsReceiver: StatsReceiver, + ): ManagedChannel = { + val label = "navi-wals-recommended-tweets-home-client" + val dest = "/s/ads-prediction/navi-wals-recommended-tweets-home" + buildClient(serviceIdentifier, timeoutConfig, statsReceiver, dest, label) + } + + @Provides + @Singleton + @Named(ModuleNames.AdsFavedNaviGRPCClient) + def providesAdsFavedNaviGRPCClient( + serviceIdentifier: ServiceIdentifier, + timeoutConfig: TimeoutConfig, + statsReceiver: StatsReceiver, + ): ManagedChannel = { + val label = "navi-wals-ads-faved-tweets" + val dest = "/s/ads-prediction/navi-wals-ads-faved-tweets" + buildClient(serviceIdentifier, timeoutConfig, statsReceiver, dest, label) + } + + @Provides + @Singleton + @Named(ModuleNames.AdsMonetizableNaviGRPCClient) + def providesAdsMonetizableNaviGRPCClient( + serviceIdentifier: ServiceIdentifier, + timeoutConfig: TimeoutConfig, + statsReceiver: StatsReceiver, + ): ManagedChannel = { + val label = "navi-wals-ads-monetizable-tweets" + val dest = "/s/ads-prediction/navi-wals-ads-monetizable-tweets" + buildClient(serviceIdentifier, timeoutConfig, statsReceiver, dest, label) + } + + private def buildClient( + serviceIdentifier: ServiceIdentifier, + timeoutConfig: TimeoutConfig, + statsReceiver: StatsReceiver, + dest: String, + label: String + ): ManagedChannel = { + + val stats = statsReceiver.scope("clnt").scope(label) + + val client = Http.client + .withLabel(label) + .withMutualTls(serviceIdentifier) + .withRequestTimeout(timeoutConfig.naviRequestTimeout) + .withTransport.connectTimeout(Duration.fromMilliseconds(10000)) + .withSession.acquisitionTimeout(Duration.fromMilliseconds(20000)) + .withStatsReceiver(stats) + .withHttpStats + + FinagleChannelBuilder + .forTarget(dest) + .overrideAuthority("rustserving") + .maxRetryAttempts(maxRetryAttempts) + .enableRetryForStatus(io.grpc.Status.RESOURCE_EXHAUSTED) + .enableRetryForStatus(io.grpc.Status.UNKNOWN) + .enableUnsafeFullyBufferingMode() + .httpClient(client) + .build() + + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/CertoTopicTweetSimilarityEngineModule.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/CertoTopicTweetSimilarityEngineModule.scala new file mode 100644 index 0000000000..6c82329b00 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/CertoTopicTweetSimilarityEngineModule.scala @@ -0,0 +1,57 @@ +package com.twitter.cr_mixer.module.similarity_engine + +import com.google.inject.Provides +import com.twitter.cr_mixer.config.TimeoutConfig +import com.twitter.cr_mixer.model.ModuleNames +import com.twitter.cr_mixer.model.TopicTweetWithScore +import com.twitter.cr_mixer.param.decider.CrMixerDecider +import com.twitter.cr_mixer.param.decider.DeciderConstants +import com.twitter.cr_mixer.similarity_engine.CertoTopicTweetSimilarityEngine +import com.twitter.cr_mixer.similarity_engine.CertoTopicTweetSimilarityEngine.Query +import com.twitter.cr_mixer.similarity_engine.EngineQuery +import com.twitter.cr_mixer.similarity_engine.SimilarityEngine.DeciderConfig +import com.twitter.cr_mixer.similarity_engine.SimilarityEngine.GatingConfig +import com.twitter.cr_mixer.similarity_engine.SimilarityEngine.SimilarityEngineConfig +import com.twitter.cr_mixer.similarity_engine.StandardSimilarityEngine +import com.twitter.cr_mixer.thriftscala.SimilarityEngineType +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.inject.TwitterModule +import com.twitter.simclusters_v2.thriftscala.TopicId +import com.twitter.storehaus.ReadableStore +import com.twitter.topic_recos.thriftscala.TweetWithScores +import javax.inject.Named +import javax.inject.Singleton + +object CertoTopicTweetSimilarityEngineModule extends TwitterModule { + + @Provides + @Singleton + @Named(ModuleNames.CertoTopicTweetSimilarityEngine) + def providesCertoTopicTweetSimilarityEngine( + @Named(ModuleNames.CertoStratoStoreName) certoStratoStore: ReadableStore[ + TopicId, + Seq[TweetWithScores] + ], + timeoutConfig: TimeoutConfig, + decider: CrMixerDecider, + statsReceiver: StatsReceiver + ): StandardSimilarityEngine[ + EngineQuery[Query], + TopicTweetWithScore + ] = { + new StandardSimilarityEngine[EngineQuery[Query], TopicTweetWithScore]( + implementingStore = CertoTopicTweetSimilarityEngine(certoStratoStore, statsReceiver), + identifier = SimilarityEngineType.CertoTopicTweet, + globalStats = statsReceiver, + engineConfig = SimilarityEngineConfig( + timeout = timeoutConfig.topicTweetEndpointTimeout, + gatingConfig = GatingConfig( + deciderConfig = + Some(DeciderConfig(decider, DeciderConstants.enableTopicTweetTrafficDeciderKey)), + enableFeatureSwitch = None + ) + ) + ) + } + +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/ConsumerBasedWalsSimilarityEngineModule.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/ConsumerBasedWalsSimilarityEngineModule.scala new file mode 100644 index 0000000000..e09f8b639e --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/ConsumerBasedWalsSimilarityEngineModule.scala @@ -0,0 +1,54 @@ +package com.twitter.cr_mixer.module.similarity_engine + +import com.google.inject.Provides +import com.twitter.cr_mixer.config.TimeoutConfig +import com.twitter.cr_mixer.model.ModuleNames +import com.twitter.cr_mixer.model.TweetWithScore +import com.twitter.cr_mixer.similarity_engine.ConsumerBasedWalsSimilarityEngine +import com.twitter.cr_mixer.similarity_engine.SimilarityEngine.GatingConfig +import com.twitter.cr_mixer.similarity_engine.SimilarityEngine.SimilarityEngineConfig +import com.twitter.cr_mixer.similarity_engine.StandardSimilarityEngine +import com.twitter.cr_mixer.thriftscala.SimilarityEngineType +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.inject.TwitterModule +import io.grpc.ManagedChannel +import javax.inject.Named + +object ConsumerBasedWalsSimilarityEngineModule extends TwitterModule { + @Provides + @Named(ModuleNames.ConsumerBasedWalsSimilarityEngine) + def providesConsumerBasedWalsSimilarityEngine( + timeoutConfig: TimeoutConfig, + statsReceiver: StatsReceiver, + @Named(ModuleNames.HomeNaviGRPCClient) homeNaviGRPCClient: ManagedChannel, + @Named(ModuleNames.AdsFavedNaviGRPCClient) adsFavedNaviGRPCClient: ManagedChannel, + @Named(ModuleNames.AdsMonetizableNaviGRPCClient) adsMonetizableNaviGRPCClient: ManagedChannel, + ): StandardSimilarityEngine[ + ConsumerBasedWalsSimilarityEngine.Query, + TweetWithScore + ] = { + + val underlyingStore = new ConsumerBasedWalsSimilarityEngine( + homeNaviGRPCClient, + adsFavedNaviGRPCClient, + adsMonetizableNaviGRPCClient, + statsReceiver + ) + + new StandardSimilarityEngine[ + ConsumerBasedWalsSimilarityEngine.Query, + TweetWithScore + ]( + implementingStore = underlyingStore, + identifier = SimilarityEngineType.ConsumerBasedWalsANN, + globalStats = statsReceiver, + engineConfig = SimilarityEngineConfig( + timeout = timeoutConfig.similarityEngineTimeout, + gatingConfig = GatingConfig( + deciderConfig = None, + enableFeatureSwitch = None + ) + ) + ) + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/ConsumerEmbeddingBasedTripSimilarityEngineModule.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/ConsumerEmbeddingBasedTripSimilarityEngineModule.scala new file mode 100644 index 0000000000..8d209798b4 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/ConsumerEmbeddingBasedTripSimilarityEngineModule.scala @@ -0,0 +1,60 @@ +package com.twitter.cr_mixer.module.similarity_engine + +import com.google.inject.Provides +import com.twitter.cr_mixer.config.TimeoutConfig +import com.twitter.cr_mixer.model.ModelConfig +import com.twitter.cr_mixer.model.ModuleNames +import com.twitter.cr_mixer.model.TripTweetWithScore +import com.twitter.cr_mixer.similarity_engine.ConsumerEmbeddingBasedTripSimilarityEngine +import com.twitter.cr_mixer.similarity_engine.SimilarityEngine.GatingConfig +import com.twitter.cr_mixer.similarity_engine.SimilarityEngine.SimilarityEngineConfig +import com.twitter.cr_mixer.similarity_engine.StandardSimilarityEngine +import com.twitter.cr_mixer.similarity_engine.TripEngineQuery +import com.twitter.cr_mixer.thriftscala.SimilarityEngineType +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.hermit.store.common.ObservedReadableStore +import com.twitter.inject.TwitterModule +import com.twitter.simclusters_v2.common.SimClustersEmbedding +import com.twitter.simclusters_v2.common.UserId +import com.twitter.storehaus.ReadableStore +import com.twitter.trends.trip_v1.trip_tweets.thriftscala.TripTweet +import com.twitter.trends.trip_v1.trip_tweets.thriftscala.TripDomain +import javax.inject.Named + +object ConsumerEmbeddingBasedTripSimilarityEngineModule extends TwitterModule { + @Provides + @Named(ModuleNames.ConsumerEmbeddingBasedTripSimilarityEngine) + def providesConsumerEmbeddingBasedTripSimilarityEngineModule( + @Named(ModuleNames.RmsUserLogFavInterestedInEmbeddingStore) + userLogFavInterestedInEmbeddingStore: ReadableStore[UserId, SimClustersEmbedding], + @Named(ModuleNames.RmsUserFollowInterestedInEmbeddingStore) + userFollowInterestedInEmbeddingStore: ReadableStore[UserId, SimClustersEmbedding], + @Named(ModuleNames.TripCandidateStore) + tripCandidateStore: ReadableStore[TripDomain, Seq[TripTweet]], + timeoutConfig: TimeoutConfig, + statsReceiver: StatsReceiver, + ): StandardSimilarityEngine[TripEngineQuery, TripTweetWithScore] = { + val underlyingStore = ObservedReadableStore( + ConsumerEmbeddingBasedTripSimilarityEngine( + embeddingStoreLookUpMap = Map( + ModelConfig.ConsumerLogFavBasedInterestedInEmbedding -> userLogFavInterestedInEmbeddingStore, + ModelConfig.ConsumerFollowBasedInterestedInEmbedding -> userFollowInterestedInEmbeddingStore, + ), + tripCandidateSource = tripCandidateStore, + statsReceiver + ))(statsReceiver.scope("TripSimilarityEngine")) + + new StandardSimilarityEngine[TripEngineQuery, TripTweetWithScore]( + implementingStore = underlyingStore, + identifier = SimilarityEngineType.ExploreTripOfflineSimClustersTweets, + globalStats = statsReceiver, + engineConfig = SimilarityEngineConfig( + timeout = timeoutConfig.similarityEngineTimeout, + gatingConfig = GatingConfig( + deciderConfig = None, + enableFeatureSwitch = None + ) + ) + ) + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/ConsumerEmbeddingBasedTwHINSimilarityEngineModule.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/ConsumerEmbeddingBasedTwHINSimilarityEngineModule.scala new file mode 100644 index 0000000000..289d052b4d --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/ConsumerEmbeddingBasedTwHINSimilarityEngineModule.scala @@ -0,0 +1,58 @@ +package com.twitter.cr_mixer.module.similarity_engine + +import com.google.inject.Provides +import com.twitter.ann.common.thriftscala.AnnQueryService +import com.twitter.cr_mixer.model.ModelConfig +import com.twitter.cr_mixer.module.EmbeddingStoreModule +import com.twitter.cr_mixer.module.thrift_client.AnnQueryServiceClientModule +import com.twitter.cr_mixer.similarity_engine.HnswANNSimilarityEngine +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.inject.TwitterModule +import com.twitter.simclusters_v2.thriftscala.InternalId +import com.twitter.storehaus.ReadableStore +import javax.inject.Named +import com.twitter.ml.api.{thriftscala => api} +import com.twitter.cr_mixer.model.ModuleNames +import com.twitter.cr_mixer.config.TimeoutConfig +import com.twitter.cr_mixer.similarity_engine.SimilarityEngine.GatingConfig +import com.twitter.cr_mixer.similarity_engine.SimilarityEngine.SimilarityEngineConfig +import com.twitter.cr_mixer.thriftscala.SimilarityEngineType + +object ConsumerEmbeddingBasedTwHINSimilarityEngineModule extends TwitterModule { + @Provides + @Named(ModuleNames.ConsumerEmbeddingBasedTwHINANNSimilarityEngine) + def providesConsumerEmbeddingBasedTwHINANNSimilarityEngine( + // MH stores + @Named(EmbeddingStoreModule.ConsumerBasedTwHINEmbeddingRegularUpdateMhStoreName) + consumerBasedTwHINEmbeddingRegularUpdateMhStore: ReadableStore[InternalId, api.Embedding], + @Named(EmbeddingStoreModule.DebuggerDemoUserEmbeddingMhStoreName) + debuggerDemoUserEmbeddingMhStore: ReadableStore[InternalId, api.Embedding], + @Named(AnnQueryServiceClientModule.TwHINRegularUpdateAnnServiceClientName) + twHINRegularUpdateAnnService: AnnQueryService.MethodPerEndpoint, + @Named(AnnQueryServiceClientModule.DebuggerDemoAnnServiceClientName) + debuggerDemoAnnService: AnnQueryService.MethodPerEndpoint, + // Other configs + timeoutConfig: TimeoutConfig, + statsReceiver: StatsReceiver + ): HnswANNSimilarityEngine = { + new HnswANNSimilarityEngine( + embeddingStoreLookUpMap = Map( + ModelConfig.ConsumerBasedTwHINRegularUpdateAll20221024 -> consumerBasedTwHINEmbeddingRegularUpdateMhStore, + ModelConfig.DebuggerDemo -> debuggerDemoUserEmbeddingMhStore, + ), + annServiceLookUpMap = Map( + ModelConfig.ConsumerBasedTwHINRegularUpdateAll20221024 -> twHINRegularUpdateAnnService, + ModelConfig.DebuggerDemo -> debuggerDemoAnnService, + ), + globalStats = statsReceiver, + identifier = SimilarityEngineType.ConsumerEmbeddingBasedTwHINANN, + engineConfig = SimilarityEngineConfig( + timeout = timeoutConfig.similarityEngineTimeout, + gatingConfig = GatingConfig( + deciderConfig = None, + enableFeatureSwitch = None + ) + ) + ) + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/ConsumerEmbeddingBasedTwoTowerSimilarityEngineModule.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/ConsumerEmbeddingBasedTwoTowerSimilarityEngineModule.scala new file mode 100644 index 0000000000..704093e366 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/ConsumerEmbeddingBasedTwoTowerSimilarityEngineModule.scala @@ -0,0 +1,51 @@ +package com.twitter.cr_mixer.module +package similarity_engine + +import com.google.inject.Provides +import com.twitter.ann.common.thriftscala.AnnQueryService +import com.twitter.cr_mixer.model.ModelConfig +import com.twitter.cr_mixer.module.EmbeddingStoreModule +import com.twitter.cr_mixer.module.thrift_client.AnnQueryServiceClientModule +import com.twitter.cr_mixer.similarity_engine.HnswANNSimilarityEngine +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.inject.TwitterModule +import com.twitter.simclusters_v2.thriftscala.InternalId +import com.twitter.storehaus.ReadableStore +import javax.inject.Named +import com.twitter.ml.api.{thriftscala => api} +import com.twitter.cr_mixer.model.ModuleNames +import com.twitter.cr_mixer.config.TimeoutConfig +import com.twitter.cr_mixer.similarity_engine.SimilarityEngine.GatingConfig +import com.twitter.cr_mixer.similarity_engine.SimilarityEngine.SimilarityEngineConfig +import com.twitter.cr_mixer.thriftscala.SimilarityEngineType + +object ConsumerEmbeddingBasedTwoTowerSimilarityEngineModule extends TwitterModule { + @Provides + @Named(ModuleNames.ConsumerEmbeddingBasedTwoTowerANNSimilarityEngine) + def providesConsumerEmbeddingBasedTwoTowerANNSimilarityEngine( + @Named(EmbeddingStoreModule.TwoTowerFavConsumerEmbeddingMhStoreName) + twoTowerFavConsumerEmbeddingMhStore: ReadableStore[InternalId, api.Embedding], + @Named(AnnQueryServiceClientModule.TwoTowerFavAnnServiceClientName) + twoTowerFavAnnService: AnnQueryService.MethodPerEndpoint, + timeoutConfig: TimeoutConfig, + statsReceiver: StatsReceiver + ): HnswANNSimilarityEngine = { + new HnswANNSimilarityEngine( + embeddingStoreLookUpMap = Map( + ModelConfig.TwoTowerFavALL20220808 -> twoTowerFavConsumerEmbeddingMhStore, + ), + annServiceLookUpMap = Map( + ModelConfig.TwoTowerFavALL20220808 -> twoTowerFavAnnService, + ), + globalStats = statsReceiver, + identifier = SimilarityEngineType.ConsumerEmbeddingBasedTwoTowerANN, + engineConfig = SimilarityEngineConfig( + timeout = timeoutConfig.similarityEngineTimeout, + gatingConfig = GatingConfig( + deciderConfig = None, + enableFeatureSwitch = None + ) + ) + ) + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/ConsumersBasedUserAdGraphSimilarityEngineModule.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/ConsumersBasedUserAdGraphSimilarityEngineModule.scala new file mode 100644 index 0000000000..e66a48a87f --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/ConsumersBasedUserAdGraphSimilarityEngineModule.scala @@ -0,0 +1,61 @@ +package com.twitter.cr_mixer.module.similarity_engine + +import com.google.inject.Provides +import com.twitter.cr_mixer.config.TimeoutConfig +import com.twitter.cr_mixer.model.ModuleNames +import com.twitter.cr_mixer.model.TweetWithScore +import com.twitter.cr_mixer.param.decider.CrMixerDecider +import com.twitter.cr_mixer.param.decider.DeciderConstants +import com.twitter.cr_mixer.similarity_engine.ConsumersBasedUserAdGraphSimilarityEngine +import com.twitter.cr_mixer.similarity_engine.SimilarityEngine.DeciderConfig +import com.twitter.cr_mixer.similarity_engine.SimilarityEngine.GatingConfig +import com.twitter.cr_mixer.similarity_engine.SimilarityEngine.SimilarityEngineConfig +import com.twitter.cr_mixer.similarity_engine.StandardSimilarityEngine +import com.twitter.cr_mixer.thriftscala.SimilarityEngineType +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.inject.TwitterModule +import com.twitter.recos.user_ad_graph.thriftscala.ConsumersBasedRelatedAdRequest +import com.twitter.recos.user_ad_graph.thriftscala.RelatedAdResponse +import com.twitter.storehaus.ReadableStore +import javax.inject.Named +import javax.inject.Singleton + +object ConsumersBasedUserAdGraphSimilarityEngineModule extends TwitterModule { + + @Provides + @Singleton + @Named(ModuleNames.ConsumersBasedUserAdGraphSimilarityEngine) + def providesConsumersBasedUserAdGraphSimilarityEngine( + @Named(ModuleNames.ConsumerBasedUserAdGraphStore) + consumersBasedUserAdGraphStore: ReadableStore[ + ConsumersBasedRelatedAdRequest, + RelatedAdResponse + ], + timeoutConfig: TimeoutConfig, + statsReceiver: StatsReceiver, + decider: CrMixerDecider + ): StandardSimilarityEngine[ + ConsumersBasedUserAdGraphSimilarityEngine.Query, + TweetWithScore + ] = { + + new StandardSimilarityEngine[ + ConsumersBasedUserAdGraphSimilarityEngine.Query, + TweetWithScore + ]( + implementingStore = + ConsumersBasedUserAdGraphSimilarityEngine(consumersBasedUserAdGraphStore, statsReceiver), + identifier = SimilarityEngineType.ConsumersBasedUserTweetGraph, + globalStats = statsReceiver, + engineConfig = SimilarityEngineConfig( + timeout = timeoutConfig.similarityEngineTimeout, + gatingConfig = GatingConfig( + deciderConfig = + Some(DeciderConfig(decider, DeciderConstants.enableUserTweetGraphTrafficDeciderKey)), + enableFeatureSwitch = None + ) + ), + memCacheConfig = None + ) + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/ConsumersBasedUserVideoGraphSimilarityEngineModule.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/ConsumersBasedUserVideoGraphSimilarityEngineModule.scala new file mode 100644 index 0000000000..977a90f259 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/ConsumersBasedUserVideoGraphSimilarityEngineModule.scala @@ -0,0 +1,62 @@ +package com.twitter.cr_mixer.module.similarity_engine + +import com.google.inject.Provides +import com.twitter.cr_mixer.config.TimeoutConfig +import com.twitter.cr_mixer.model.ModuleNames +import com.twitter.cr_mixer.model.TweetWithScore +import com.twitter.cr_mixer.param.decider.CrMixerDecider +import com.twitter.cr_mixer.param.decider.DeciderConstants +import com.twitter.cr_mixer.similarity_engine.ConsumersBasedUserVideoGraphSimilarityEngine +import com.twitter.cr_mixer.similarity_engine.SimilarityEngine.DeciderConfig +import com.twitter.cr_mixer.similarity_engine.SimilarityEngine.GatingConfig +import com.twitter.cr_mixer.similarity_engine.SimilarityEngine.SimilarityEngineConfig +import com.twitter.cr_mixer.similarity_engine.StandardSimilarityEngine +import com.twitter.cr_mixer.thriftscala.SimilarityEngineType +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.inject.TwitterModule +import com.twitter.recos.user_video_graph.thriftscala.ConsumersBasedRelatedTweetRequest +import com.twitter.recos.user_video_graph.thriftscala.RelatedTweetResponse +import com.twitter.storehaus.ReadableStore +import javax.inject.Named +import javax.inject.Singleton + +object ConsumersBasedUserVideoGraphSimilarityEngineModule extends TwitterModule { + + @Provides + @Singleton + @Named(ModuleNames.ConsumersBasedUserVideoGraphSimilarityEngine) + def providesConsumersBasedUserVideoGraphSimilarityEngine( + @Named(ModuleNames.ConsumerBasedUserVideoGraphStore) + consumersBasedUserVideoGraphStore: ReadableStore[ + ConsumersBasedRelatedTweetRequest, + RelatedTweetResponse + ], + timeoutConfig: TimeoutConfig, + statsReceiver: StatsReceiver, + decider: CrMixerDecider + ): StandardSimilarityEngine[ + ConsumersBasedUserVideoGraphSimilarityEngine.Query, + TweetWithScore + ] = { + + new StandardSimilarityEngine[ + ConsumersBasedUserVideoGraphSimilarityEngine.Query, + TweetWithScore + ]( + implementingStore = ConsumersBasedUserVideoGraphSimilarityEngine( + consumersBasedUserVideoGraphStore, + statsReceiver), + identifier = SimilarityEngineType.ConsumersBasedUserVideoGraph, + globalStats = statsReceiver, + engineConfig = SimilarityEngineConfig( + timeout = timeoutConfig.similarityEngineTimeout, + gatingConfig = GatingConfig( + deciderConfig = + Some(DeciderConfig(decider, DeciderConstants.enableUserVideoGraphTrafficDeciderKey)), + enableFeatureSwitch = None + ) + ), + memCacheConfig = None + ) + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/DiffusionBasedSimilarityEngineModule.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/DiffusionBasedSimilarityEngineModule.scala new file mode 100644 index 0000000000..f485210853 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/DiffusionBasedSimilarityEngineModule.scala @@ -0,0 +1,52 @@ +package com.twitter.cr_mixer.module +package similarity_engine + +import com.google.inject.Provides +import com.twitter.cr_mixer.config.TimeoutConfig +import com.twitter.cr_mixer.model.ModelConfig +import com.twitter.simclusters_v2.thriftscala.TweetsWithScore +import com.twitter.cr_mixer.model.TweetWithScore +import com.twitter.cr_mixer.model.ModuleNames +import com.twitter.cr_mixer.similarity_engine.DiffusionBasedSimilarityEngine +import com.twitter.cr_mixer.similarity_engine.DiffusionBasedSimilarityEngine.Query +import com.twitter.cr_mixer.similarity_engine.LookupSimilarityEngine +import com.twitter.cr_mixer.similarity_engine.SimilarityEngine.GatingConfig +import com.twitter.cr_mixer.similarity_engine.SimilarityEngine.SimilarityEngineConfig +import com.twitter.cr_mixer.thriftscala.SimilarityEngineType +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.inject.TwitterModule +import com.twitter.storehaus.ReadableStore +import javax.inject.Named +import javax.inject.Singleton + +object DiffusionBasedSimilarityEngineModule extends TwitterModule { + @Provides + @Singleton + @Named(ModuleNames.DiffusionBasedSimilarityEngine) + def providesDiffusionBasedSimilarityEngineModule( + @Named(ModuleNames.RetweetBasedDiffusionRecsMhStore) + retweetBasedDiffusionRecsMhStore: ReadableStore[Long, TweetsWithScore], + timeoutConfig: TimeoutConfig, + globalStats: StatsReceiver + ): LookupSimilarityEngine[Query, TweetWithScore] = { + + val versionedStoreMap = Map( + ModelConfig.RetweetBasedDiffusion -> DiffusionBasedSimilarityEngine( + retweetBasedDiffusionRecsMhStore, + globalStats), + ) + + new LookupSimilarityEngine[Query, TweetWithScore]( + versionedStoreMap = versionedStoreMap, + identifier = SimilarityEngineType.DiffusionBasedTweet, + globalStats = globalStats, + engineConfig = SimilarityEngineConfig( + timeout = timeoutConfig.similarityEngineTimeout, + gatingConfig = GatingConfig( + deciderConfig = None, + enableFeatureSwitch = None + ) + ) + ) + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/EarlybirdSimilarityEngineModule.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/EarlybirdSimilarityEngineModule.scala new file mode 100644 index 0000000000..6cdabfce4d --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/EarlybirdSimilarityEngineModule.scala @@ -0,0 +1,120 @@ +package com.twitter.cr_mixer.module.similarity_engine + +import com.google.inject.Provides +import com.twitter.cr_mixer.config.TimeoutConfig +import com.twitter.cr_mixer.param.decider.CrMixerDecider +import com.twitter.cr_mixer.param.decider.DeciderConstants +import com.twitter.cr_mixer.similarity_engine.EarlybirdModelBasedSimilarityEngine +import com.twitter.cr_mixer.similarity_engine.EarlybirdRecencyBasedSimilarityEngine +import com.twitter.cr_mixer.similarity_engine.EarlybirdSimilarityEngine +import com.twitter.cr_mixer.similarity_engine.EarlybirdTensorflowBasedSimilarityEngine +import com.twitter.cr_mixer.similarity_engine.SimilarityEngine.DeciderConfig +import com.twitter.cr_mixer.similarity_engine.SimilarityEngine.GatingConfig +import com.twitter.cr_mixer.similarity_engine.SimilarityEngine.SimilarityEngineConfig +import com.twitter.cr_mixer.thriftscala.SimilarityEngineType +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.inject.TwitterModule +import javax.inject.Singleton + +object EarlybirdSimilarityEngineModule extends TwitterModule { + + @Provides + @Singleton + def providesRecencyBasedEarlybirdSimilarityEngine( + earlybirdRecencyBasedSimilarityEngine: EarlybirdRecencyBasedSimilarityEngine, + timeoutConfig: TimeoutConfig, + decider: CrMixerDecider, + statsReceiver: StatsReceiver + ): EarlybirdSimilarityEngine[ + EarlybirdRecencyBasedSimilarityEngine.EarlybirdRecencyBasedSearchQuery, + EarlybirdRecencyBasedSimilarityEngine + ] = { + new EarlybirdSimilarityEngine[ + EarlybirdRecencyBasedSimilarityEngine.EarlybirdRecencyBasedSearchQuery, + EarlybirdRecencyBasedSimilarityEngine + ]( + implementingStore = earlybirdRecencyBasedSimilarityEngine, + identifier = SimilarityEngineType.EarlybirdRecencyBasedSimilarityEngine, + globalStats = + statsReceiver.scope(SimilarityEngineType.EarlybirdRecencyBasedSimilarityEngine.name), + engineConfig = SimilarityEngineConfig( + timeout = timeoutConfig.earlybirdSimilarityEngineTimeout, + gatingConfig = GatingConfig( + deciderConfig = Some( + DeciderConfig( + decider = decider, + deciderString = DeciderConstants.enableEarlybirdTrafficDeciderKey + )), + enableFeatureSwitch = None + ) + ) + ) + } + + @Provides + @Singleton + def providesModelBasedEarlybirdSimilarityEngine( + earlybirdModelBasedSimilarityEngine: EarlybirdModelBasedSimilarityEngine, + timeoutConfig: TimeoutConfig, + decider: CrMixerDecider, + statsReceiver: StatsReceiver + ): EarlybirdSimilarityEngine[ + EarlybirdModelBasedSimilarityEngine.EarlybirdModelBasedSearchQuery, + EarlybirdModelBasedSimilarityEngine + ] = { + new EarlybirdSimilarityEngine[ + EarlybirdModelBasedSimilarityEngine.EarlybirdModelBasedSearchQuery, + EarlybirdModelBasedSimilarityEngine + ]( + implementingStore = earlybirdModelBasedSimilarityEngine, + identifier = SimilarityEngineType.EarlybirdModelBasedSimilarityEngine, + globalStats = + statsReceiver.scope(SimilarityEngineType.EarlybirdModelBasedSimilarityEngine.name), + engineConfig = SimilarityEngineConfig( + timeout = timeoutConfig.earlybirdSimilarityEngineTimeout, + gatingConfig = GatingConfig( + deciderConfig = Some( + DeciderConfig( + decider = decider, + deciderString = DeciderConstants.enableEarlybirdTrafficDeciderKey + )), + enableFeatureSwitch = None + ) + ) + ) + } + + @Provides + @Singleton + def providesTensorflowBasedEarlybirdSimilarityEngine( + earlybirdTensorflowBasedSimilarityEngine: EarlybirdTensorflowBasedSimilarityEngine, + timeoutConfig: TimeoutConfig, + decider: CrMixerDecider, + statsReceiver: StatsReceiver + ): EarlybirdSimilarityEngine[ + EarlybirdTensorflowBasedSimilarityEngine.EarlybirdTensorflowBasedSearchQuery, + EarlybirdTensorflowBasedSimilarityEngine + ] = { + new EarlybirdSimilarityEngine[ + EarlybirdTensorflowBasedSimilarityEngine.EarlybirdTensorflowBasedSearchQuery, + EarlybirdTensorflowBasedSimilarityEngine + ]( + implementingStore = earlybirdTensorflowBasedSimilarityEngine, + identifier = SimilarityEngineType.EarlybirdTensorflowBasedSimilarityEngine, + globalStats = + statsReceiver.scope(SimilarityEngineType.EarlybirdTensorflowBasedSimilarityEngine.name), + engineConfig = SimilarityEngineConfig( + timeout = timeoutConfig.earlybirdSimilarityEngineTimeout, + gatingConfig = GatingConfig( + deciderConfig = Some( + DeciderConfig( + decider = decider, + deciderString = DeciderConstants.enableEarlybirdTrafficDeciderKey + )), + enableFeatureSwitch = None + ) + ) + ) + } + +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/ProducerBasedUnifiedSimilarityEngineModule.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/ProducerBasedUnifiedSimilarityEngineModule.scala new file mode 100644 index 0000000000..b16d59924b --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/ProducerBasedUnifiedSimilarityEngineModule.scala @@ -0,0 +1,68 @@ +package com.twitter.cr_mixer.module.similarity_engine + +import com.google.inject.Provides +import com.twitter.cr_mixer.model.ModuleNames +import com.twitter.cr_mixer.model.TweetWithCandidateGenerationInfo +import com.twitter.cr_mixer.model.TweetWithScore +import com.twitter.cr_mixer.config.TimeoutConfig +import com.twitter.cr_mixer.similarity_engine.ProducerBasedUserTweetGraphSimilarityEngine +import com.twitter.cr_mixer.similarity_engine.ProducerBasedUnifiedSimilarityEngine +import com.twitter.cr_mixer.similarity_engine.SimilarityEngine.GatingConfig +import com.twitter.cr_mixer.similarity_engine.SimilarityEngine.SimilarityEngineConfig +import com.twitter.cr_mixer.similarity_engine.StandardSimilarityEngine +import com.twitter.cr_mixer.thriftscala.SimilarityEngineType +import com.twitter.cr_mixer.similarity_engine.SimClustersANNSimilarityEngine +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.inject.TwitterModule +import com.twitter.storehaus.ReadableStore +import javax.inject.Named +import javax.inject.Singleton + +object ProducerBasedUnifiedSimilarityEngineModule extends TwitterModule { + + @Provides + @Singleton + @Named(ModuleNames.ProducerBasedUnifiedSimilarityEngine) + def providesProducerBasedUnifiedSimilarityEngine( + @Named(ModuleNames.ProducerBasedUserTweetGraphSimilarityEngine) + producerBasedUserTweetGraphSimilarityEngine: StandardSimilarityEngine[ + ProducerBasedUserTweetGraphSimilarityEngine.Query, + TweetWithScore + ], + @Named(ModuleNames.SimClustersANNSimilarityEngine) + simClustersANNSimilarityEngine: StandardSimilarityEngine[ + SimClustersANNSimilarityEngine.Query, + TweetWithScore + ], + timeoutConfig: TimeoutConfig, + statsReceiver: StatsReceiver, + ): StandardSimilarityEngine[ + ProducerBasedUnifiedSimilarityEngine.Query, + TweetWithCandidateGenerationInfo + ] = { + + val underlyingStore: ReadableStore[ProducerBasedUnifiedSimilarityEngine.Query, Seq[ + TweetWithCandidateGenerationInfo + ]] = ProducerBasedUnifiedSimilarityEngine( + producerBasedUserTweetGraphSimilarityEngine, + simClustersANNSimilarityEngine, + statsReceiver + ) + + new StandardSimilarityEngine[ + ProducerBasedUnifiedSimilarityEngine.Query, + TweetWithCandidateGenerationInfo + ]( + implementingStore = underlyingStore, + identifier = SimilarityEngineType.ProducerBasedUnifiedSimilarityEngine, + globalStats = statsReceiver, + engineConfig = SimilarityEngineConfig( + timeout = timeoutConfig.similarityEngineTimeout, + gatingConfig = GatingConfig( + deciderConfig = None, + enableFeatureSwitch = None + ) + ) + ) + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/ProducerBasedUserAdGraphSimilarityEngineModule.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/ProducerBasedUserAdGraphSimilarityEngineModule.scala new file mode 100644 index 0000000000..d221a58a92 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/ProducerBasedUserAdGraphSimilarityEngineModule.scala @@ -0,0 +1,67 @@ +package com.twitter.cr_mixer.module.similarity_engine + +import com.google.inject.Provides +import com.twitter.conversions.DurationOps._ +import com.twitter.cr_mixer.config.TimeoutConfig +import com.twitter.cr_mixer.model.ModuleNames +import com.twitter.cr_mixer.model.TweetWithScore +import com.twitter.cr_mixer.param.decider.CrMixerDecider +import com.twitter.cr_mixer.param.decider.DeciderConstants +import com.twitter.cr_mixer.similarity_engine.ProducerBasedUserAdGraphSimilarityEngine +import com.twitter.cr_mixer.similarity_engine.SimilarityEngine.DeciderConfig +import com.twitter.cr_mixer.similarity_engine.SimilarityEngine.GatingConfig +import com.twitter.cr_mixer.similarity_engine.SimilarityEngine.SimilarityEngineConfig +import com.twitter.cr_mixer.similarity_engine.SimilarityEngine._ +import com.twitter.cr_mixer.similarity_engine.SimilarityEngine.keyHasher +import com.twitter.cr_mixer.similarity_engine.StandardSimilarityEngine +import com.twitter.cr_mixer.thriftscala.SimilarityEngineType +import com.twitter.finagle.memcached.{Client => MemcachedClient} +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.inject.TwitterModule +import com.twitter.recos.user_ad_graph.thriftscala.UserAdGraph +import javax.inject.Named +import javax.inject.Singleton + +object ProducerBasedUserAdGraphSimilarityEngineModule extends TwitterModule { + + @Provides + @Singleton + @Named(ModuleNames.ProducerBasedUserAdGraphSimilarityEngine) + def providesProducerBasedUserAdGraphSimilarityEngine( + userAdGraphService: UserAdGraph.MethodPerEndpoint, + @Named(ModuleNames.UnifiedCache) crMixerUnifiedCacheClient: MemcachedClient, + timeoutConfig: TimeoutConfig, + statsReceiver: StatsReceiver, + decider: CrMixerDecider + ): StandardSimilarityEngine[ + ProducerBasedUserAdGraphSimilarityEngine.Query, + TweetWithScore + ] = { + new StandardSimilarityEngine[ + ProducerBasedUserAdGraphSimilarityEngine.Query, + TweetWithScore + ]( + implementingStore = + ProducerBasedUserAdGraphSimilarityEngine(userAdGraphService, statsReceiver), + identifier = SimilarityEngineType.ProducerBasedUserAdGraph, + globalStats = statsReceiver, + engineConfig = SimilarityEngineConfig( + timeout = timeoutConfig.similarityEngineTimeout, + gatingConfig = GatingConfig( + deciderConfig = + Some(DeciderConfig(decider, DeciderConstants.enableUserAdGraphTrafficDeciderKey)), + enableFeatureSwitch = None + ) + ), + memCacheConfig = Some( + MemCacheConfig( + cacheClient = crMixerUnifiedCacheClient, + ttl = 10.minutes, + keyToString = { k => + //Example Query CRMixer:ProducerBasedUTG:1234567890ABCDEF + f"ProducerBasedUTG:${keyHasher.hashKey(k.toString.getBytes)}%X" + } + )) + ) + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/ProducerBasedUserTweetGraphSimilarityEngineModule.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/ProducerBasedUserTweetGraphSimilarityEngineModule.scala new file mode 100644 index 0000000000..a5821d01c2 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/ProducerBasedUserTweetGraphSimilarityEngineModule.scala @@ -0,0 +1,67 @@ +package com.twitter.cr_mixer.module.similarity_engine + +import com.google.inject.Provides +import com.twitter.conversions.DurationOps._ +import com.twitter.cr_mixer.model.ModuleNames +import com.twitter.cr_mixer.model.TweetWithScore +import com.twitter.cr_mixer.config.TimeoutConfig +import com.twitter.cr_mixer.param.decider.CrMixerDecider +import com.twitter.cr_mixer.param.decider.DeciderConstants +import com.twitter.cr_mixer.similarity_engine.ProducerBasedUserTweetGraphSimilarityEngine +import com.twitter.cr_mixer.similarity_engine.SimilarityEngine._ +import com.twitter.cr_mixer.similarity_engine.SimilarityEngine.keyHasher +import com.twitter.cr_mixer.similarity_engine.SimilarityEngine.DeciderConfig +import com.twitter.cr_mixer.similarity_engine.SimilarityEngine.GatingConfig +import com.twitter.cr_mixer.similarity_engine.SimilarityEngine.SimilarityEngineConfig +import com.twitter.cr_mixer.similarity_engine.StandardSimilarityEngine +import com.twitter.cr_mixer.thriftscala.SimilarityEngineType +import com.twitter.finagle.memcached.{Client => MemcachedClient} +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.inject.TwitterModule +import com.twitter.recos.user_tweet_graph.thriftscala.UserTweetGraph +import javax.inject.Named +import javax.inject.Singleton + +object ProducerBasedUserTweetGraphSimilarityEngineModule extends TwitterModule { + + @Provides + @Singleton + @Named(ModuleNames.ProducerBasedUserTweetGraphSimilarityEngine) + def providesProducerBasedUserTweetGraphSimilarityEngine( + userTweetGraphService: UserTweetGraph.MethodPerEndpoint, + @Named(ModuleNames.UnifiedCache) crMixerUnifiedCacheClient: MemcachedClient, + timeoutConfig: TimeoutConfig, + statsReceiver: StatsReceiver, + decider: CrMixerDecider + ): StandardSimilarityEngine[ + ProducerBasedUserTweetGraphSimilarityEngine.Query, + TweetWithScore + ] = { + new StandardSimilarityEngine[ + ProducerBasedUserTweetGraphSimilarityEngine.Query, + TweetWithScore + ]( + implementingStore = + ProducerBasedUserTweetGraphSimilarityEngine(userTweetGraphService, statsReceiver), + identifier = SimilarityEngineType.ProducerBasedUserTweetGraph, + globalStats = statsReceiver, + engineConfig = SimilarityEngineConfig( + timeout = timeoutConfig.similarityEngineTimeout, + gatingConfig = GatingConfig( + deciderConfig = + Some(DeciderConfig(decider, DeciderConstants.enableUserTweetGraphTrafficDeciderKey)), + enableFeatureSwitch = None + ) + ), + memCacheConfig = Some( + MemCacheConfig( + cacheClient = crMixerUnifiedCacheClient, + ttl = 10.minutes, + keyToString = { k => + //Example Query CRMixer:ProducerBasedUTG:1234567890ABCDEF + f"ProducerBasedUTG:${keyHasher.hashKey(k.toString.getBytes)}%X" + } + )) + ) + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/SimClustersANNSimilarityEngineModule.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/SimClustersANNSimilarityEngineModule.scala new file mode 100644 index 0000000000..7af68327d2 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/SimClustersANNSimilarityEngineModule.scala @@ -0,0 +1,117 @@ +package com.twitter.cr_mixer.module.similarity_engine + +import com.google.inject.Provides +import com.twitter.conversions.DurationOps._ +import com.twitter.cr_mixer.model.ModuleNames +import com.twitter.cr_mixer.model.TweetWithScore +import com.twitter.cr_mixer.config.TimeoutConfig +import com.twitter.cr_mixer.similarity_engine.SimClustersANNSimilarityEngine +import com.twitter.cr_mixer.similarity_engine.SimClustersANNSimilarityEngine.Query +import com.twitter.cr_mixer.similarity_engine.SimilarityEngine.GatingConfig +import com.twitter.cr_mixer.similarity_engine.SimilarityEngine.SimilarityEngineConfig +import com.twitter.cr_mixer.similarity_engine.StandardSimilarityEngine +import com.twitter.cr_mixer.thriftscala.SimilarityEngineType +import com.twitter.finagle.memcached.{Client => MemcachedClient} +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.hashing.KeyHasher +import com.twitter.hermit.store.common.ObservedMemcachedReadableStore +import com.twitter.hermit.store.common.ObservedReadableStore +import com.twitter.inject.TwitterModule +import com.twitter.relevance_platform.common.injection.LZ4Injection +import com.twitter.relevance_platform.common.injection.SeqObjectInjection +import com.twitter.simclusters_v2.candidate_source.SimClustersANNCandidateSource.CacheableShortTTLEmbeddingTypes +import com.twitter.simclustersann.thriftscala.SimClustersANNService +import com.twitter.storehaus.ReadableStore +import com.twitter.util.Future +import javax.inject.Named +import javax.inject.Singleton + +object SimClustersANNSimilarityEngineModule extends TwitterModule { + + private val keyHasher: KeyHasher = KeyHasher.FNV1A_64 + + @Provides + @Singleton + @Named(ModuleNames.SimClustersANNSimilarityEngine) + def providesProdSimClustersANNSimilarityEngine( + @Named(ModuleNames.UnifiedCache) crMixerUnifiedCacheClient: MemcachedClient, + simClustersANNServiceNameToClientMapper: Map[String, SimClustersANNService.MethodPerEndpoint], + timeoutConfig: TimeoutConfig, + statsReceiver: StatsReceiver + ): StandardSimilarityEngine[Query, TweetWithScore] = { + + val underlyingStore = + SimClustersANNSimilarityEngine(simClustersANNServiceNameToClientMapper, statsReceiver) + + val observedReadableStore = + ObservedReadableStore(underlyingStore)(statsReceiver.scope("SimClustersANNServiceStore")) + + val memCachedStore: ReadableStore[Query, Seq[TweetWithScore]] = + ObservedMemcachedReadableStore + .fromCacheClient( + backingStore = observedReadableStore, + cacheClient = crMixerUnifiedCacheClient, + ttl = 10.minutes + )( + valueInjection = LZ4Injection.compose(SeqObjectInjection[TweetWithScore]()), + statsReceiver = statsReceiver.scope("simclusters_ann_store_memcache"), + keyToString = { k => + //Example Query CRMixer:SCANN:1:2:1234567890ABCDEF:1234567890ABCDEF + f"CRMixer:SCANN:${k.simClustersANNQuery.sourceEmbeddingId.embeddingType.getValue()}%X" + + f":${k.simClustersANNQuery.sourceEmbeddingId.modelVersion.getValue()}%X" + + f":${keyHasher.hashKey(k.simClustersANNQuery.sourceEmbeddingId.internalId.toString.getBytes)}%X" + + f":${keyHasher.hashKey(k.simClustersANNQuery.config.toString.getBytes)}%X" + } + ) + + // Only cache the candidates if it's not Consumer-source. For example, TweetSource, + // ProducerSource, TopicSource + val wrapperStats = statsReceiver.scope("SimClustersANNWrapperStore") + + val wrapperStore: ReadableStore[Query, Seq[TweetWithScore]] = + buildWrapperStore(memCachedStore, observedReadableStore, wrapperStats) + + new StandardSimilarityEngine[ + Query, + TweetWithScore + ]( + implementingStore = wrapperStore, + identifier = SimilarityEngineType.SimClustersANN, + globalStats = statsReceiver, + engineConfig = SimilarityEngineConfig( + timeout = timeoutConfig.similarityEngineTimeout, + gatingConfig = GatingConfig( + deciderConfig = None, + enableFeatureSwitch = None + ) + ) + ) + } + + def buildWrapperStore( + memCachedStore: ReadableStore[Query, Seq[TweetWithScore]], + underlyingStore: ReadableStore[Query, Seq[TweetWithScore]], + wrapperStats: StatsReceiver + ): ReadableStore[Query, Seq[TweetWithScore]] = { + + // Only cache the candidates if it's not Consumer-source. For example, TweetSource, + // ProducerSource, TopicSource + val wrapperStore: ReadableStore[Query, Seq[TweetWithScore]] = + new ReadableStore[Query, Seq[TweetWithScore]] { + + override def multiGet[K1 <: Query]( + queries: Set[K1] + ): Map[K1, Future[Option[Seq[TweetWithScore]]]] = { + val (cacheableQueries, nonCacheableQueries) = + queries.partition { query => + CacheableShortTTLEmbeddingTypes.contains( + query.simClustersANNQuery.sourceEmbeddingId.embeddingType) + } + memCachedStore.multiGet(cacheableQueries) ++ + underlyingStore.multiGet(nonCacheableQueries) + } + } + wrapperStore + } + +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/SkitTopicTweetSimilarityEngineModule.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/SkitTopicTweetSimilarityEngineModule.scala new file mode 100644 index 0000000000..4de20fcfec --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/SkitTopicTweetSimilarityEngineModule.scala @@ -0,0 +1,88 @@ +package com.twitter.cr_mixer.module.similarity_engine + +import com.google.inject.Provides +import com.twitter.cr_mixer.config.TimeoutConfig +import com.twitter.cr_mixer.model.ModuleNames +import com.twitter.cr_mixer.model.TopicTweetWithScore +import com.twitter.cr_mixer.param.decider.CrMixerDecider +import com.twitter.cr_mixer.param.decider.DeciderConstants +import com.twitter.cr_mixer.similarity_engine.EngineQuery +import com.twitter.cr_mixer.similarity_engine.SimilarityEngine.DeciderConfig +import com.twitter.cr_mixer.similarity_engine.SimilarityEngine.GatingConfig +import com.twitter.cr_mixer.similarity_engine.SimilarityEngine.SimilarityEngineConfig +import com.twitter.cr_mixer.similarity_engine.SkitHighPrecisionTopicTweetSimilarityEngine +import com.twitter.cr_mixer.similarity_engine.SkitTopicTweetSimilarityEngine +import com.twitter.cr_mixer.similarity_engine.SkitTopicTweetSimilarityEngine.Query +import com.twitter.cr_mixer.similarity_engine.StandardSimilarityEngine +import com.twitter.cr_mixer.thriftscala.SimilarityEngineType +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.inject.TwitterModule +import com.twitter.storehaus.ReadableStore +import com.twitter.topic_recos.thriftscala.TopicTweet +import com.twitter.topic_recos.thriftscala.TopicTweetPartitionFlatKey +import javax.inject.Named +import javax.inject.Singleton + +object SkitTopicTweetSimilarityEngineModule extends TwitterModule { + + @Provides + @Singleton + @Named(ModuleNames.SkitHighPrecisionTopicTweetSimilarityEngine) + def providesSkitHighPrecisionTopicTweetSimilarityEngine( + @Named(ModuleNames.SkitStratoStoreName) skitStratoStore: ReadableStore[ + TopicTweetPartitionFlatKey, + Seq[TopicTweet] + ], + timeoutConfig: TimeoutConfig, + decider: CrMixerDecider, + statsReceiver: StatsReceiver + ): StandardSimilarityEngine[ + EngineQuery[Query], + TopicTweetWithScore + ] = { + new StandardSimilarityEngine[EngineQuery[Query], TopicTweetWithScore]( + implementingStore = + SkitHighPrecisionTopicTweetSimilarityEngine(skitStratoStore, statsReceiver), + identifier = SimilarityEngineType.SkitHighPrecisionTopicTweet, + globalStats = statsReceiver.scope(SimilarityEngineType.SkitHighPrecisionTopicTweet.name), + engineConfig = SimilarityEngineConfig( + timeout = timeoutConfig.topicTweetEndpointTimeout, + gatingConfig = GatingConfig( + deciderConfig = + Some(DeciderConfig(decider, DeciderConstants.enableTopicTweetTrafficDeciderKey)), + enableFeatureSwitch = None + ) + ) + ) + } + @Provides + @Singleton + @Named(ModuleNames.SkitTopicTweetSimilarityEngine) + def providesSkitTfgTopicTweetSimilarityEngine( + @Named(ModuleNames.SkitStratoStoreName) skitStratoStore: ReadableStore[ + TopicTweetPartitionFlatKey, + Seq[TopicTweet] + ], + timeoutConfig: TimeoutConfig, + decider: CrMixerDecider, + statsReceiver: StatsReceiver + ): StandardSimilarityEngine[ + EngineQuery[Query], + TopicTweetWithScore + ] = { + new StandardSimilarityEngine[EngineQuery[Query], TopicTweetWithScore]( + implementingStore = SkitTopicTweetSimilarityEngine(skitStratoStore, statsReceiver), + identifier = SimilarityEngineType.SkitTfgTopicTweet, + globalStats = statsReceiver.scope(SimilarityEngineType.SkitTfgTopicTweet.name), + engineConfig = SimilarityEngineConfig( + timeout = timeoutConfig.topicTweetEndpointTimeout, + gatingConfig = GatingConfig( + deciderConfig = + Some(DeciderConfig(decider, DeciderConstants.enableTopicTweetTrafficDeciderKey)), + enableFeatureSwitch = None + ) + ) + ) + } + +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/TweetBasedQigSimilarityEngineModule.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/TweetBasedQigSimilarityEngineModule.scala new file mode 100644 index 0000000000..06d9a21861 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/TweetBasedQigSimilarityEngineModule.scala @@ -0,0 +1,66 @@ +package com.twitter.cr_mixer.module.similarity_engine + +import com.google.inject.Provides +import com.twitter.conversions.DurationOps._ +import com.twitter.cr_mixer.model.ModuleNames +import com.twitter.cr_mixer.model.TweetWithScore +import com.twitter.cr_mixer.config.TimeoutConfig +import com.twitter.cr_mixer.param.decider.CrMixerDecider +import com.twitter.cr_mixer.param.decider.DeciderConstants +import com.twitter.cr_mixer.similarity_engine.SimilarityEngine._ +import com.twitter.cr_mixer.similarity_engine.SimilarityEngine.keyHasher +import com.twitter.cr_mixer.similarity_engine.SimilarityEngine.DeciderConfig +import com.twitter.cr_mixer.similarity_engine.SimilarityEngine.GatingConfig +import com.twitter.cr_mixer.similarity_engine.SimilarityEngine.SimilarityEngineConfig +import com.twitter.cr_mixer.similarity_engine.StandardSimilarityEngine +import com.twitter.cr_mixer.similarity_engine.TweetBasedQigSimilarityEngine +import com.twitter.cr_mixer.thriftscala.SimilarityEngineType +import com.twitter.finagle.memcached.{Client => MemcachedClient} +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.inject.TwitterModule +import com.twitter.qig_ranker.thriftscala.QigRanker +import javax.inject.Named +import javax.inject.Singleton + +object TweetBasedQigSimilarityEngineModule extends TwitterModule { + + @Provides + @Singleton + @Named(ModuleNames.TweetBasedQigSimilarityEngine) + def providesTweetBasedQigSimilarTweetsCandidateSource( + qigRanker: QigRanker.MethodPerEndpoint, + @Named(ModuleNames.UnifiedCache) crMixerUnifiedCacheClient: MemcachedClient, + timeoutConfig: TimeoutConfig, + statsReceiver: StatsReceiver, + decider: CrMixerDecider + ): StandardSimilarityEngine[ + TweetBasedQigSimilarityEngine.Query, + TweetWithScore + ] = { + new StandardSimilarityEngine[ + TweetBasedQigSimilarityEngine.Query, + TweetWithScore + ]( + implementingStore = TweetBasedQigSimilarityEngine(qigRanker, statsReceiver), + identifier = SimilarityEngineType.Qig, + globalStats = statsReceiver, + engineConfig = SimilarityEngineConfig( + timeout = timeoutConfig.similarityEngineTimeout, + gatingConfig = GatingConfig( + deciderConfig = + Some(DeciderConfig(decider, DeciderConstants.enableQigSimilarTweetsTrafficDeciderKey)), + enableFeatureSwitch = None + ) + ), + memCacheConfig = Some( + MemCacheConfig( + cacheClient = crMixerUnifiedCacheClient, + ttl = 10.minutes, + keyToString = { k => + f"TweetBasedQIGRanker:${keyHasher.hashKey(k.sourceId.toString.getBytes)}%X" + } + ) + ) + ) + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/TweetBasedTwHINSimlarityEngineModule.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/TweetBasedTwHINSimlarityEngineModule.scala new file mode 100644 index 0000000000..cc9da4772d --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/TweetBasedTwHINSimlarityEngineModule.scala @@ -0,0 +1,70 @@ +package com.twitter.cr_mixer.module.similarity_engine +import com.google.inject.Provides +import com.twitter.ann.common.thriftscala.AnnQueryService +import com.twitter.cr_mixer.model.ModelConfig +import com.twitter.cr_mixer.module.EmbeddingStoreModule +import com.twitter.cr_mixer.module.thrift_client.AnnQueryServiceClientModule +import com.twitter.cr_mixer.similarity_engine.HnswANNSimilarityEngine +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.inject.TwitterModule +import com.twitter.simclusters_v2.thriftscala.InternalId +import com.twitter.storehaus.ReadableStore +import javax.inject.Named +import com.twitter.ml.api.{thriftscala => api} +import com.twitter.conversions.DurationOps._ +import com.twitter.cr_mixer.model.ModuleNames +import com.twitter.cr_mixer.config.TimeoutConfig +import com.twitter.cr_mixer.similarity_engine.HnswANNEngineQuery +import com.twitter.cr_mixer.similarity_engine.SimilarityEngine +import com.twitter.cr_mixer.similarity_engine.SimilarityEngine.GatingConfig +import com.twitter.cr_mixer.similarity_engine.SimilarityEngine.SimilarityEngineConfig +import com.twitter.cr_mixer.thriftscala.SimilarityEngineType +import com.twitter.finagle.memcached.{Client => MemcachedClient} + +object TweetBasedTwHINSimlarityEngineModule extends TwitterModule { + @Provides + @Named(ModuleNames.TweetBasedTwHINANNSimilarityEngine) + def providesTweetBasedTwHINANNSimilarityEngine( + // MH stores + @Named(EmbeddingStoreModule.TwHINEmbeddingRegularUpdateMhStoreName) + twHINEmbeddingRegularUpdateMhStore: ReadableStore[InternalId, api.Embedding], + @Named(EmbeddingStoreModule.DebuggerDemoTweetEmbeddingMhStoreName) + debuggerDemoTweetEmbeddingMhStore: ReadableStore[InternalId, api.Embedding], + // ANN clients + @Named(AnnQueryServiceClientModule.TwHINRegularUpdateAnnServiceClientName) + twHINRegularUpdateAnnService: AnnQueryService.MethodPerEndpoint, + @Named(AnnQueryServiceClientModule.DebuggerDemoAnnServiceClientName) + debuggerDemoAnnService: AnnQueryService.MethodPerEndpoint, + // Other configs + @Named(ModuleNames.UnifiedCache) crMixerUnifiedCacheClient: MemcachedClient, + timeoutConfig: TimeoutConfig, + statsReceiver: StatsReceiver + ): HnswANNSimilarityEngine = { + new HnswANNSimilarityEngine( + embeddingStoreLookUpMap = Map( + ModelConfig.TweetBasedTwHINRegularUpdateAll20221024 -> twHINEmbeddingRegularUpdateMhStore, + ModelConfig.DebuggerDemo -> debuggerDemoTweetEmbeddingMhStore, + ), + annServiceLookUpMap = Map( + ModelConfig.TweetBasedTwHINRegularUpdateAll20221024 -> twHINRegularUpdateAnnService, + ModelConfig.DebuggerDemo -> debuggerDemoAnnService, + ), + globalStats = statsReceiver, + identifier = SimilarityEngineType.TweetBasedTwHINANN, + engineConfig = SimilarityEngineConfig( + timeout = timeoutConfig.similarityEngineTimeout, + gatingConfig = GatingConfig( + deciderConfig = None, + enableFeatureSwitch = None + ) + ), + memCacheConfigOpt = Some( + SimilarityEngine.MemCacheConfig[HnswANNEngineQuery]( + cacheClient = crMixerUnifiedCacheClient, + ttl = 30.minutes, + keyToString = (query: HnswANNEngineQuery) => + SimilarityEngine.keyHasher.hashKey(query.cacheKey.getBytes).toString + )) + ) + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/TweetBasedUnifiedSimilarityEngineModule.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/TweetBasedUnifiedSimilarityEngineModule.scala new file mode 100644 index 0000000000..aa54bf0711 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/TweetBasedUnifiedSimilarityEngineModule.scala @@ -0,0 +1,83 @@ +package com.twitter.cr_mixer.module.similarity_engine + +import com.google.inject.Provides +import com.twitter.cr_mixer.model.ModuleNames +import com.twitter.cr_mixer.model.TweetWithCandidateGenerationInfo +import com.twitter.cr_mixer.model.TweetWithScore +import com.twitter.cr_mixer.config.TimeoutConfig +import com.twitter.cr_mixer.similarity_engine.HnswANNSimilarityEngine +import com.twitter.cr_mixer.similarity_engine.SimClustersANNSimilarityEngine +import com.twitter.cr_mixer.similarity_engine.SimilarityEngine.GatingConfig +import com.twitter.cr_mixer.similarity_engine.SimilarityEngine.SimilarityEngineConfig +import com.twitter.cr_mixer.similarity_engine.StandardSimilarityEngine +import com.twitter.cr_mixer.similarity_engine.TweetBasedQigSimilarityEngine +import com.twitter.cr_mixer.similarity_engine.TweetBasedUnifiedSimilarityEngine +import com.twitter.cr_mixer.similarity_engine.TweetBasedUserTweetGraphSimilarityEngine +import com.twitter.cr_mixer.similarity_engine.TweetBasedUserVideoGraphSimilarityEngine +import com.twitter.cr_mixer.thriftscala.SimilarityEngineType +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.inject.TwitterModule +import com.twitter.storehaus.ReadableStore +import javax.inject.Named +import javax.inject.Singleton + +object TweetBasedUnifiedSimilarityEngineModule extends TwitterModule { + + @Provides + @Singleton + @Named(ModuleNames.TweetBasedUnifiedSimilarityEngine) + def providesTweetBasedUnifiedSimilarityEngine( + @Named(ModuleNames.TweetBasedUserTweetGraphSimilarityEngine) tweetBasedUserTweetGraphSimilarityEngine: StandardSimilarityEngine[ + TweetBasedUserTweetGraphSimilarityEngine.Query, + TweetWithScore + ], + @Named(ModuleNames.TweetBasedUserVideoGraphSimilarityEngine) tweetBasedUserVideoGraphSimilarityEngine: StandardSimilarityEngine[ + TweetBasedUserVideoGraphSimilarityEngine.Query, + TweetWithScore + ], + @Named(ModuleNames.TweetBasedTwHINANNSimilarityEngine) + tweetBasedTwHINANNSimilarityEngine: HnswANNSimilarityEngine, + @Named(ModuleNames.TweetBasedQigSimilarityEngine) tweetBasedQigSimilarityEngine: StandardSimilarityEngine[ + TweetBasedQigSimilarityEngine.Query, + TweetWithScore + ], + @Named(ModuleNames.SimClustersANNSimilarityEngine) + simClustersANNSimilarityEngine: StandardSimilarityEngine[ + SimClustersANNSimilarityEngine.Query, + TweetWithScore + ], + timeoutConfig: TimeoutConfig, + statsReceiver: StatsReceiver, + ): StandardSimilarityEngine[ + TweetBasedUnifiedSimilarityEngine.Query, + TweetWithCandidateGenerationInfo + ] = { + + val underlyingStore: ReadableStore[TweetBasedUnifiedSimilarityEngine.Query, Seq[ + TweetWithCandidateGenerationInfo + ]] = TweetBasedUnifiedSimilarityEngine( + tweetBasedUserTweetGraphSimilarityEngine, + tweetBasedUserVideoGraphSimilarityEngine, + simClustersANNSimilarityEngine, + tweetBasedQigSimilarityEngine, + tweetBasedTwHINANNSimilarityEngine, + statsReceiver + ) + + new StandardSimilarityEngine[ + TweetBasedUnifiedSimilarityEngine.Query, + TweetWithCandidateGenerationInfo + ]( + implementingStore = underlyingStore, + identifier = SimilarityEngineType.TweetBasedUnifiedSimilarityEngine, + globalStats = statsReceiver, + engineConfig = SimilarityEngineConfig( + timeout = timeoutConfig.similarityEngineTimeout, + gatingConfig = GatingConfig( + deciderConfig = None, + enableFeatureSwitch = None + ) + ) + ) + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/TweetBasedUserAdGraphSimilarityEngineModule.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/TweetBasedUserAdGraphSimilarityEngineModule.scala new file mode 100644 index 0000000000..7288e603ff --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/TweetBasedUserAdGraphSimilarityEngineModule.scala @@ -0,0 +1,91 @@ +package com.twitter.cr_mixer.module.similarity_engine + +import com.google.inject.Provides +import com.twitter.conversions.DurationOps._ +import com.twitter.cr_mixer.config.TimeoutConfig +import com.twitter.cr_mixer.model.ModuleNames +import com.twitter.cr_mixer.model.TweetWithScore +import com.twitter.cr_mixer.param.decider.CrMixerDecider +import com.twitter.cr_mixer.param.decider.DeciderConstants +import com.twitter.cr_mixer.similarity_engine.SimilarityEngine.DeciderConfig +import com.twitter.cr_mixer.similarity_engine.SimilarityEngine.GatingConfig +import com.twitter.cr_mixer.similarity_engine.SimilarityEngine.SimilarityEngineConfig +import com.twitter.cr_mixer.similarity_engine.StandardSimilarityEngine +import com.twitter.cr_mixer.similarity_engine.TweetBasedUserAdGraphSimilarityEngine +import com.twitter.cr_mixer.thriftscala.SimilarityEngineType +import com.twitter.finagle.memcached.{Client => MemcachedClient} +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.hashing.KeyHasher +import com.twitter.hermit.store.common.ObservedMemcachedReadableStore +import com.twitter.inject.TwitterModule +import com.twitter.recos.user_ad_graph.thriftscala.UserAdGraph +import com.twitter.relevance_platform.common.injection.LZ4Injection +import com.twitter.relevance_platform.common.injection.SeqObjectInjection +import com.twitter.simclusters_v2.common.TweetId +import com.twitter.storehaus.ReadableStore +import com.twitter.twistly.thriftscala.TweetRecentEngagedUsers +import javax.inject.Named +import javax.inject.Singleton + +object TweetBasedUserAdGraphSimilarityEngineModule extends TwitterModule { + + private val keyHasher: KeyHasher = KeyHasher.FNV1A_64 + + @Provides + @Singleton + @Named(ModuleNames.TweetBasedUserAdGraphSimilarityEngine) + def providesTweetBasedUserAdGraphSimilarityEngine( + userAdGraphService: UserAdGraph.MethodPerEndpoint, + tweetRecentEngagedUserStore: ReadableStore[TweetId, TweetRecentEngagedUsers], + @Named(ModuleNames.UnifiedCache) crMixerUnifiedCacheClient: MemcachedClient, + timeoutConfig: TimeoutConfig, + statsReceiver: StatsReceiver, + decider: CrMixerDecider + ): StandardSimilarityEngine[ + TweetBasedUserAdGraphSimilarityEngine.Query, + TweetWithScore + ] = { + + val underlyingStore = TweetBasedUserAdGraphSimilarityEngine( + userAdGraphService, + tweetRecentEngagedUserStore, + statsReceiver) + + val memCachedStore: ReadableStore[ + TweetBasedUserAdGraphSimilarityEngine.Query, + Seq[ + TweetWithScore + ] + ] = + ObservedMemcachedReadableStore + .fromCacheClient( + backingStore = underlyingStore, + cacheClient = crMixerUnifiedCacheClient, + ttl = 10.minutes + )( + valueInjection = LZ4Injection.compose(SeqObjectInjection[TweetWithScore]()), + statsReceiver = statsReceiver.scope("tweet_based_user_ad_graph_store_memcache"), + keyToString = { k => + //Example Query CRMixer:TweetBasedUTG:1234567890ABCDEF + f"CRMixer:TweetBasedUAG:${keyHasher.hashKey(k.toString.getBytes)}%X" + } + ) + + new StandardSimilarityEngine[ + TweetBasedUserAdGraphSimilarityEngine.Query, + TweetWithScore + ]( + implementingStore = memCachedStore, + identifier = SimilarityEngineType.TweetBasedUserAdGraph, + globalStats = statsReceiver, + engineConfig = SimilarityEngineConfig( + timeout = timeoutConfig.similarityEngineTimeout, + gatingConfig = GatingConfig( + deciderConfig = + Some(DeciderConfig(decider, DeciderConstants.enableUserAdGraphTrafficDeciderKey)), + enableFeatureSwitch = None + ) + ) + ) + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/TweetBasedUserTweetGraphSimilarityEngineModule.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/TweetBasedUserTweetGraphSimilarityEngineModule.scala new file mode 100644 index 0000000000..a7a3881999 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/TweetBasedUserTweetGraphSimilarityEngineModule.scala @@ -0,0 +1,92 @@ +package com.twitter.cr_mixer.module +package similarity_engine + +import com.google.inject.Provides +import com.twitter.conversions.DurationOps._ +import com.twitter.cr_mixer.model.ModuleNames +import com.twitter.cr_mixer.model.TweetWithScore +import com.twitter.cr_mixer.config.TimeoutConfig +import com.twitter.cr_mixer.param.decider.CrMixerDecider +import com.twitter.cr_mixer.param.decider.DeciderConstants +import com.twitter.cr_mixer.similarity_engine.SimilarityEngine.DeciderConfig +import com.twitter.cr_mixer.similarity_engine.SimilarityEngine.GatingConfig +import com.twitter.cr_mixer.similarity_engine.SimilarityEngine.SimilarityEngineConfig +import com.twitter.cr_mixer.similarity_engine.StandardSimilarityEngine +import com.twitter.cr_mixer.similarity_engine.TweetBasedUserTweetGraphSimilarityEngine +import com.twitter.cr_mixer.thriftscala.SimilarityEngineType +import com.twitter.finagle.memcached.{Client => MemcachedClient} +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.hashing.KeyHasher +import com.twitter.hermit.store.common.ObservedMemcachedReadableStore +import com.twitter.inject.TwitterModule +import com.twitter.recos.user_tweet_graph.thriftscala.UserTweetGraph +import com.twitter.relevance_platform.common.injection.LZ4Injection +import com.twitter.relevance_platform.common.injection.SeqObjectInjection +import com.twitter.simclusters_v2.common.TweetId +import com.twitter.storehaus.ReadableStore +import com.twitter.twistly.thriftscala.TweetRecentEngagedUsers +import javax.inject.Named +import javax.inject.Singleton + +object TweetBasedUserTweetGraphSimilarityEngineModule extends TwitterModule { + + private val keyHasher: KeyHasher = KeyHasher.FNV1A_64 + + @Provides + @Singleton + @Named(ModuleNames.TweetBasedUserTweetGraphSimilarityEngine) + def providesTweetBasedUserTweetGraphSimilarityEngine( + userTweetGraphService: UserTweetGraph.MethodPerEndpoint, + tweetRecentEngagedUserStore: ReadableStore[TweetId, TweetRecentEngagedUsers], + @Named(ModuleNames.UnifiedCache) crMixerUnifiedCacheClient: MemcachedClient, + timeoutConfig: TimeoutConfig, + statsReceiver: StatsReceiver, + decider: CrMixerDecider + ): StandardSimilarityEngine[ + TweetBasedUserTweetGraphSimilarityEngine.Query, + TweetWithScore + ] = { + + val underlyingStore = TweetBasedUserTweetGraphSimilarityEngine( + userTweetGraphService, + tweetRecentEngagedUserStore, + statsReceiver) + + val memCachedStore: ReadableStore[ + TweetBasedUserTweetGraphSimilarityEngine.Query, + Seq[ + TweetWithScore + ] + ] = + ObservedMemcachedReadableStore + .fromCacheClient( + backingStore = underlyingStore, + cacheClient = crMixerUnifiedCacheClient, + ttl = 10.minutes + )( + valueInjection = LZ4Injection.compose(SeqObjectInjection[TweetWithScore]()), + statsReceiver = statsReceiver.scope("tweet_based_user_tweet_graph_store_memcache"), + keyToString = { k => + //Example Query CRMixer:TweetBasedUTG:1234567890ABCDEF + f"CRMixer:TweetBasedUTG:${keyHasher.hashKey(k.toString.getBytes)}%X" + } + ) + + new StandardSimilarityEngine[ + TweetBasedUserTweetGraphSimilarityEngine.Query, + TweetWithScore + ]( + implementingStore = memCachedStore, + identifier = SimilarityEngineType.TweetBasedUserTweetGraph, + globalStats = statsReceiver, + engineConfig = SimilarityEngineConfig( + timeout = timeoutConfig.similarityEngineTimeout, + gatingConfig = GatingConfig( + deciderConfig = + Some(DeciderConfig(decider, DeciderConstants.enableUserTweetGraphTrafficDeciderKey)), + enableFeatureSwitch = None + ) + ) + ) + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/TweetBasedUserVideoGraphSimilarityEngineModule.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/TweetBasedUserVideoGraphSimilarityEngineModule.scala new file mode 100644 index 0000000000..efc354d21f --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/TweetBasedUserVideoGraphSimilarityEngineModule.scala @@ -0,0 +1,92 @@ +package com.twitter.cr_mixer.module.similarity_engine + +import com.google.inject.Provides +import com.twitter.conversions.DurationOps._ +import com.twitter.cr_mixer.config.TimeoutConfig +import com.twitter.cr_mixer.model.ModuleNames +import com.twitter.cr_mixer.model.TweetWithScore +import com.twitter.cr_mixer.param.decider.CrMixerDecider +import com.twitter.cr_mixer.param.decider.DeciderConstants +import com.twitter.cr_mixer.similarity_engine.SimilarityEngine.DeciderConfig +import com.twitter.cr_mixer.similarity_engine.SimilarityEngine.GatingConfig +import com.twitter.cr_mixer.similarity_engine.SimilarityEngine.SimilarityEngineConfig +import com.twitter.cr_mixer.similarity_engine.StandardSimilarityEngine +import com.twitter.cr_mixer.similarity_engine.TweetBasedUserVideoGraphSimilarityEngine +import com.twitter.cr_mixer.thriftscala.SimilarityEngineType +import com.twitter.finagle.memcached.{Client => MemcachedClient} +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.hashing.KeyHasher +import com.twitter.hermit.store.common.ObservedMemcachedReadableStore +import com.twitter.inject.TwitterModule +import com.twitter.recos.user_video_graph.thriftscala.UserVideoGraph +import com.twitter.relevance_platform.common.injection.LZ4Injection +import com.twitter.relevance_platform.common.injection.SeqObjectInjection +import com.twitter.simclusters_v2.common.TweetId +import com.twitter.storehaus.ReadableStore +import com.twitter.twistly.thriftscala.TweetRecentEngagedUsers +import javax.inject.Named +import javax.inject.Singleton + +object TweetBasedUserVideoGraphSimilarityEngineModule extends TwitterModule { + + private val keyHasher: KeyHasher = KeyHasher.FNV1A_64 + + @Provides + @Singleton + @Named(ModuleNames.TweetBasedUserVideoGraphSimilarityEngine) + def providesTweetBasedUserVideoGraphSimilarityEngine( + userVideoGraphService: UserVideoGraph.MethodPerEndpoint, + tweetRecentEngagedUserStore: ReadableStore[TweetId, TweetRecentEngagedUsers], + @Named(ModuleNames.UnifiedCache) crMixerUnifiedCacheClient: MemcachedClient, + timeoutConfig: TimeoutConfig, + statsReceiver: StatsReceiver, + decider: CrMixerDecider + ): StandardSimilarityEngine[ + TweetBasedUserVideoGraphSimilarityEngine.Query, + TweetWithScore + ] = { + + val underlyingStore = + TweetBasedUserVideoGraphSimilarityEngine( + userVideoGraphService, + tweetRecentEngagedUserStore, + statsReceiver) + + val memCachedStore: ReadableStore[ + TweetBasedUserVideoGraphSimilarityEngine.Query, + Seq[ + TweetWithScore + ] + ] = + ObservedMemcachedReadableStore + .fromCacheClient( + backingStore = underlyingStore, + cacheClient = crMixerUnifiedCacheClient, + ttl = 10.minutes + )( + valueInjection = LZ4Injection.compose(SeqObjectInjection[TweetWithScore]()), + statsReceiver = statsReceiver.scope("tweet_based_user_video_graph_store_memcache"), + keyToString = { k => + //Example Query CRMixer:TweetBasedUVG:1234567890ABCDEF + f"CRMixer:TweetBasedUVG:${keyHasher.hashKey(k.toString.getBytes)}%X" + } + ) + + new StandardSimilarityEngine[ + TweetBasedUserVideoGraphSimilarityEngine.Query, + TweetWithScore + ]( + implementingStore = memCachedStore, + identifier = SimilarityEngineType.TweetBasedUserVideoGraph, + globalStats = statsReceiver, + engineConfig = SimilarityEngineConfig( + timeout = timeoutConfig.similarityEngineTimeout, + gatingConfig = GatingConfig( + deciderConfig = + Some(DeciderConfig(decider, DeciderConstants.enableUserVideoGraphTrafficDeciderKey)), + enableFeatureSwitch = None + ) + ) + ) + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/TwhinCollabFilterLookupSimilarityEngineModule.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/TwhinCollabFilterLookupSimilarityEngineModule.scala new file mode 100644 index 0000000000..4f7c909e32 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/TwhinCollabFilterLookupSimilarityEngineModule.scala @@ -0,0 +1,71 @@ +package com.twitter.cr_mixer.module +package similarity_engine + +import com.google.inject.Provides +import com.twitter.cr_mixer.config.TimeoutConfig +import com.twitter.simclusters_v2.common.TweetId +import com.twitter.cr_mixer.model.TweetWithScore +import com.twitter.cr_mixer.model.ModelConfig +import com.twitter.cr_mixer.model.ModuleNames +import com.twitter.cr_mixer.similarity_engine.LookupSimilarityEngine +import com.twitter.cr_mixer.similarity_engine.TwhinCollabFilterSimilarityEngine.Query +import com.twitter.cr_mixer.similarity_engine.SimilarityEngine.GatingConfig +import com.twitter.cr_mixer.similarity_engine.SimilarityEngine.SimilarityEngineConfig +import com.twitter.cr_mixer.similarity_engine.TwhinCollabFilterSimilarityEngine +import com.twitter.cr_mixer.thriftscala.SimilarityEngineType +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.inject.TwitterModule +import com.twitter.storehaus.ReadableStore +import javax.inject.Named +import javax.inject.Singleton + +/** + * TwhinCandidatesLookupSimilarityEngineModule routes the request to the corresponding + * twhin based candidate store which follow the same pattern as TwHIN Collaborative Filtering. + */ + +object TwhinCollabFilterLookupSimilarityEngineModule extends TwitterModule { + @Provides + @Singleton + @Named(ModuleNames.TwhinCollabFilterSimilarityEngine) + def providesTwhinCollabFilterLookupSimilarityEngineModule( + @Named(ModuleNames.TwhinCollabFilterStratoStoreForFollow) + twhinCollabFilterStratoStoreForFollow: ReadableStore[Long, Seq[TweetId]], + @Named(ModuleNames.TwhinCollabFilterStratoStoreForEngagement) + twhinCollabFilterStratoStoreForEngagement: ReadableStore[Long, Seq[TweetId]], + @Named(ModuleNames.TwhinMultiClusterStratoStoreForFollow) + twhinMultiClusterStratoStoreForFollow: ReadableStore[Long, Seq[TweetId]], + @Named(ModuleNames.TwhinMultiClusterStratoStoreForEngagement) + twhinMultiClusterStratoStoreForEngagement: ReadableStore[Long, Seq[TweetId]], + timeoutConfig: TimeoutConfig, + globalStats: StatsReceiver + ): LookupSimilarityEngine[Query, TweetWithScore] = { + val versionedStoreMap = Map( + ModelConfig.TwhinCollabFilterForFollow -> TwhinCollabFilterSimilarityEngine( + twhinCollabFilterStratoStoreForFollow, + globalStats), + ModelConfig.TwhinCollabFilterForEngagement -> TwhinCollabFilterSimilarityEngine( + twhinCollabFilterStratoStoreForEngagement, + globalStats), + ModelConfig.TwhinMultiClusterForFollow -> TwhinCollabFilterSimilarityEngine( + twhinMultiClusterStratoStoreForFollow, + globalStats), + ModelConfig.TwhinMultiClusterForEngagement -> TwhinCollabFilterSimilarityEngine( + twhinMultiClusterStratoStoreForEngagement, + globalStats), + ) + + new LookupSimilarityEngine[Query, TweetWithScore]( + versionedStoreMap = versionedStoreMap, + identifier = SimilarityEngineType.TwhinCollabFilter, + globalStats = globalStats, + engineConfig = SimilarityEngineConfig( + timeout = timeoutConfig.similarityEngineTimeout, + gatingConfig = GatingConfig( + deciderConfig = None, + enableFeatureSwitch = None + ) + ) + ) + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/UserTweetEntityGraphSimilarityEngineModule.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/UserTweetEntityGraphSimilarityEngineModule.scala new file mode 100644 index 0000000000..cf2093208c --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/similarity_engine/UserTweetEntityGraphSimilarityEngineModule.scala @@ -0,0 +1,55 @@ +package com.twitter.cr_mixer.module.similarity_engine + +import com.google.inject.Provides +import com.twitter.cr_mixer.model.ModuleNames +import com.twitter.cr_mixer.model.TweetWithScoreAndSocialProof +import com.twitter.cr_mixer.config.TimeoutConfig +import com.twitter.cr_mixer.param.decider.CrMixerDecider +import com.twitter.cr_mixer.param.decider.DeciderConstants +import com.twitter.cr_mixer.similarity_engine.UserTweetEntityGraphSimilarityEngine +import com.twitter.cr_mixer.similarity_engine.SimilarityEngine.DeciderConfig +import com.twitter.cr_mixer.similarity_engine.SimilarityEngine.GatingConfig +import com.twitter.cr_mixer.similarity_engine.SimilarityEngine.SimilarityEngineConfig +import com.twitter.cr_mixer.similarity_engine.StandardSimilarityEngine +import com.twitter.cr_mixer.thriftscala.SimilarityEngineType +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.inject.TwitterModule +import com.twitter.recos.user_tweet_entity_graph.thriftscala.UserTweetEntityGraph +import javax.inject.Named +import javax.inject.Singleton + +object UserTweetEntityGraphSimilarityEngineModule extends TwitterModule { + + @Provides + @Singleton + @Named(ModuleNames.UserTweetEntityGraphSimilarityEngine) + def providesUserTweetEntityGraphSimilarityEngine( + userTweetEntityGraphService: UserTweetEntityGraph.MethodPerEndpoint, + timeoutConfig: TimeoutConfig, + statsReceiver: StatsReceiver, + decider: CrMixerDecider + ): StandardSimilarityEngine[ + UserTweetEntityGraphSimilarityEngine.Query, + TweetWithScoreAndSocialProof + ] = { + new StandardSimilarityEngine[ + UserTweetEntityGraphSimilarityEngine.Query, + TweetWithScoreAndSocialProof + ]( + implementingStore = + UserTweetEntityGraphSimilarityEngine(userTweetEntityGraphService, statsReceiver), + identifier = SimilarityEngineType.Uteg, + globalStats = statsReceiver, + engineConfig = SimilarityEngineConfig( + timeout = timeoutConfig.utegSimilarityEngineTimeout, + gatingConfig = GatingConfig( + deciderConfig = Some( + DeciderConfig(decider, DeciderConstants.enableUserTweetEntityGraphTrafficDeciderKey)), + enableFeatureSwitch = None + ) + ), + // We cannot use the key to cache anything in UTEG because the key contains a long list of userIds + memCacheConfig = None + ) + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/thrift_client/AnnQueryServiceClientModule.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/thrift_client/AnnQueryServiceClientModule.scala new file mode 100644 index 0000000000..17dbfcae58 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/thrift_client/AnnQueryServiceClientModule.scala @@ -0,0 +1,107 @@ +package com.twitter.cr_mixer.module.thrift_client + +import com.google.inject.Provides +import com.twitter.ann.common.thriftscala.AnnQueryService +import com.twitter.conversions.DurationOps._ +import com.twitter.conversions.PercentOps._ +import com.twitter.cr_mixer.config.TimeoutConfig +import com.twitter.finagle.ThriftMux +import com.twitter.finagle.mtls.authentication.ServiceIdentifier +import com.twitter.finagle.mtls.client.MtlsStackClient._ +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.finagle.thrift.ClientId +import com.twitter.inject.TwitterModule +import javax.inject.Named +import javax.inject.Singleton + +object AnnQueryServiceClientModule extends TwitterModule { + final val DebuggerDemoAnnServiceClientName = "DebuggerDemoAnnServiceClient" + + @Provides + @Singleton + @Named(DebuggerDemoAnnServiceClientName) + def debuggerDemoAnnServiceClient( + serviceIdentifier: ServiceIdentifier, + clientId: ClientId, + statsReceiver: StatsReceiver, + timeoutConfig: TimeoutConfig, + ): AnnQueryService.MethodPerEndpoint = { + // This ANN is built from the embeddings in src/scala/com/twitter/wtf/beam/bq_embedding_export/sql/MlfExperimentalTweetEmbeddingScalaDataset.sql + // Change the above sql if you want to build the index from a diff embedding + val dest = "/s/cassowary/mlf-experimental-ann-service" + val label = "experimental-ann" + buildClient(serviceIdentifier, clientId, timeoutConfig, statsReceiver, dest, label) + } + + final val TwHINUuaAnnServiceClientName = "TwHINUuaAnnServiceClient" + @Provides + @Singleton + @Named(TwHINUuaAnnServiceClientName) + def twhinUuaAnnServiceClient( + serviceIdentifier: ServiceIdentifier, + clientId: ClientId, + statsReceiver: StatsReceiver, + timeoutConfig: TimeoutConfig, + ): AnnQueryService.MethodPerEndpoint = { + val dest = "/s/cassowary/twhin-uua-ann-service" + val label = "twhin_uua_ann" + + buildClient(serviceIdentifier, clientId, timeoutConfig, statsReceiver, dest, label) + } + + final val TwHINRegularUpdateAnnServiceClientName = "TwHINRegularUpdateAnnServiceClient" + @Provides + @Singleton + @Named(TwHINRegularUpdateAnnServiceClientName) + def twHINRegularUpdateAnnServiceClient( + serviceIdentifier: ServiceIdentifier, + clientId: ClientId, + statsReceiver: StatsReceiver, + timeoutConfig: TimeoutConfig, + ): AnnQueryService.MethodPerEndpoint = { + val dest = "/s/cassowary/twhin-regular-update-ann-service" + val label = "twhin_regular_update" + + buildClient(serviceIdentifier, clientId, timeoutConfig, statsReceiver, dest, label) + } + + final val TwoTowerFavAnnServiceClientName = "TwoTowerFavAnnServiceClient" + @Provides + @Singleton + @Named(TwoTowerFavAnnServiceClientName) + def twoTowerFavAnnServiceClient( + serviceIdentifier: ServiceIdentifier, + clientId: ClientId, + statsReceiver: StatsReceiver, + timeoutConfig: TimeoutConfig, + ): AnnQueryService.MethodPerEndpoint = { + val dest = "/s/cassowary/tweet-rec-two-tower-fav-ann" + val label = "tweet_rec_two_tower_fav_ann" + + buildClient(serviceIdentifier, clientId, timeoutConfig, statsReceiver, dest, label) + } + + private def buildClient( + serviceIdentifier: ServiceIdentifier, + clientId: ClientId, + timeoutConfig: TimeoutConfig, + statsReceiver: StatsReceiver, + dest: String, + label: String + ): AnnQueryService.MethodPerEndpoint = { + val thriftClient = ThriftMux.client + .withMutualTls(serviceIdentifier) + .withClientId(clientId) + .withLabel(label) + .withStatsReceiver(statsReceiver) + .withTransport.connectTimeout(500.milliseconds) + .withSession.acquisitionTimeout(500.milliseconds) + .methodBuilder(dest) + .withTimeoutPerRequest(timeoutConfig.annServiceClientTimeout) + .withRetryDisabled + .idempotent(5.percent) + .servicePerEndpoint[AnnQueryService.ServicePerEndpoint] + + ThriftMux.Client.methodPerEndpoint(thriftClient) + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/thrift_client/EarlybirdSearchClientModule.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/thrift_client/EarlybirdSearchClientModule.scala new file mode 100644 index 0000000000..c399a5a373 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/thrift_client/EarlybirdSearchClientModule.scala @@ -0,0 +1,39 @@ +package com.twitter.cr_mixer.module.thrift_client +import com.twitter.app.Flag +import com.twitter.finagle.ThriftMux +import com.twitter.finatra.mtls.thriftmux.modules.MtlsClient +import com.twitter.inject.thrift.modules.ThriftMethodBuilderClientModule +import com.twitter.search.earlybird.thriftscala.EarlybirdService +import com.twitter.inject.Injector +import com.twitter.conversions.DurationOps._ +import com.twitter.cr_mixer.module.core.TimeoutConfigModule.EarlybirdClientTimeoutFlagName +import com.twitter.finagle.service.RetryBudget +import com.twitter.util.Duration +import org.apache.thrift.protocol.TCompactProtocol + +object EarlybirdSearchClientModule + extends ThriftMethodBuilderClientModule[ + EarlybirdService.ServicePerEndpoint, + EarlybirdService.MethodPerEndpoint + ] + with MtlsClient { + + override def label: String = "earlybird" + override def dest: String = "/s/earlybird-root-superroot/root-superroot" + private val requestTimeoutFlag: Flag[Duration] = + flag[Duration](EarlybirdClientTimeoutFlagName, "Earlybird client timeout") + override protected def requestTimeout: Duration = requestTimeoutFlag() + + override def retryBudget: RetryBudget = RetryBudget.Empty + + override def configureThriftMuxClient( + injector: Injector, + client: ThriftMux.Client + ): ThriftMux.Client = { + super + .configureThriftMuxClient(injector, client) + .withProtocolFactory(new TCompactProtocol.Factory()) + .withSessionQualifier + .successRateFailureAccrual(successRate = 0.9, window = 30.seconds) + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/thrift_client/FrsClientModule.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/thrift_client/FrsClientModule.scala new file mode 100644 index 0000000000..1084f2c1a2 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/thrift_client/FrsClientModule.scala @@ -0,0 +1,41 @@ +package com.twitter.cr_mixer.module.thrift_client + +import com.twitter.app.Flag +import com.twitter.finagle.ThriftMux +import com.twitter.conversions.DurationOps._ +import com.twitter.cr_mixer.module.core.TimeoutConfigModule.FrsClientTimeoutFlagName +import com.twitter.finagle.service.RetryBudget +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.finatra.mtls.thriftmux.modules.MtlsClient +import com.twitter.follow_recommendations.thriftscala.FollowRecommendationsThriftService +import com.twitter.inject.Injector +import com.twitter.inject.thrift.modules.ThriftMethodBuilderClientModule +import com.twitter.util.Duration + +object FrsClientModule + extends ThriftMethodBuilderClientModule[ + FollowRecommendationsThriftService.ServicePerEndpoint, + FollowRecommendationsThriftService.MethodPerEndpoint + ] + with MtlsClient { + + override def label: String = "follow-recommendations-service" + override def dest: String = "/s/follow-recommendations/follow-recos-service" + + private val frsSignalFetchTimeout: Flag[Duration] = + flag[Duration](FrsClientTimeoutFlagName, "FRS signal fetch client timeout") + override def requestTimeout: Duration = frsSignalFetchTimeout() + + override def retryBudget: RetryBudget = RetryBudget.Empty + + override def configureThriftMuxClient( + injector: Injector, + client: ThriftMux.Client + ): ThriftMux.Client = { + super + .configureThriftMuxClient(injector, client) + .withStatsReceiver(injector.instance[StatsReceiver].scope("clnt")) + .withSessionQualifier + .successRateFailureAccrual(successRate = 0.9, window = 30.seconds) + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/thrift_client/HydraPartitionClientModule.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/thrift_client/HydraPartitionClientModule.scala new file mode 100644 index 0000000000..c208e111c0 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/thrift_client/HydraPartitionClientModule.scala @@ -0,0 +1,25 @@ +package com.twitter.cr_mixer.module.thrift_client + +import com.twitter.conversions.DurationOps._ +import com.twitter.finagle.thriftmux.MethodBuilder +import com.twitter.finatra.mtls.thriftmux.modules.MtlsClient +import com.twitter.inject.Injector +import com.twitter.inject.thrift.modules.ThriftMethodBuilderClientModule +import com.twitter.hydra.partition.{thriftscala => ht} + +object HydraPartitionClientModule + extends ThriftMethodBuilderClientModule[ + ht.HydraPartition.ServicePerEndpoint, + ht.HydraPartition.MethodPerEndpoint + ] + with MtlsClient { + override def label: String = "hydra-partition" + + override def dest: String = "/s/hydra/hydra-partition" + + override protected def configureMethodBuilder( + injector: Injector, + methodBuilder: MethodBuilder + ): MethodBuilder = methodBuilder.withTimeoutTotal(500.milliseconds) + +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/thrift_client/HydraRootClientModule.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/thrift_client/HydraRootClientModule.scala new file mode 100644 index 0000000000..28d5b1767c --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/thrift_client/HydraRootClientModule.scala @@ -0,0 +1,25 @@ +package com.twitter.cr_mixer.module.thrift_client + +import com.twitter.conversions.DurationOps._ +import com.twitter.finagle.thriftmux.MethodBuilder +import com.twitter.finatra.mtls.thriftmux.modules.MtlsClient +import com.twitter.hydra.root.{thriftscala => ht} +import com.twitter.inject.Injector +import com.twitter.inject.thrift.modules.ThriftMethodBuilderClientModule + +object HydraRootClientModule + extends ThriftMethodBuilderClientModule[ + ht.HydraRoot.ServicePerEndpoint, + ht.HydraRoot.MethodPerEndpoint + ] + with MtlsClient { + override def label: String = "hydra-root" + + override def dest: String = "/s/hydra/hydra-root" + + override protected def configureMethodBuilder( + injector: Injector, + methodBuilder: MethodBuilder + ): MethodBuilder = methodBuilder.withTimeoutTotal(500.milliseconds) + +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/thrift_client/QigServiceClientModule.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/thrift_client/QigServiceClientModule.scala new file mode 100644 index 0000000000..86675e3499 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/thrift_client/QigServiceClientModule.scala @@ -0,0 +1,40 @@ +package com.twitter.cr_mixer.module.thrift_client + +import com.twitter.app.Flag +import com.twitter.cr_mixer.module.core.TimeoutConfigModule.QigRankerClientTimeoutFlagName +import com.twitter.finagle.ThriftMux +import com.twitter.finagle.mux.ClientDiscardedRequestException +import com.twitter.finagle.service.ReqRep +import com.twitter.finagle.service.ResponseClass +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.finatra.mtls.thriftmux.modules.MtlsClient +import com.twitter.inject.Injector +import com.twitter.inject.thrift.modules.ThriftMethodBuilderClientModule +import com.twitter.qig_ranker.thriftscala.QigRanker +import com.twitter.util.Duration +import com.twitter.util.Throw + +object QigServiceClientModule + extends ThriftMethodBuilderClientModule[ + QigRanker.ServicePerEndpoint, + QigRanker.MethodPerEndpoint + ] + with MtlsClient { + override val label: String = "qig-ranker" + override val dest: String = "/s/qig-shared/qig-ranker" + private val qigRankerClientTimeout: Flag[Duration] = + flag[Duration](QigRankerClientTimeoutFlagName, "ranking timeout") + + override def requestTimeout: Duration = qigRankerClientTimeout() + + override def configureThriftMuxClient( + injector: Injector, + client: ThriftMux.Client + ): ThriftMux.Client = + super + .configureThriftMuxClient(injector, client) + .withStatsReceiver(injector.instance[StatsReceiver].scope("clnt")) + .withResponseClassifier { + case ReqRep(_, Throw(_: ClientDiscardedRequestException)) => ResponseClass.Ignorable + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/thrift_client/SimClustersAnnServiceClientModule.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/thrift_client/SimClustersAnnServiceClientModule.scala new file mode 100644 index 0000000000..7504ab6c35 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/thrift_client/SimClustersAnnServiceClientModule.scala @@ -0,0 +1,147 @@ +package com.twitter.cr_mixer.module.thrift_client + +import com.google.inject.Provides +import com.twitter.conversions.PercentOps._ +import com.twitter.cr_mixer.model.ModuleNames +import com.twitter.cr_mixer.config.TimeoutConfig +import com.twitter.finagle.ThriftMux +import com.twitter.finagle.mtls.authentication.ServiceIdentifier +import com.twitter.finagle.mtls.client.MtlsStackClient._ +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.finagle.thrift.ClientId +import com.twitter.inject.TwitterModule +import com.twitter.simclustersann.{thriftscala => t} +import javax.inject.Named +import javax.inject.Singleton + +object SimClustersAnnServiceClientModule extends TwitterModule { + + @Provides + @Singleton + @Named(ModuleNames.ProdSimClustersANNServiceClientName) + def providesProdSimClustersANNServiceClient( + serviceIdentifier: ServiceIdentifier, + clientId: ClientId, + timeoutConfig: TimeoutConfig, + statsReceiver: StatsReceiver, + ): t.SimClustersANNService.MethodPerEndpoint = { + val label = "simclusters-ann-server" + val dest = "/s/simclusters-ann/simclusters-ann" + + buildClient(serviceIdentifier, clientId, timeoutConfig, statsReceiver, dest, label) + } + + @Provides + @Singleton + @Named(ModuleNames.ExperimentalSimClustersANNServiceClientName) + def providesExperimentalSimClustersANNServiceClient( + serviceIdentifier: ServiceIdentifier, + clientId: ClientId, + timeoutConfig: TimeoutConfig, + statsReceiver: StatsReceiver, + ): t.SimClustersANNService.MethodPerEndpoint = { + val label = "simclusters-ann-experimental-server" + val dest = "/s/simclusters-ann/simclusters-ann-experimental" + + buildClient(serviceIdentifier, clientId, timeoutConfig, statsReceiver, dest, label) + } + + @Provides + @Singleton + @Named(ModuleNames.SimClustersANNServiceClientName1) + def providesSimClustersANNServiceClient1( + serviceIdentifier: ServiceIdentifier, + clientId: ClientId, + timeoutConfig: TimeoutConfig, + statsReceiver: StatsReceiver, + ): t.SimClustersANNService.MethodPerEndpoint = { + val label = "simclusters-ann-server-1" + val dest = "/s/simclusters-ann/simclusters-ann-1" + + buildClient(serviceIdentifier, clientId, timeoutConfig, statsReceiver, dest, label) + } + + @Provides + @Singleton + @Named(ModuleNames.SimClustersANNServiceClientName2) + def providesSimClustersANNServiceClient2( + serviceIdentifier: ServiceIdentifier, + clientId: ClientId, + timeoutConfig: TimeoutConfig, + statsReceiver: StatsReceiver, + ): t.SimClustersANNService.MethodPerEndpoint = { + val label = "simclusters-ann-server-2" + val dest = "/s/simclusters-ann/simclusters-ann-2" + + buildClient(serviceIdentifier, clientId, timeoutConfig, statsReceiver, dest, label) + } + + @Provides + @Singleton + @Named(ModuleNames.SimClustersANNServiceClientName3) + def providesSimClustersANNServiceClient3( + serviceIdentifier: ServiceIdentifier, + clientId: ClientId, + timeoutConfig: TimeoutConfig, + statsReceiver: StatsReceiver, + ): t.SimClustersANNService.MethodPerEndpoint = { + val label = "simclusters-ann-server-3" + val dest = "/s/simclusters-ann/simclusters-ann-3" + + buildClient(serviceIdentifier, clientId, timeoutConfig, statsReceiver, dest, label) + } + + @Provides + @Singleton + @Named(ModuleNames.SimClustersANNServiceClientName5) + def providesSimClustersANNServiceClient5( + serviceIdentifier: ServiceIdentifier, + clientId: ClientId, + timeoutConfig: TimeoutConfig, + statsReceiver: StatsReceiver, + ): t.SimClustersANNService.MethodPerEndpoint = { + val label = "simclusters-ann-server-5" + val dest = "/s/simclusters-ann/simclusters-ann-5" + + buildClient(serviceIdentifier, clientId, timeoutConfig, statsReceiver, dest, label) + } + + @Provides + @Singleton + @Named(ModuleNames.SimClustersANNServiceClientName4) + def providesSimClustersANNServiceClient4( + serviceIdentifier: ServiceIdentifier, + clientId: ClientId, + timeoutConfig: TimeoutConfig, + statsReceiver: StatsReceiver, + ): t.SimClustersANNService.MethodPerEndpoint = { + val label = "simclusters-ann-server-4" + val dest = "/s/simclusters-ann/simclusters-ann-4" + + buildClient(serviceIdentifier, clientId, timeoutConfig, statsReceiver, dest, label) + } + private def buildClient( + serviceIdentifier: ServiceIdentifier, + clientId: ClientId, + timeoutConfig: TimeoutConfig, + statsReceiver: StatsReceiver, + dest: String, + label: String + ): t.SimClustersANNService.MethodPerEndpoint = { + val stats = statsReceiver.scope("clnt") + + val thriftClient = ThriftMux.client + .withMutualTls(serviceIdentifier) + .withClientId(clientId) + .withLabel(label) + .withStatsReceiver(stats) + .methodBuilder(dest) + .idempotent(5.percent) + .withTimeoutPerRequest(timeoutConfig.annServiceClientTimeout) + .withRetryDisabled + .servicePerEndpoint[t.SimClustersANNService.ServicePerEndpoint] + + ThriftMux.Client.methodPerEndpoint(thriftClient) + } + +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/thrift_client/TweetyPieClientModule.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/thrift_client/TweetyPieClientModule.scala new file mode 100644 index 0000000000..610ccc95a3 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/thrift_client/TweetyPieClientModule.scala @@ -0,0 +1,60 @@ +package com.twitter.cr_mixer.module.thrift_client + +import com.google.inject.Provides +import com.twitter.app.Flag +import com.twitter.conversions.DurationOps.richDurationFromInt +import com.twitter.cr_mixer.module.core.TimeoutConfigModule.TweetypieClientTimeoutFlagName +import com.twitter.finagle.ThriftMux +import com.twitter.finagle.mux.ClientDiscardedRequestException +import com.twitter.finagle.service.ReqRep +import com.twitter.finagle.service.ResponseClass +import com.twitter.finagle.service.RetryBudget +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.finatra.mtls.thriftmux.modules.MtlsClient +import com.twitter.inject.Injector +import com.twitter.inject.thrift.modules.ThriftMethodBuilderClientModule +import com.twitter.stitch.tweetypie.{TweetyPie => STweetyPie} +import com.twitter.tweetypie.thriftscala.TweetService +import com.twitter.util.Duration +import com.twitter.util.Throw +import javax.inject.Singleton + +object TweetyPieClientModule + extends ThriftMethodBuilderClientModule[ + TweetService.ServicePerEndpoint, + TweetService.MethodPerEndpoint + ] + with MtlsClient { + + override val label = "tweetypie" + override val dest = "/s/tweetypie/tweetypie" + + private val tweetypieClientTimeout: Flag[Duration] = + flag[Duration](TweetypieClientTimeoutFlagName, "tweetypie client timeout") + override def requestTimeout: Duration = tweetypieClientTimeout() + + override def retryBudget: RetryBudget = RetryBudget.Empty + + // We bump the success rate from the default of 0.8 to 0.9 since we're dropping the + // consecutive failures part of the default policy. + override def configureThriftMuxClient( + injector: Injector, + client: ThriftMux.Client + ): ThriftMux.Client = + super + .configureThriftMuxClient(injector, client) + .withStatsReceiver(injector.instance[StatsReceiver].scope("clnt")) + .withSessionQualifier + .successRateFailureAccrual(successRate = 0.9, window = 30.seconds) + .withResponseClassifier { + case ReqRep(_, Throw(_: ClientDiscardedRequestException)) => ResponseClass.Ignorable + } + + @Provides + @Singleton + def providesTweetyPie( + tweetyPieService: TweetService.MethodPerEndpoint + ): STweetyPie = { + STweetyPie(tweetyPieService) + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/thrift_client/UserAdGraphClientModule.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/thrift_client/UserAdGraphClientModule.scala new file mode 100644 index 0000000000..4c1f337ab9 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/thrift_client/UserAdGraphClientModule.scala @@ -0,0 +1,47 @@ +package com.twitter.cr_mixer.module.thrift_client + +import com.twitter.app.Flag +import com.twitter.cr_mixer.module.core.TimeoutConfigModule.UserAdGraphClientTimeoutFlagName +import com.twitter.finagle.ThriftMux +import com.twitter.finagle.mtls.authentication.ServiceIdentifier +import com.twitter.finagle.mtls.client.MtlsStackClient.MtlsThriftMuxClientSyntax +import com.twitter.finagle.mux.ClientDiscardedRequestException +import com.twitter.finagle.service.ReqRep +import com.twitter.finagle.service.ResponseClass +import com.twitter.finagle.service.RetryBudget +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.finatra.mtls.thriftmux.modules.MtlsClient +import com.twitter.inject.Injector +import com.twitter.inject.thrift.modules.ThriftMethodBuilderClientModule +import com.twitter.recos.user_ad_graph.thriftscala.UserAdGraph +import com.twitter.util.Duration +import com.twitter.util.Throw + +object UserAdGraphClientModule + extends ThriftMethodBuilderClientModule[ + UserAdGraph.ServicePerEndpoint, + UserAdGraph.MethodPerEndpoint + ] + with MtlsClient { + + override val label = "user-ad-graph" + override val dest = "/s/user-tweet-graph/user-ad-graph" + private val userAdGraphClientTimeout: Flag[Duration] = + flag[Duration](UserAdGraphClientTimeoutFlagName, "userAdGraph client timeout") + override def requestTimeout: Duration = userAdGraphClientTimeout() + + override def retryBudget: RetryBudget = RetryBudget.Empty + + override def configureThriftMuxClient( + injector: Injector, + client: ThriftMux.Client + ): ThriftMux.Client = + super + .configureThriftMuxClient(injector, client) + .withMutualTls(injector.instance[ServiceIdentifier]) + .withStatsReceiver(injector.instance[StatsReceiver].scope("clnt")) + .withResponseClassifier { + case ReqRep(_, Throw(_: ClientDiscardedRequestException)) => ResponseClass.Ignorable + } + +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/thrift_client/UserTweetEntityGraphClientModule.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/thrift_client/UserTweetEntityGraphClientModule.scala new file mode 100644 index 0000000000..337f943f72 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/thrift_client/UserTweetEntityGraphClientModule.scala @@ -0,0 +1,44 @@ +package com.twitter.cr_mixer.module.thrift_client + +import com.twitter.app.Flag +import com.twitter.cr_mixer.module.core.TimeoutConfigModule.UtegClientTimeoutFlagName +import com.twitter.finagle.ThriftMux +import com.twitter.finagle.mux.ClientDiscardedRequestException +import com.twitter.finagle.service.ReqRep +import com.twitter.finagle.service.ResponseClass +import com.twitter.finagle.service.RetryBudget +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.finatra.mtls.thriftmux.modules.MtlsClient +import com.twitter.inject.Injector +import com.twitter.inject.thrift.modules.ThriftMethodBuilderClientModule +import com.twitter.recos.user_tweet_entity_graph.thriftscala.UserTweetEntityGraph +import com.twitter.util.Duration +import com.twitter.util.Throw + +object UserTweetEntityGraphClientModule + extends ThriftMethodBuilderClientModule[ + UserTweetEntityGraph.ServicePerEndpoint, + UserTweetEntityGraph.MethodPerEndpoint + ] + with MtlsClient { + + override val label = "user-tweet-entity-graph" + override val dest = "/s/cassowary/user_tweet_entity_graph" + private val userTweetEntityGraphClientTimeout: Flag[Duration] = + flag[Duration](UtegClientTimeoutFlagName, "user tweet entity graph client timeout") + override def requestTimeout: Duration = userTweetEntityGraphClientTimeout() + + override def retryBudget: RetryBudget = RetryBudget.Empty + + override def configureThriftMuxClient( + injector: Injector, + client: ThriftMux.Client + ): ThriftMux.Client = + super + .configureThriftMuxClient(injector, client) + .withStatsReceiver(injector.instance[StatsReceiver].scope("clnt")) + .withResponseClassifier { + case ReqRep(_, Throw(_: ClientDiscardedRequestException)) => ResponseClass.Ignorable + } + +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/thrift_client/UserTweetGraphClientModule.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/thrift_client/UserTweetGraphClientModule.scala new file mode 100644 index 0000000000..572786fd1c --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/thrift_client/UserTweetGraphClientModule.scala @@ -0,0 +1,43 @@ +package com.twitter.cr_mixer.module.thrift_client + +import com.twitter.app.Flag +import com.twitter.finagle.ThriftMux +import com.twitter.finagle.mux.ClientDiscardedRequestException +import com.twitter.finagle.service.ReqRep +import com.twitter.finagle.service.ResponseClass +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.finatra.mtls.thriftmux.modules.MtlsClient +import com.twitter.inject.Injector +import com.twitter.inject.thrift.modules.ThriftMethodBuilderClientModule +import com.twitter.recos.user_tweet_graph.thriftscala.UserTweetGraph +import com.twitter.util.Duration +import com.twitter.util.Throw +import com.twitter.cr_mixer.module.core.TimeoutConfigModule.UserTweetGraphClientTimeoutFlagName +import com.twitter.finagle.service.RetryBudget + +object UserTweetGraphClientModule + extends ThriftMethodBuilderClientModule[ + UserTweetGraph.ServicePerEndpoint, + UserTweetGraph.MethodPerEndpoint + ] + with MtlsClient { + + override val label = "user-tweet-graph" + override val dest = "/s/user-tweet-graph/user-tweet-graph" + private val userTweetGraphClientTimeout: Flag[Duration] = + flag[Duration](UserTweetGraphClientTimeoutFlagName, "userTweetGraph client timeout") + override def requestTimeout: Duration = userTweetGraphClientTimeout() + + override def retryBudget: RetryBudget = RetryBudget.Empty + + override def configureThriftMuxClient( + injector: Injector, + client: ThriftMux.Client + ): ThriftMux.Client = + super + .configureThriftMuxClient(injector, client) + .withStatsReceiver(injector.instance[StatsReceiver].scope("clnt")) + .withResponseClassifier { + case ReqRep(_, Throw(_: ClientDiscardedRequestException)) => ResponseClass.Ignorable + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/thrift_client/UserTweetGraphPlusClientModule.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/thrift_client/UserTweetGraphPlusClientModule.scala new file mode 100644 index 0000000000..41ae96e539 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/thrift_client/UserTweetGraphPlusClientModule.scala @@ -0,0 +1,46 @@ +package com.twitter.cr_mixer.module.thrift_client + +import com.twitter.app.Flag +import com.twitter.cr_mixer.module.core.TimeoutConfigModule.UserTweetGraphPlusClientTimeoutFlagName +import com.twitter.finagle.ThriftMux +import com.twitter.finagle.mux.ClientDiscardedRequestException +import com.twitter.finagle.service.ReqRep +import com.twitter.finagle.service.ResponseClass +import com.twitter.finagle.service.RetryBudget +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.finatra.mtls.thriftmux.modules.MtlsClient +import com.twitter.inject.Injector +import com.twitter.inject.thrift.modules.ThriftMethodBuilderClientModule +import com.twitter.recos.user_tweet_graph_plus.thriftscala.UserTweetGraphPlus +import com.twitter.util.Duration +import com.twitter.util.Throw + +object UserTweetGraphPlusClientModule + extends ThriftMethodBuilderClientModule[ + UserTweetGraphPlus.ServicePerEndpoint, + UserTweetGraphPlus.MethodPerEndpoint + ] + with MtlsClient { + + override val label = "user-tweet-graph-plus" + override val dest = "/s/user-tweet-graph/user-tweet-graph-plus" + private val userTweetGraphPlusClientTimeout: Flag[Duration] = + flag[Duration]( + UserTweetGraphPlusClientTimeoutFlagName, + "userTweetGraphPlus client timeout" + ) + override def requestTimeout: Duration = userTweetGraphPlusClientTimeout() + + override def retryBudget: RetryBudget = RetryBudget.Empty + + override def configureThriftMuxClient( + injector: Injector, + client: ThriftMux.Client + ): ThriftMux.Client = + super + .configureThriftMuxClient(injector, client) + .withStatsReceiver(injector.instance[StatsReceiver].scope("clnt")) + .withResponseClassifier { + case ReqRep(_, Throw(_: ClientDiscardedRequestException)) => ResponseClass.Ignorable + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/thrift_client/UserVideoGraphClientModule.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/thrift_client/UserVideoGraphClientModule.scala new file mode 100644 index 0000000000..7c311cbfac --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/module/thrift_client/UserVideoGraphClientModule.scala @@ -0,0 +1,46 @@ +package com.twitter.cr_mixer.module.thrift_client + +import com.twitter.app.Flag +import com.twitter.cr_mixer.module.core.TimeoutConfigModule.UserVideoGraphClientTimeoutFlagName +import com.twitter.finagle.ThriftMux +import com.twitter.finagle.mux.ClientDiscardedRequestException +import com.twitter.finagle.service.ReqRep +import com.twitter.finagle.service.ResponseClass +import com.twitter.finagle.service.RetryBudget +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.finatra.mtls.thriftmux.modules.MtlsClient +import com.twitter.inject.Injector +import com.twitter.inject.thrift.modules.ThriftMethodBuilderClientModule +import com.twitter.recos.user_video_graph.thriftscala.UserVideoGraph +import com.twitter.util.Duration +import com.twitter.util.Throw + +object UserVideoGraphClientModule + extends ThriftMethodBuilderClientModule[ + UserVideoGraph.ServicePerEndpoint, + UserVideoGraph.MethodPerEndpoint + ] + with MtlsClient { + + override val label = "user-video-graph" + override val dest = "/s/user-tweet-graph/user-video-graph" + private val userVideoGraphClientTimeout: Flag[Duration] = + flag[Duration]( + UserVideoGraphClientTimeoutFlagName, + "userVideoGraph client timeout" + ) + override def requestTimeout: Duration = userVideoGraphClientTimeout() + + override def retryBudget: RetryBudget = RetryBudget.Empty + + override def configureThriftMuxClient( + injector: Injector, + client: ThriftMux.Client + ): ThriftMux.Client = + super + .configureThriftMuxClient(injector, client) + .withStatsReceiver(injector.instance[StatsReceiver].scope("clnt")) + .withResponseClassifier { + case ReqRep(_, Throw(_: ClientDiscardedRequestException)) => ResponseClass.Ignorable + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/AdsParams.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/AdsParams.scala new file mode 100644 index 0000000000..880f1b27c9 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/AdsParams.scala @@ -0,0 +1,64 @@ +package com.twitter.cr_mixer.param + +import com.twitter.timelines.configapi.BaseConfig +import com.twitter.timelines.configapi.BaseConfigBuilder +import com.twitter.timelines.configapi.FSBoundedParam +import com.twitter.timelines.configapi.FSName +import com.twitter.timelines.configapi.FSParam +import com.twitter.timelines.configapi.FeatureSwitchOverrideUtil +import com.twitter.timelines.configapi.Param + +object AdsParams { + object AdsCandidateGenerationMaxCandidatesNumParam + extends FSBoundedParam[Int]( + name = "ads_candidate_generation_max_candidates_num", + default = 400, + min = 0, + max = 2000 + ) + + object EnableScoreBoost + extends FSParam[Boolean]( + name = "ads_candidate_generation_enable_score_boost", + default = false + ) + + object AdsCandidateGenerationScoreBoostFactor + extends FSBoundedParam[Double]( + name = "ads_candidate_generation_score_boost_factor", + default = 10000.0, + min = 1.0, + max = 100000.0 + ) + + object EnableScribe + extends FSParam[Boolean]( + name = "ads_candidate_generation_enable_scribe", + default = false + ) + + val AllParams: Seq[Param[_] with FSName] = Seq( + AdsCandidateGenerationMaxCandidatesNumParam, + EnableScoreBoost, + AdsCandidateGenerationScoreBoostFactor + ) + + lazy val config: BaseConfig = { + val intOverrides = FeatureSwitchOverrideUtil.getBoundedIntFSOverrides( + AdsCandidateGenerationMaxCandidatesNumParam) + + val booleanOverrides = FeatureSwitchOverrideUtil.getBooleanFSOverrides( + EnableScoreBoost, + EnableScribe + ) + + val doubleOverrides = + FeatureSwitchOverrideUtil.getBoundedDoubleFSOverrides(AdsCandidateGenerationScoreBoostFactor) + + BaseConfigBuilder() + .set(intOverrides: _*) + .set(booleanOverrides: _*) + .set(doubleOverrides: _*) + .build() + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/BUILD b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/BUILD new file mode 100644 index 0000000000..b24a213944 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/BUILD @@ -0,0 +1,27 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + strict_deps = True, + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/javax/inject:javax.inject", + "abdecider/src/main/scala", + "configapi/configapi-abdecider", + "configapi/configapi-core", + "configapi/configapi-featureswitches:v2", + "cr-mixer/server/src/main/scala/com/twitter/cr_mixer/model", + "cr-mixer/thrift/src/main/thrift:thrift-scala", + "decider/src/main/scala", + "discovery-common/src/main/scala/com/twitter/discovery/common/configapi", + "featureswitches/featureswitches-core", + "featureswitches/featureswitches-core/src/main/scala/com/twitter/featureswitches/v2/builder", + "finagle-internal/mtls/src/main/scala/com/twitter/finagle/mtls/authentication", + "follow-recommendations-service/thrift/src/main/thrift:thrift-scala", + "product-mixer/core/src/main/thrift/com/twitter/product_mixer/core:thrift-scala", + "scribelib/marshallers/src/main/scala/com/twitter/scribelib/marshallers", + "src/scala/com/twitter/simclusters_v2/common", + "src/thrift/com/twitter/search:earlybird-scala", + "src/thrift/com/twitter/simclusters_v2:simclusters_v2-thrift-scala", + "user-signal-service/thrift/src/main/thrift:thrift-scala", + ], +) diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/BlenderParams.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/BlenderParams.scala new file mode 100644 index 0000000000..185fc4440a --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/BlenderParams.scala @@ -0,0 +1,152 @@ +package com.twitter.cr_mixer.param + +import com.twitter.finagle.stats.NullStatsReceiver +import com.twitter.logging.Logger +import com.twitter.timelines.configapi.BaseConfig +import com.twitter.timelines.configapi.BaseConfigBuilder +import com.twitter.timelines.configapi.FSBoundedParam +import com.twitter.timelines.configapi.FSEnumParam +import com.twitter.timelines.configapi.FSName +import com.twitter.timelines.configapi.FSParam +import com.twitter.timelines.configapi.FeatureSwitchOverrideUtil +import com.twitter.timelines.configapi.Param + +object BlenderParams { + object BlendingAlgorithmEnum extends Enumeration { + val RoundRobin: Value = Value + val SourceTypeBackFill: Value = Value + val SourceSignalSorting: Value = Value + } + object ContentBasedSortingAlgorithmEnum extends Enumeration { + val FavoriteCount: Value = Value + val SourceSignalRecency: Value = Value + val RandomSorting: Value = Value + val SimilarityToSignalSorting: Value = Value + val CandidateRecency: Value = Value + } + + object BlendingAlgorithmParam + extends FSEnumParam[BlendingAlgorithmEnum.type]( + name = "blending_algorithm_id", + default = BlendingAlgorithmEnum.RoundRobin, + enum = BlendingAlgorithmEnum + ) + + object RankingInterleaveWeightShrinkageParam + extends FSBoundedParam[Double]( + name = "blending_enable_ml_ranking_interleave_weights_shrinkage", + default = 1.0, + min = 0.0, + max = 1.0 + ) + + object RankingInterleaveMaxWeightAdjustments + extends FSBoundedParam[Int]( + name = "blending_interleave_max_weighted_adjustments", + default = 3000, + min = 0, + max = 9999 + ) + + object SignalTypeSortingAlgorithmParam + extends FSEnumParam[ContentBasedSortingAlgorithmEnum.type]( + name = "blending_algorithm_inner_signal_sorting_id", + default = ContentBasedSortingAlgorithmEnum.SourceSignalRecency, + enum = ContentBasedSortingAlgorithmEnum + ) + + object ContentBlenderTypeSortingAlgorithmParam + extends FSEnumParam[ContentBasedSortingAlgorithmEnum.type]( + name = "blending_algorithm_content_blender_sorting_id", + default = ContentBasedSortingAlgorithmEnum.FavoriteCount, + enum = ContentBasedSortingAlgorithmEnum + ) + + //UserAffinities Algo Param: whether to distributed the source type weights + object EnableDistributedSourceTypeWeightsParam + extends FSParam[Boolean]( + name = "blending_algorithm_enable_distributed_source_type_weights", + default = false + ) + + object BlendGroupingMethodEnum extends Enumeration { + val SourceKeyDefault: Value = Value("SourceKey") + val SourceTypeSimilarityEngine: Value = Value("SourceTypeSimilarityEngine") + val AuthorId: Value = Value("AuthorId") + } + + object BlendGroupingMethodParam + extends FSEnumParam[BlendGroupingMethodEnum.type]( + name = "blending_grouping_method_id", + default = BlendGroupingMethodEnum.SourceKeyDefault, + enum = BlendGroupingMethodEnum + ) + + object RecencyBasedRandomSamplingHalfLifeInDays + extends FSBoundedParam[Int]( + name = "blending_interleave_random_sampling_recency_based_half_life_in_days", + default = 7, + min = 1, + max = 28 + ) + + object RecencyBasedRandomSamplingDefaultWeight + extends FSBoundedParam[Double]( + name = "blending_interleave_random_sampling_recency_based_default_weight", + default = 1.0, + min = 0.1, + max = 2.0 + ) + + object SourceTypeBackFillEnableVideoBackFill + extends FSParam[Boolean]( + name = "blending_enable_video_backfill", + default = false + ) + + val AllParams: Seq[Param[_] with FSName] = Seq( + BlendingAlgorithmParam, + RankingInterleaveWeightShrinkageParam, + RankingInterleaveMaxWeightAdjustments, + EnableDistributedSourceTypeWeightsParam, + BlendGroupingMethodParam, + RecencyBasedRandomSamplingHalfLifeInDays, + RecencyBasedRandomSamplingDefaultWeight, + SourceTypeBackFillEnableVideoBackFill, + SignalTypeSortingAlgorithmParam, + ContentBlenderTypeSortingAlgorithmParam, + ) + + lazy val config: BaseConfig = { + val enumOverrides = FeatureSwitchOverrideUtil.getEnumFSOverrides( + NullStatsReceiver, + Logger(getClass), + BlendingAlgorithmParam, + BlendGroupingMethodParam, + SignalTypeSortingAlgorithmParam, + ContentBlenderTypeSortingAlgorithmParam + ) + + val booleanOverrides = FeatureSwitchOverrideUtil.getBooleanFSOverrides( + EnableDistributedSourceTypeWeightsParam, + SourceTypeBackFillEnableVideoBackFill + ) + + val intOverrides = FeatureSwitchOverrideUtil.getBoundedIntFSOverrides( + RankingInterleaveMaxWeightAdjustments, + RecencyBasedRandomSamplingHalfLifeInDays + ) + + val doubleOverrides = FeatureSwitchOverrideUtil.getBoundedDoubleFSOverrides( + RankingInterleaveWeightShrinkageParam, + RecencyBasedRandomSamplingDefaultWeight + ) + + BaseConfigBuilder() + .set(enumOverrides: _*) + .set(booleanOverrides: _*) + .set(intOverrides: _*) + .set(doubleOverrides: _*) + .build() + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/BypassInterleaveAndRankParams.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/BypassInterleaveAndRankParams.scala new file mode 100644 index 0000000000..20cbc369af --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/BypassInterleaveAndRankParams.scala @@ -0,0 +1,98 @@ +package com.twitter.cr_mixer.param + +import com.twitter.timelines.configapi.BaseConfig +import com.twitter.timelines.configapi.BaseConfigBuilder +import com.twitter.timelines.configapi.FSBoundedParam +import com.twitter.timelines.configapi.FSName +import com.twitter.timelines.configapi.FSParam +import com.twitter.timelines.configapi.FeatureSwitchOverrideUtil +import com.twitter.timelines.configapi.Param + +object BypassInterleaveAndRankParams { + object EnableTwhinCollabFilterBypassParam + extends FSParam[Boolean]( + name = "bypass_interleave_and_rank_twhin_collab_filter", + default = false + ) + + object EnableTwoTowerBypassParam + extends FSParam[Boolean]( + name = "bypass_interleave_and_rank_two_tower", + default = false + ) + + object EnableConsumerBasedTwhinBypassParam + extends FSParam[Boolean]( + name = "bypass_interleave_and_rank_consumer_based_twhin", + default = false + ) + + object EnableConsumerBasedWalsBypassParam + extends FSParam[Boolean]( + name = "bypass_interleave_and_rank_consumer_based_wals", + default = false + ) + + object TwhinCollabFilterBypassPercentageParam + extends FSBoundedParam[Double]( + name = "bypass_interleave_and_rank_twhin_collab_filter_percentage", + default = 0.0, + min = 0.0, + max = 1.0 + ) + + object TwoTowerBypassPercentageParam + extends FSBoundedParam[Double]( + name = "bypass_interleave_and_rank_two_tower_percentage", + default = 0.0, + min = 0.0, + max = 1.0 + ) + + object ConsumerBasedTwhinBypassPercentageParam + extends FSBoundedParam[Double]( + name = "bypass_interleave_and_rank_consumer_based_twhin_percentage", + default = 0.0, + min = 0.0, + max = 1.0 + ) + + object ConsumerBasedWalsBypassPercentageParam + extends FSBoundedParam[Double]( + name = "bypass_interleave_and_rank_consumer_based_wals_percentage", + default = 0.0, + min = 0.0, + max = 1.0 + ) + + val AllParams: Seq[Param[_] with FSName] = Seq( + EnableTwhinCollabFilterBypassParam, + EnableTwoTowerBypassParam, + EnableConsumerBasedTwhinBypassParam, + EnableConsumerBasedWalsBypassParam, + TwhinCollabFilterBypassPercentageParam, + TwoTowerBypassPercentageParam, + ConsumerBasedTwhinBypassPercentageParam, + ConsumerBasedWalsBypassPercentageParam, + ) + + lazy val config: BaseConfig = { + val booleanOverrides = FeatureSwitchOverrideUtil.getBooleanFSOverrides( + EnableTwhinCollabFilterBypassParam, + EnableTwoTowerBypassParam, + EnableConsumerBasedTwhinBypassParam, + EnableConsumerBasedWalsBypassParam, + ) + + val doubleOverrides = FeatureSwitchOverrideUtil.getBoundedDoubleFSOverrides( + TwhinCollabFilterBypassPercentageParam, + TwoTowerBypassPercentageParam, + ConsumerBasedTwhinBypassPercentageParam, + ConsumerBasedWalsBypassPercentageParam, + ) + BaseConfigBuilder() + .set(booleanOverrides: _*) + .set(doubleOverrides: _*) + .build() + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/ConsumerBasedWalsParams.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/ConsumerBasedWalsParams.scala new file mode 100644 index 0000000000..15f4d36ab4 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/ConsumerBasedWalsParams.scala @@ -0,0 +1,96 @@ +package com.twitter.cr_mixer.param + +import com.twitter.conversions.DurationOps.richDurationFromInt +import com.twitter.timelines.configapi.BaseConfig +import com.twitter.timelines.configapi.BaseConfigBuilder +import com.twitter.timelines.configapi.DurationConversion +import com.twitter.timelines.configapi.FSName +import com.twitter.timelines.configapi.FSBoundedParam +import com.twitter.timelines.configapi.FSParam +import com.twitter.timelines.configapi.FeatureSwitchOverrideUtil +import com.twitter.timelines.configapi.HasDurationConversion +import com.twitter.timelines.configapi.Param +import com.twitter.util.Duration + +object ConsumerBasedWalsParams { + + object EnableSourceParam + extends FSParam[Boolean]( + name = "consumer_based_wals_enable_source", + default = false + ) + + object ModelNameParam + extends FSParam[String]( + name = "consumer_based_wals_model_name", + default = "model_0" + ) + + object WilyNsNameParam + extends FSParam[String]( + name = "consumer_based_wals_wily_ns_name", + default = "" + ) + + object ModelInputNameParam + extends FSParam[String]( + name = "consumer_based_wals_model_input_name", + default = "examples" + ) + + object ModelOutputNameParam + extends FSParam[String]( + name = "consumer_based_wals_model_output_name", + default = "all_tweet_ids" + ) + + object ModelSignatureNameParam + extends FSParam[String]( + name = "consumer_based_wals_model_signature_name", + default = "serving_default" + ) + + object MaxTweetSignalAgeHoursParam + extends FSBoundedParam[Duration]( + name = "consumer_based_wals_max_tweet_signal_age_hours", + default = 72.hours, + min = 1.hours, + max = 720.hours + ) + with HasDurationConversion { + + override val durationConversion: DurationConversion = DurationConversion.FromHours + } + + val AllParams: Seq[Param[_] with FSName] = Seq( + EnableSourceParam, + ModelNameParam, + ModelInputNameParam, + ModelOutputNameParam, + ModelSignatureNameParam, + MaxTweetSignalAgeHoursParam, + WilyNsNameParam, + ) + + lazy val config: BaseConfig = { + val booleanOverrides = FeatureSwitchOverrideUtil.getBooleanFSOverrides( + EnableSourceParam, + ) + val stringOverrides = FeatureSwitchOverrideUtil.getStringFSOverrides( + ModelNameParam, + ModelInputNameParam, + ModelOutputNameParam, + ModelSignatureNameParam, + WilyNsNameParam + ) + + val boundedDurationFSOverrides = + FeatureSwitchOverrideUtil.getBoundedDurationFSOverrides(MaxTweetSignalAgeHoursParam) + + BaseConfigBuilder() + .set(booleanOverrides: _*) + .set(stringOverrides: _*) + .set(boundedDurationFSOverrides: _*) + .build() + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/ConsumerEmbeddingBasedCandidateGenerationParams.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/ConsumerEmbeddingBasedCandidateGenerationParams.scala new file mode 100644 index 0000000000..bedbaf0b92 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/ConsumerEmbeddingBasedCandidateGenerationParams.scala @@ -0,0 +1,55 @@ +package com.twitter.cr_mixer.param + +import com.twitter.timelines.configapi.BaseConfig +import com.twitter.timelines.configapi.BaseConfigBuilder +import com.twitter.timelines.configapi.FSName +import com.twitter.timelines.configapi.FSParam +import com.twitter.timelines.configapi.FeatureSwitchOverrideUtil +import com.twitter.timelines.configapi.Param + +object ConsumerEmbeddingBasedCandidateGenerationParams { + + object EnableTwHINParam + extends FSParam[Boolean]( + name = "consumer_embedding_based_candidate_generation_enable_twhin", + default = false + ) + + object EnableTwoTowerParam + extends FSParam[Boolean]( + name = "consumer_embedding_based_candidate_generation_enable_two_tower", + default = false + ) + + object EnableLogFavBasedSimClustersTripParam + extends FSParam[Boolean]( + name = "consumer_embedding_based_candidate_generation_enable_logfav_based_simclusters_trip", + default = false + ) + + object EnableFollowBasedSimClustersTripParam + extends FSParam[Boolean]( + name = "consumer_embedding_based_candidate_generation_enable_follow_based_simclusters_trip", + default = false + ) + + val AllParams: Seq[Param[_] with FSName] = Seq( + EnableTwHINParam, + EnableTwoTowerParam, + EnableFollowBasedSimClustersTripParam, + EnableLogFavBasedSimClustersTripParam + ) + + lazy val config: BaseConfig = { + val booleanOverrides = FeatureSwitchOverrideUtil.getBooleanFSOverrides( + EnableTwHINParam, + EnableTwoTowerParam, + EnableFollowBasedSimClustersTripParam, + EnableLogFavBasedSimClustersTripParam + ) + + BaseConfigBuilder() + .set(booleanOverrides: _*) + .build() + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/ConsumerEmbeddingBasedTripParams.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/ConsumerEmbeddingBasedTripParams.scala new file mode 100644 index 0000000000..4b43d42ab4 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/ConsumerEmbeddingBasedTripParams.scala @@ -0,0 +1,46 @@ +package com.twitter.cr_mixer.param + +import com.twitter.timelines.configapi.BaseConfig +import com.twitter.timelines.configapi.BaseConfigBuilder +import com.twitter.timelines.configapi.FSBoundedParam +import com.twitter.timelines.configapi.FSName +import com.twitter.timelines.configapi.FSParam +import com.twitter.timelines.configapi.FeatureSwitchOverrideUtil +import com.twitter.timelines.configapi.Param + +object ConsumerEmbeddingBasedTripParams { + object SourceIdParam + extends FSParam[String]( + name = "consumer_embedding_based_trip_source_id", + default = "EXPLR_TOPK_VID_48H_V3") + + object MaxNumCandidatesParam + extends FSBoundedParam[Int]( + name = "consumer_embedding_based_trip_max_num_candidates", + default = 80, + min = 0, + max = 200 + ) + + val AllParams: Seq[Param[_] with FSName] = Seq( + SourceIdParam, + MaxNumCandidatesParam + ) + + lazy val config: BaseConfig = { + val stringFSOverrides = + FeatureSwitchOverrideUtil.getStringFSOverrides( + SourceIdParam + ) + + val intFSOverrides = + FeatureSwitchOverrideUtil.getBoundedIntFSOverrides( + MaxNumCandidatesParam + ) + + BaseConfigBuilder() + .set(stringFSOverrides: _*) + .set(intFSOverrides: _*) + .build() + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/ConsumerEmbeddingBasedTwHINParams.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/ConsumerEmbeddingBasedTwHINParams.scala new file mode 100644 index 0000000000..bda14d5d48 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/ConsumerEmbeddingBasedTwHINParams.scala @@ -0,0 +1,33 @@ +package com.twitter.cr_mixer.param + +import com.twitter.cr_mixer.model.ModelConfig +import com.twitter.timelines.configapi.BaseConfig +import com.twitter.timelines.configapi.BaseConfigBuilder +import com.twitter.timelines.configapi.FSName +import com.twitter.timelines.configapi.FSParam +import com.twitter.timelines.configapi.Param + +import com.twitter.timelines.configapi.FeatureSwitchOverrideUtil + +object ConsumerEmbeddingBasedTwHINParams { + object ModelIdParam + extends FSParam[String]( + name = "consumer_embedding_based_twhin_model_id", + default = ModelConfig.ConsumerBasedTwHINRegularUpdateAll20221024, + ) // Note: this default value does not match with ModelIds yet. This FS is a placeholder + + val AllParams: Seq[Param[_] with FSName] = Seq( + ModelIdParam + ) + + lazy val config: BaseConfig = { + val stringFSOverrides = + FeatureSwitchOverrideUtil.getStringFSOverrides( + ModelIdParam + ) + + BaseConfigBuilder() + .set(stringFSOverrides: _*) + .build() + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/ConsumerEmbeddingBasedTwoTowerParams.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/ConsumerEmbeddingBasedTwoTowerParams.scala new file mode 100644 index 0000000000..2a6474adcf --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/ConsumerEmbeddingBasedTwoTowerParams.scala @@ -0,0 +1,32 @@ +package com.twitter.cr_mixer.param + +import com.twitter.cr_mixer.model.ModelConfig.TwoTowerFavALL20220808 +import com.twitter.timelines.configapi.BaseConfig +import com.twitter.timelines.configapi.BaseConfigBuilder +import com.twitter.timelines.configapi.FSName +import com.twitter.timelines.configapi.FSParam +import com.twitter.timelines.configapi.FeatureSwitchOverrideUtil +import com.twitter.timelines.configapi.Param + +object ConsumerEmbeddingBasedTwoTowerParams { + object ModelIdParam + extends FSParam[String]( + name = "consumer_embedding_based_two_tower_model_id", + default = TwoTowerFavALL20220808, + ) // Note: this default value does not match with ModelIds yet. This FS is a placeholder + + val AllParams: Seq[Param[_] with FSName] = Seq( + ModelIdParam + ) + + lazy val config: BaseConfig = { + val stringFSOverrides = + FeatureSwitchOverrideUtil.getStringFSOverrides( + ModelIdParam + ) + + BaseConfigBuilder() + .set(stringFSOverrides: _*) + .build() + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/ConsumersBasedUserAdGraphParams.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/ConsumersBasedUserAdGraphParams.scala new file mode 100644 index 0000000000..a730e0994a --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/ConsumersBasedUserAdGraphParams.scala @@ -0,0 +1,54 @@ +package com.twitter.cr_mixer.param + +import com.twitter.timelines.configapi.BaseConfig +import com.twitter.timelines.configapi.BaseConfigBuilder +import com.twitter.timelines.configapi.FSBoundedParam +import com.twitter.timelines.configapi.FSName +import com.twitter.timelines.configapi.FSParam +import com.twitter.timelines.configapi.FeatureSwitchOverrideUtil +import com.twitter.timelines.configapi.Param + +object ConsumersBasedUserAdGraphParams { + + object EnableSourceParam + extends FSParam[Boolean]( + name = "consumers_based_user_ad_graph_enable_source", + default = false + ) + + // UTG-Lookalike + object MinCoOccurrenceParam + extends FSBoundedParam[Int]( + name = "consumers_based_user_ad_graph_min_co_occurrence", + default = 2, + min = 0, + max = 500 + ) + + object MinScoreParam + extends FSBoundedParam[Double]( + name = "consumers_based_user_ad_graph_min_score", + default = 0.0, + min = 0.0, + max = 10.0 + ) + + val AllParams: Seq[Param[_] with FSName] = Seq( + EnableSourceParam, + MinCoOccurrenceParam, + MinScoreParam + ) + + lazy val config: BaseConfig = { + + val intOverrides = FeatureSwitchOverrideUtil.getBoundedIntFSOverrides(MinCoOccurrenceParam) + val doubleOverrides = FeatureSwitchOverrideUtil.getBoundedDoubleFSOverrides(MinScoreParam) + val booleanOverrides = FeatureSwitchOverrideUtil.getBooleanFSOverrides(EnableSourceParam) + + BaseConfigBuilder() + .set(intOverrides: _*) + .set(booleanOverrides: _*) + .set(doubleOverrides: _*) + .build() + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/ConsumersBasedUserTweetGraphParams.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/ConsumersBasedUserTweetGraphParams.scala new file mode 100644 index 0000000000..47c67887f1 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/ConsumersBasedUserTweetGraphParams.scala @@ -0,0 +1,44 @@ +package com.twitter.cr_mixer.param + +import com.twitter.timelines.configapi.BaseConfig +import com.twitter.timelines.configapi.BaseConfigBuilder +import com.twitter.timelines.configapi.FSName +import com.twitter.timelines.configapi.FSParam +import com.twitter.timelines.configapi.FeatureSwitchOverrideUtil +import com.twitter.timelines.configapi.Param + +/** + * ConsumersBasedUserTweetGraph Params, there are multiple ways (e.g. FRS, RealGraphOon) to generate consumersSeedSet for ConsumersBasedUserTweetGraph + * for now we allow flexibility in tuning UTG params for different consumersSeedSet generation algo by giving the param name {consumerSeedSetAlgo}{ParamName} + */ + +object ConsumersBasedUserTweetGraphParams { + + object EnableSourceParam + extends FSParam[Boolean]( + name = "consumers_based_user_tweet_graph_enable_source", + default = false + ) + + val AllParams: Seq[Param[_] with FSName] = Seq( + EnableSourceParam, + ) + + lazy val config: BaseConfig = { + + val intOverrides = FeatureSwitchOverrideUtil.getBoundedIntFSOverrides() + + val doubleOverrides = + FeatureSwitchOverrideUtil.getBoundedDoubleFSOverrides() + + val booleanOverrides = FeatureSwitchOverrideUtil.getBooleanFSOverrides( + EnableSourceParam + ) + + BaseConfigBuilder() + .set(intOverrides: _*) + .set(booleanOverrides: _*) + .set(doubleOverrides: _*) + .build() + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/ConsumersBasedUserVideoGraphParams.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/ConsumersBasedUserVideoGraphParams.scala new file mode 100644 index 0000000000..ab01336326 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/ConsumersBasedUserVideoGraphParams.scala @@ -0,0 +1,65 @@ +package com.twitter.cr_mixer.param + +import com.twitter.timelines.configapi.BaseConfig +import com.twitter.timelines.configapi.BaseConfigBuilder +import com.twitter.timelines.configapi.FSBoundedParam +import com.twitter.timelines.configapi.FSName +import com.twitter.timelines.configapi.FSParam +import com.twitter.timelines.configapi.FeatureSwitchOverrideUtil +import com.twitter.timelines.configapi.Param + +/** + * ConsumersBasedUserVideoGraph Params: there are multiple ways (e.g. FRS, RealGraphIn) to generate consumersSeedSet for ConsumersBasedUserTweetGraph + * for now we allow flexibility in tuning UVG params for different consumersSeedSet generation algo by giving the param name {consumerSeedSetAlgo}{ParamName} + */ + +object ConsumersBasedUserVideoGraphParams { + + object EnableSourceParam + extends FSParam[Boolean]( + name = "consumers_based_user_video_graph_enable_source", + default = false + ) + + // UTG-RealGraphIN + object RealGraphInMinCoOccurrenceParam + extends FSBoundedParam[Int]( + name = "consumers_based_user_video_graph_real_graph_in_min_co_occurrence", + default = 3, + min = 0, + max = 500 + ) + + object RealGraphInMinScoreParam + extends FSBoundedParam[Double]( + name = "consumers_based_user_video_graph_real_graph_in_min_score", + default = 2.0, + min = 0.0, + max = 10.0 + ) + + val AllParams: Seq[Param[_] with FSName] = Seq( + EnableSourceParam, + RealGraphInMinCoOccurrenceParam, + RealGraphInMinScoreParam + ) + + lazy val config: BaseConfig = { + + val intOverrides = + FeatureSwitchOverrideUtil.getBoundedIntFSOverrides(RealGraphInMinCoOccurrenceParam) + + val doubleOverrides = + FeatureSwitchOverrideUtil.getBoundedDoubleFSOverrides(RealGraphInMinScoreParam) + + val booleanOverrides = FeatureSwitchOverrideUtil.getBooleanFSOverrides( + EnableSourceParam + ) + + BaseConfigBuilder() + .set(intOverrides: _*) + .set(booleanOverrides: _*) + .set(doubleOverrides: _*) + .build() + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/CrMixerParamConfig.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/CrMixerParamConfig.scala new file mode 100644 index 0000000000..ada50d965b --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/CrMixerParamConfig.scala @@ -0,0 +1,122 @@ +package com.twitter.cr_mixer.param + +import com.twitter.timelines.configapi.CompositeConfig +import com.twitter.timelines.configapi.FSName +import com.twitter.timelines.configapi.Param + +object CrMixerParamConfig { + + lazy val config: CompositeConfig = new CompositeConfig( + configs = Seq( + AdsParams.config, + BlenderParams.config, + BypassInterleaveAndRankParams.config, + RankerParams.config, + ConsumerBasedWalsParams.config, + ConsumerEmbeddingBasedCandidateGenerationParams.config, + ConsumerEmbeddingBasedTripParams.config, + ConsumerEmbeddingBasedTwHINParams.config, + ConsumerEmbeddingBasedTwoTowerParams.config, + ConsumersBasedUserAdGraphParams.config, + ConsumersBasedUserTweetGraphParams.config, + ConsumersBasedUserVideoGraphParams.config, + CustomizedRetrievalBasedCandidateGenerationParams.config, + CustomizedRetrievalBasedOfflineInterestedInParams.config, + CustomizedRetrievalBasedFTROfflineInterestedInParams.config, + CustomizedRetrievalBasedTwhinParams.config, + EarlybirdFrsBasedCandidateGenerationParams.config, + FrsParams.config, + GlobalParams.config, + InterestedInParams.config, + ProducerBasedCandidateGenerationParams.config, + ProducerBasedUserAdGraphParams.config, + ProducerBasedUserTweetGraphParams.config, + RecentFollowsParams.config, + RecentNegativeSignalParams.config, + RecentNotificationsParams.config, + RecentOriginalTweetsParams.config, + RecentReplyTweetsParams.config, + RecentRetweetsParams.config, + RecentTweetFavoritesParams.config, + RelatedTweetGlobalParams.config, + RelatedVideoTweetGlobalParams.config, + RelatedTweetProducerBasedParams.config, + RelatedTweetTweetBasedParams.config, + RelatedVideoTweetTweetBasedParams.config, + RealGraphInParams.config, + RealGraphOonParams.config, + RepeatedProfileVisitsParams.config, + SimClustersANNParams.config, + TopicTweetParams.config, + TweetBasedCandidateGenerationParams.config, + TweetBasedUserAdGraphParams.config, + TweetBasedUserTweetGraphParams.config, + TweetBasedUserVideoGraphParams.config, + TweetSharesParams.config, + TweetBasedTwHINParams.config, + RealGraphOonParams.config, + GoodTweetClickParams.config, + GoodProfileClickParams.config, + UtegTweetGlobalParams.config, + VideoTweetFilterParams.config, + VideoViewTweetsParams.config, + UnifiedUSSSignalParams.config, + ), + simpleName = "CrMixerConfig" + ) + + val allParams: Seq[Param[_] with FSName] = { + AdsParams.AllParams ++ + BlenderParams.AllParams ++ + BypassInterleaveAndRankParams.AllParams ++ + RankerParams.AllParams ++ + ConsumerBasedWalsParams.AllParams ++ + ConsumerEmbeddingBasedCandidateGenerationParams.AllParams ++ + ConsumerEmbeddingBasedTripParams.AllParams ++ + ConsumerEmbeddingBasedTwHINParams.AllParams ++ + ConsumerEmbeddingBasedTwoTowerParams.AllParams ++ + ConsumersBasedUserAdGraphParams.AllParams ++ + ConsumersBasedUserTweetGraphParams.AllParams ++ + ConsumersBasedUserVideoGraphParams.AllParams ++ + CustomizedRetrievalBasedCandidateGenerationParams.AllParams ++ + CustomizedRetrievalBasedOfflineInterestedInParams.AllParams ++ + CustomizedRetrievalBasedFTROfflineInterestedInParams.AllParams ++ + CustomizedRetrievalBasedTwhinParams.AllParams ++ + EarlybirdFrsBasedCandidateGenerationParams.AllParams ++ + FrsParams.AllParams ++ + GlobalParams.AllParams ++ + InterestedInParams.AllParams ++ + ProducerBasedCandidateGenerationParams.AllParams ++ + ProducerBasedUserAdGraphParams.AllParams ++ + ProducerBasedUserTweetGraphParams.AllParams ++ + RecentFollowsParams.AllParams ++ + RecentNegativeSignalParams.AllParams ++ + RecentNotificationsParams.AllParams ++ + RecentOriginalTweetsParams.AllParams ++ + RecentReplyTweetsParams.AllParams ++ + RecentRetweetsParams.AllParams ++ + RecentTweetFavoritesParams.AllParams ++ + RelatedTweetGlobalParams.AllParams ++ + RelatedVideoTweetGlobalParams.AllParams ++ + RelatedTweetProducerBasedParams.AllParams ++ + RelatedTweetTweetBasedParams.AllParams ++ + RelatedVideoTweetTweetBasedParams.AllParams ++ + RepeatedProfileVisitsParams.AllParams ++ + SimClustersANNParams.AllParams ++ + TopicTweetParams.AllParams ++ + TweetBasedCandidateGenerationParams.AllParams ++ + TweetBasedUserAdGraphParams.AllParams ++ + TweetBasedUserTweetGraphParams.AllParams ++ + TweetBasedUserVideoGraphParams.AllParams ++ + TweetSharesParams.AllParams ++ + TweetBasedTwHINParams.AllParams ++ + RealGraphOonParams.AllParams ++ + RealGraphInParams.AllParams ++ + GoodTweetClickParams.AllParams ++ + GoodProfileClickParams.AllParams ++ + UtegTweetGlobalParams.AllParams ++ + VideoTweetFilterParams.AllParams ++ + VideoViewTweetsParams.AllParams ++ + UnifiedUSSSignalParams.AllParams + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/CustomizedRetrievalBasedCandidateGenerationParams.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/CustomizedRetrievalBasedCandidateGenerationParams.scala new file mode 100644 index 0000000000..966048b0f4 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/CustomizedRetrievalBasedCandidateGenerationParams.scala @@ -0,0 +1,81 @@ +package com.twitter.cr_mixer.param + +import com.twitter.cr_mixer.model.ModelConfig +import com.twitter.timelines.configapi.BaseConfig +import com.twitter.timelines.configapi.BaseConfigBuilder +import com.twitter.timelines.configapi.FSName +import com.twitter.timelines.configapi.FSParam +import com.twitter.timelines.configapi.FeatureSwitchOverrideUtil +import com.twitter.timelines.configapi.Param + +object CustomizedRetrievalBasedCandidateGenerationParams { + + // Offline SimClusters InterestedIn params + object EnableOfflineInterestedInParam + extends FSParam[Boolean]( + name = "customized_retrieval_based_candidate_generation_enable_offline_interestedin", + default = false + ) + + // Offline SimClusters FTR-based InterestedIn + object EnableOfflineFTRInterestedInParam + extends FSParam[Boolean]( + name = "customized_retrieval_based_candidate_generation_enable_ftr_offline_interestedin", + default = false + ) + + // TwHin Collab Filter Cluster params + object EnableTwhinCollabFilterClusterParam + extends FSParam[Boolean]( + name = "customized_retrieval_based_candidate_generation_enable_twhin_collab_filter_cluster", + default = false + ) + + // TwHin Multi Cluster params + object EnableTwhinMultiClusterParam + extends FSParam[Boolean]( + name = "customized_retrieval_based_candidate_generation_enable_twhin_multi_cluster", + default = false + ) + + object EnableRetweetBasedDiffusionParam + extends FSParam[Boolean]( + name = "customized_retrieval_based_candidate_generation_enable_retweet_based_diffusion", + default = false + ) + object CustomizedRetrievalBasedRetweetDiffusionSource + extends FSParam[String]( + name = + "customized_retrieval_based_candidate_generation_offline_retweet_based_diffusion_model_id", + default = ModelConfig.RetweetBasedDiffusion + ) + + val AllParams: Seq[Param[_] with FSName] = Seq( + EnableOfflineInterestedInParam, + EnableOfflineFTRInterestedInParam, + EnableTwhinCollabFilterClusterParam, + EnableTwhinMultiClusterParam, + EnableRetweetBasedDiffusionParam, + CustomizedRetrievalBasedRetweetDiffusionSource + ) + + lazy val config: BaseConfig = { + val booleanOverrides = FeatureSwitchOverrideUtil.getBooleanFSOverrides( + EnableOfflineInterestedInParam, + EnableOfflineFTRInterestedInParam, + EnableTwhinCollabFilterClusterParam, + EnableTwhinMultiClusterParam, + EnableRetweetBasedDiffusionParam + ) + + val stringFSOverrides = + FeatureSwitchOverrideUtil.getStringFSOverrides( + CustomizedRetrievalBasedRetweetDiffusionSource + ) + + BaseConfigBuilder() + .set(booleanOverrides: _*) + .set(stringFSOverrides: _*) + .build() + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/CustomizedRetrievalBasedFTROfflineInterestedInParams.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/CustomizedRetrievalBasedFTROfflineInterestedInParams.scala new file mode 100644 index 0000000000..d6d1b0430e --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/CustomizedRetrievalBasedFTROfflineInterestedInParams.scala @@ -0,0 +1,31 @@ +package com.twitter.cr_mixer.param +import com.twitter.cr_mixer.model.ModelConfig +import com.twitter.timelines.configapi.BaseConfig +import com.twitter.timelines.configapi.BaseConfigBuilder +import com.twitter.timelines.configapi.FSName +import com.twitter.timelines.configapi.FSParam +import com.twitter.timelines.configapi.FeatureSwitchOverrideUtil +import com.twitter.timelines.configapi.Param + +object CustomizedRetrievalBasedFTROfflineInterestedInParams { + object CustomizedRetrievalBasedFTROfflineInterestedInSource + extends FSParam[String]( + name = "customized_retrieval_based_ftr_offline_interestedin_model_id", + default = ModelConfig.OfflineFavDecayedSum + ) + + val AllParams: Seq[Param[_] with FSName] = Seq( + CustomizedRetrievalBasedFTROfflineInterestedInSource) + + lazy val config: BaseConfig = { + + val stringFSOverrides = + FeatureSwitchOverrideUtil.getStringFSOverrides( + CustomizedRetrievalBasedFTROfflineInterestedInSource + ) + + BaseConfigBuilder() + .set(stringFSOverrides: _*) + .build() + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/CustomizedRetrievalBasedOfflineInterestedInParams.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/CustomizedRetrievalBasedOfflineInterestedInParams.scala new file mode 100644 index 0000000000..d5244e135d --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/CustomizedRetrievalBasedOfflineInterestedInParams.scala @@ -0,0 +1,33 @@ +package com.twitter.cr_mixer.param + +import com.twitter.cr_mixer.model.ModelConfig +import com.twitter.timelines.configapi.BaseConfig +import com.twitter.timelines.configapi.BaseConfigBuilder +import com.twitter.timelines.configapi.FSName +import com.twitter.timelines.configapi.FSParam +import com.twitter.timelines.configapi.FeatureSwitchOverrideUtil +import com.twitter.timelines.configapi.Param + +object CustomizedRetrievalBasedOfflineInterestedInParams { + + // Model slots available for offline InterestedIn candidate generation + object CustomizedRetrievalBasedOfflineInterestedInSource + extends FSParam[String]( + name = "customized_retrieval_based_offline_interestedin_model_id", + default = ModelConfig.OfflineInterestedInFromKnownFor2020 + ) + + val AllParams: Seq[Param[_] with FSName] = Seq(CustomizedRetrievalBasedOfflineInterestedInSource) + + lazy val config: BaseConfig = { + + val stringFSOverrides = + FeatureSwitchOverrideUtil.getStringFSOverrides( + CustomizedRetrievalBasedOfflineInterestedInSource + ) + + BaseConfigBuilder() + .set(stringFSOverrides: _*) + .build() + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/CustomizedRetrievalBasedTwhinParams.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/CustomizedRetrievalBasedTwhinParams.scala new file mode 100644 index 0000000000..646cdb1631 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/CustomizedRetrievalBasedTwhinParams.scala @@ -0,0 +1,60 @@ +package com.twitter.cr_mixer.param + +import com.twitter.cr_mixer.model.ModelConfig +import com.twitter.timelines.configapi.BaseConfig +import com.twitter.timelines.configapi.BaseConfigBuilder +import com.twitter.timelines.configapi.FSName +import com.twitter.timelines.configapi.FSParam +import com.twitter.timelines.configapi.FeatureSwitchOverrideUtil +import com.twitter.timelines.configapi.Param + +object CustomizedRetrievalBasedTwhinParams { + + // Model slots available for TwhinCollab and MultiCluster + object CustomizedRetrievalBasedTwhinCollabFilterFollowSource + extends FSParam[String]( + name = "customized_retrieval_based_offline_twhin_collab_filter_follow_model_id", + default = ModelConfig.TwhinCollabFilterForFollow + ) + + object CustomizedRetrievalBasedTwhinCollabFilterEngagementSource + extends FSParam[String]( + name = "customized_retrieval_based_offline_twhin_collab_filter_engagement_model_id", + default = ModelConfig.TwhinCollabFilterForEngagement + ) + + object CustomizedRetrievalBasedTwhinMultiClusterFollowSource + extends FSParam[String]( + name = "customized_retrieval_based_offline_twhin_multi_cluster_follow_model_id", + default = ModelConfig.TwhinMultiClusterForFollow + ) + + object CustomizedRetrievalBasedTwhinMultiClusterEngagementSource + extends FSParam[String]( + name = "customized_retrieval_based_offline_twhin_multi_cluster_engagement_model_id", + default = ModelConfig.TwhinMultiClusterForEngagement + ) + + val AllParams: Seq[Param[_] with FSName] = + Seq( + CustomizedRetrievalBasedTwhinCollabFilterFollowSource, + CustomizedRetrievalBasedTwhinCollabFilterEngagementSource, + CustomizedRetrievalBasedTwhinMultiClusterFollowSource, + CustomizedRetrievalBasedTwhinMultiClusterEngagementSource, + ) + + lazy val config: BaseConfig = { + + val stringFSOverrides = + FeatureSwitchOverrideUtil.getStringFSOverrides( + CustomizedRetrievalBasedTwhinCollabFilterFollowSource, + CustomizedRetrievalBasedTwhinCollabFilterEngagementSource, + CustomizedRetrievalBasedTwhinMultiClusterFollowSource, + CustomizedRetrievalBasedTwhinMultiClusterEngagementSource, + ) + + BaseConfigBuilder() + .set(stringFSOverrides: _*) + .build() + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/EarlybirdFrsBasedCandidateGenerationParams.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/EarlybirdFrsBasedCandidateGenerationParams.scala new file mode 100644 index 0000000000..2a9ffb4246 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/EarlybirdFrsBasedCandidateGenerationParams.scala @@ -0,0 +1,117 @@ +package com.twitter.cr_mixer.param + +import com.twitter.conversions.DurationOps._ +import com.twitter.cr_mixer.model.EarlybirdSimilarityEngineType +import com.twitter.cr_mixer.model.EarlybirdSimilarityEngineType_ModelBased +import com.twitter.cr_mixer.model.EarlybirdSimilarityEngineType_RecencyBased +import com.twitter.cr_mixer.model.EarlybirdSimilarityEngineType_TensorflowBased +import com.twitter.finagle.stats.NullStatsReceiver +import com.twitter.logging.Logger +import com.twitter.timelines.configapi.BaseConfig +import com.twitter.timelines.configapi.BaseConfigBuilder +import com.twitter.timelines.configapi.DurationConversion +import com.twitter.timelines.configapi.FSBoundedParam +import com.twitter.timelines.configapi.FSEnumParam +import com.twitter.timelines.configapi.FSName +import com.twitter.timelines.configapi.FSParam +import com.twitter.timelines.configapi.FeatureSwitchOverrideUtil +import com.twitter.timelines.configapi.HasDurationConversion +import com.twitter.timelines.configapi.Param +import com.twitter.util.Duration + +object EarlybirdFrsBasedCandidateGenerationParams { + object CandidateGenerationEarlybirdSimilarityEngineType extends Enumeration { + protected case class SimilarityEngineType(rankingMode: EarlybirdSimilarityEngineType) + extends super.Val + import scala.language.implicitConversions + implicit def valueToEarlybirdRankingMode(x: Value): SimilarityEngineType = + x.asInstanceOf[SimilarityEngineType] + + val EarlybirdRankingMode_RecencyBased: SimilarityEngineType = SimilarityEngineType( + EarlybirdSimilarityEngineType_RecencyBased) + val EarlybirdRankingMode_ModelBased: SimilarityEngineType = SimilarityEngineType( + EarlybirdSimilarityEngineType_ModelBased) + val EarlybirdRankingMode_TensorflowBased: SimilarityEngineType = SimilarityEngineType( + EarlybirdSimilarityEngineType_TensorflowBased) + } + + object FrsBasedCandidateGenerationEarlybirdSimilarityEngineTypeParam + extends FSEnumParam[CandidateGenerationEarlybirdSimilarityEngineType.type]( + name = "frs_based_candidate_generation_earlybird_ranking_mode_id", + default = + CandidateGenerationEarlybirdSimilarityEngineType.EarlybirdRankingMode_RecencyBased, + enum = CandidateGenerationEarlybirdSimilarityEngineType + ) + + object FrsBasedCandidateGenerationRecencyBasedEarlybirdMaxTweetsPerUser + extends FSBoundedParam[Int]( + name = "frs_based_candidate_generation_earlybird_max_tweets_per_user", + default = 100, + min = 0, + /** + * Note max should be equal to EarlybirdRecencyBasedCandidateStoreModule.DefaultMaxNumTweetPerUser. + * Which is the size of the memcached result list. + */ + max = 100 + ) + + object FrsBasedCandidateGenerationEarlybirdMaxTweetAge + extends FSBoundedParam[Duration]( + name = "frs_based_candidate_generation_earlybird_max_tweet_age_hours", + default = 24.hours, + min = 12.hours, + /** + * Note max could be related to EarlybirdRecencyBasedCandidateStoreModule.DefaultMaxNumTweetPerUser. + * Which is the size of the memcached result list for recency based earlybird candidate source. + * E.g. if max = 720.hours, we may want to increase the DefaultMaxNumTweetPerUser. + */ + max = 96.hours + ) + with HasDurationConversion { + override val durationConversion: DurationConversion = DurationConversion.FromHours + } + + object FrsBasedCandidateGenerationEarlybirdFilterOutRetweetsAndReplies + extends FSParam[Boolean]( + name = "frs_based_candidate_generation_earlybird_filter_out_retweets_and_replies", + default = true + ) + + val AllParams: Seq[Param[_] with FSName] = Seq( + FrsBasedCandidateGenerationEarlybirdSimilarityEngineTypeParam, + FrsBasedCandidateGenerationRecencyBasedEarlybirdMaxTweetsPerUser, + FrsBasedCandidateGenerationEarlybirdMaxTweetAge, + FrsBasedCandidateGenerationEarlybirdFilterOutRetweetsAndReplies, + ) + + lazy val config: BaseConfig = { + val booleanOverrides = FeatureSwitchOverrideUtil.getBooleanFSOverrides( + FrsBasedCandidateGenerationEarlybirdFilterOutRetweetsAndReplies, + ) + + val doubleOverrides = FeatureSwitchOverrideUtil.getBoundedDoubleFSOverrides() + + val intOverrides = FeatureSwitchOverrideUtil.getBoundedIntFSOverrides( + FrsBasedCandidateGenerationRecencyBasedEarlybirdMaxTweetsPerUser + ) + + val durationFSOverrides = + FeatureSwitchOverrideUtil.getDurationFSOverrides( + FrsBasedCandidateGenerationEarlybirdMaxTweetAge + ) + + val enumOverrides = FeatureSwitchOverrideUtil.getEnumFSOverrides( + NullStatsReceiver, + Logger(getClass), + FrsBasedCandidateGenerationEarlybirdSimilarityEngineTypeParam, + ) + + BaseConfigBuilder() + .set(booleanOverrides: _*) + .set(doubleOverrides: _*) + .set(intOverrides: _*) + .set(enumOverrides: _*) + .set(durationFSOverrides: _*) + .build() + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/FrsParams.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/FrsParams.scala new file mode 100644 index 0000000000..18bf1d4741 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/FrsParams.scala @@ -0,0 +1,131 @@ +package com.twitter.cr_mixer.param + +import com.twitter.timelines.configapi.BaseConfig +import com.twitter.timelines.configapi.BaseConfigBuilder +import com.twitter.timelines.configapi.FSBoundedParam +import com.twitter.timelines.configapi.FSName +import com.twitter.timelines.configapi.FSParam +import com.twitter.timelines.configapi.FeatureSwitchOverrideUtil +import com.twitter.timelines.configapi.Param +import com.twitter.follow_recommendations.thriftscala.DisplayLocation +import com.twitter.timelines.configapi.FSEnumParam +import com.twitter.logging.Logger +import com.twitter.finagle.stats.NullStatsReceiver + +object FrsParams { + object EnableSourceParam + extends FSParam[Boolean]( + name = "signal_frs_enable_source", + default = false + ) + + object EnableSourceGraphParam + extends FSParam[Boolean]( + name = "graph_frs_enable_source", + default = false + ) + + object MinScoreParam + extends FSBoundedParam[Double]( + name = "signal_frs_min_score", + default = 0.4, + min = 0.0, + max = 1.0 + ) + + object MaxConsumerSeedsNumParam + extends FSBoundedParam[Int]( + name = "graph_frs_max_user_seeds_num", + default = 200, + min = 0, + max = 1000 + ) + + /** + * These params below are only used for FrsTweetCandidateGenerator and shouldn't be used in other endpoints + * * FrsBasedCandidateGenerationMaxSeedsNumParam + * * FrsCandidateGenerationDisplayLocationParam + * * FrsCandidateGenerationDisplayLocation + * * FrsBasedCandidateGenerationMaxCandidatesNumParam + */ + object FrsBasedCandidateGenerationEnableVisibilityFilteringParam + extends FSParam[Boolean]( + name = "frs_based_candidate_generation_enable_vf", + default = true + ) + + object FrsBasedCandidateGenerationMaxSeedsNumParam + extends FSBoundedParam[Int]( + name = "frs_based_candidate_generation_max_seeds_num", + default = 100, + min = 0, + max = 800 + ) + + object FrsBasedCandidateGenerationDisplayLocation extends Enumeration { + protected case class FrsDisplayLocationValue(displayLocation: DisplayLocation) extends super.Val + import scala.language.implicitConversions + implicit def valueToDisplayLocationValue(x: Value): FrsDisplayLocationValue = + x.asInstanceOf[FrsDisplayLocationValue] + + val DisplayLocation_ContentRecommender: FrsDisplayLocationValue = FrsDisplayLocationValue( + DisplayLocation.ContentRecommender) + val DisplayLocation_Home: FrsDisplayLocationValue = FrsDisplayLocationValue( + DisplayLocation.HomeTimelineTweetRecs) + val DisplayLocation_Notifications: FrsDisplayLocationValue = FrsDisplayLocationValue( + DisplayLocation.TweetNotificationRecs) + } + + object FrsBasedCandidateGenerationDisplayLocationParam + extends FSEnumParam[FrsBasedCandidateGenerationDisplayLocation.type]( + name = "frs_based_candidate_generation_display_location_id", + default = FrsBasedCandidateGenerationDisplayLocation.DisplayLocation_Home, + enum = FrsBasedCandidateGenerationDisplayLocation + ) + + object FrsBasedCandidateGenerationMaxCandidatesNumParam + extends FSBoundedParam[Int]( + name = "frs_based_candidate_generation_max_candidates_num", + default = 100, + min = 0, + max = 2000 + ) + + val AllParams: Seq[Param[_] with FSName] = Seq( + EnableSourceParam, + EnableSourceGraphParam, + MinScoreParam, + MaxConsumerSeedsNumParam, + FrsBasedCandidateGenerationMaxSeedsNumParam, + FrsBasedCandidateGenerationDisplayLocationParam, + FrsBasedCandidateGenerationMaxCandidatesNumParam, + FrsBasedCandidateGenerationEnableVisibilityFilteringParam + ) + + lazy val config: BaseConfig = { + val booleanOverrides = FeatureSwitchOverrideUtil.getBooleanFSOverrides( + EnableSourceParam, + EnableSourceGraphParam, + FrsBasedCandidateGenerationEnableVisibilityFilteringParam + ) + + val doubleOverrides = FeatureSwitchOverrideUtil.getBoundedDoubleFSOverrides(MinScoreParam) + + val intOverrides = FeatureSwitchOverrideUtil.getBoundedIntFSOverrides( + MaxConsumerSeedsNumParam, + FrsBasedCandidateGenerationMaxSeedsNumParam, + FrsBasedCandidateGenerationMaxCandidatesNumParam) + + val enumOverrides = FeatureSwitchOverrideUtil.getEnumFSOverrides( + NullStatsReceiver, + Logger(getClass), + FrsBasedCandidateGenerationDisplayLocationParam, + ) + BaseConfigBuilder() + .set(booleanOverrides: _*) + .set(doubleOverrides: _*) + .set(intOverrides: _*) + .set(enumOverrides: _*) + .build() + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/GlobalParams.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/GlobalParams.scala new file mode 100644 index 0000000000..77def9a2ae --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/GlobalParams.scala @@ -0,0 +1,106 @@ +package com.twitter.cr_mixer.param + +import com.twitter.conversions.DurationOps._ +import com.twitter.finagle.stats.NullStatsReceiver +import com.twitter.logging.Logger +import com.twitter.simclusters_v2.common.ModelVersions +import com.twitter.timelines.configapi.BaseConfig +import com.twitter.timelines.configapi.BaseConfigBuilder +import com.twitter.timelines.configapi.DurationConversion +import com.twitter.timelines.configapi.FSBoundedParam +import com.twitter.timelines.configapi.FSEnumParam +import com.twitter.timelines.configapi.FSName +import com.twitter.timelines.configapi.FeatureSwitchOverrideUtil +import com.twitter.timelines.configapi.HasDurationConversion +import com.twitter.timelines.configapi.Param +import com.twitter.util.Duration + +/** + * Instantiate Params that do not relate to a specific product. + * The params in this file correspond to config repo file + * [[https://sourcegraph.twitter.biz/config-git.twitter.biz/config/-/blob/features/cr-mixer/main/twistly_core.yml]] + */ +object GlobalParams { + + object MaxCandidatesPerRequestParam + extends FSBoundedParam[Int]( + name = "twistly_core_max_candidates_per_request", + default = 100, + min = 0, + max = 9000 + ) + + object ModelVersionParam + extends FSEnumParam[ModelVersions.Enum.type]( + name = "twistly_core_simclusters_model_version_id", + default = ModelVersions.Enum.Model20M145K2020, + enum = ModelVersions.Enum + ) + + object UnifiedMaxSourceKeyNum + extends FSBoundedParam[Int]( + name = "twistly_core_unified_max_sourcekey_num", + default = 15, + min = 0, + max = 100 + ) + + object MaxCandidateNumPerSourceKeyParam + extends FSBoundedParam[Int]( + name = "twistly_core_candidate_per_sourcekey_max_num", + default = 200, + min = 0, + max = 1000 + ) + + // 1 hours to 30 days + object MaxTweetAgeHoursParam + extends FSBoundedParam[Duration]( + name = "twistly_core_max_tweet_age_hours", + default = 720.hours, + min = 1.hours, + max = 720.hours + ) + with HasDurationConversion { + + override val durationConversion: DurationConversion = DurationConversion.FromHours + } + + val AllParams: Seq[Param[_] with FSName] = Seq( + MaxCandidatesPerRequestParam, + UnifiedMaxSourceKeyNum, + MaxCandidateNumPerSourceKeyParam, + ModelVersionParam, + MaxTweetAgeHoursParam + ) + + lazy val config: BaseConfig = { + + val booleanOverrides = FeatureSwitchOverrideUtil.getBooleanFSOverrides() + + val intOverrides = FeatureSwitchOverrideUtil.getBoundedIntFSOverrides( + MaxCandidatesPerRequestParam, + UnifiedMaxSourceKeyNum, + MaxCandidateNumPerSourceKeyParam + ) + + val enumOverrides = FeatureSwitchOverrideUtil.getEnumFSOverrides( + NullStatsReceiver, + Logger(getClass), + ModelVersionParam + ) + + val boundedDurationFSOverrides = + FeatureSwitchOverrideUtil.getBoundedDurationFSOverrides(MaxTweetAgeHoursParam) + + val seqOverrides = FeatureSwitchOverrideUtil.getLongSeqFSOverrides() + + BaseConfigBuilder() + .set(booleanOverrides: _*) + .set(intOverrides: _*) + .set(boundedDurationFSOverrides: _*) + .set(enumOverrides: _*) + .set(seqOverrides: _*) + .build() + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/GoodProfileClickParams.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/GoodProfileClickParams.scala new file mode 100644 index 0000000000..175dccfac8 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/GoodProfileClickParams.scala @@ -0,0 +1,60 @@ +package com.twitter.cr_mixer.param + +import com.twitter.finagle.stats.NullStatsReceiver +import com.twitter.logging.Logger +import com.twitter.timelines.configapi.BaseConfig +import com.twitter.timelines.configapi.BaseConfigBuilder +import com.twitter.timelines.configapi.FSEnumParam +import com.twitter.timelines.configapi.FSName +import com.twitter.timelines.configapi.FSParam +import com.twitter.timelines.configapi.FeatureSwitchOverrideUtil +import com.twitter.timelines.configapi.Param +import com.twitter.usersignalservice.thriftscala.SignalType + +object GoodProfileClickParams { + + object ClickMinDwellTimeParam extends Enumeration { + protected case class SignalTypeValue(signalType: SignalType) extends super.Val + import scala.language.implicitConversions + implicit def valueToSignalTypeValue(x: Value): SignalTypeValue = + x.asInstanceOf[SignalTypeValue] + + val TotalDwellTime10s = SignalTypeValue(SignalType.GoodProfileClick) + val TotalDwellTime20s = SignalTypeValue(SignalType.GoodProfileClick20s) + val TotalDwellTime30s = SignalTypeValue(SignalType.GoodProfileClick30s) + + } + + object EnableSourceParam + extends FSParam[Boolean]( + name = "signal_good_profile_clicks_enable_source", + default = false + ) + + object ClickMinDwellTimeType + extends FSEnumParam[ClickMinDwellTimeParam.type]( + name = "signal_good_profile_clicks_min_dwelltime_type_id", + default = ClickMinDwellTimeParam.TotalDwellTime10s, + enum = ClickMinDwellTimeParam + ) + + val AllParams: Seq[Param[_] with FSName] = + Seq(EnableSourceParam, ClickMinDwellTimeType) + + lazy val config: BaseConfig = { + val booleanOverrides = FeatureSwitchOverrideUtil.getBooleanFSOverrides( + EnableSourceParam + ) + + val enumOverrides = FeatureSwitchOverrideUtil.getEnumFSOverrides( + NullStatsReceiver, + Logger(getClass), + ClickMinDwellTimeType + ) + + BaseConfigBuilder() + .set(booleanOverrides: _*) + .set(enumOverrides: _*) + .build() + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/GoodTweetClickParams.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/GoodTweetClickParams.scala new file mode 100644 index 0000000000..949048821f --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/GoodTweetClickParams.scala @@ -0,0 +1,75 @@ +package com.twitter.cr_mixer.param + +import com.twitter.finagle.stats.NullStatsReceiver +import com.twitter.logging.Logger +import com.twitter.timelines.configapi.BaseConfig +import com.twitter.timelines.configapi.BaseConfigBuilder +import com.twitter.timelines.configapi.FSBoundedParam +import com.twitter.timelines.configapi.FSEnumParam +import com.twitter.timelines.configapi.FSName +import com.twitter.timelines.configapi.FSParam +import com.twitter.timelines.configapi.FeatureSwitchOverrideUtil +import com.twitter.timelines.configapi.Param +import com.twitter.usersignalservice.thriftscala.SignalType + +object GoodTweetClickParams { + + object ClickMinDwellTimeParam extends Enumeration { + protected case class SignalTypeValue(signalType: SignalType) extends super.Val + import scala.language.implicitConversions + implicit def valueToSignalTypeValue(x: Value): SignalTypeValue = + x.asInstanceOf[SignalTypeValue] + + val TotalDwellTime2s = SignalTypeValue(SignalType.GoodTweetClick) + val TotalDwellTime5s = SignalTypeValue(SignalType.GoodTweetClick5s) + val TotalDwellTime10s = SignalTypeValue(SignalType.GoodTweetClick10s) + val TotalDwellTime30s = SignalTypeValue(SignalType.GoodTweetClick30s) + + } + + object EnableSourceParam + extends FSParam[Boolean]( + name = "signal_good_tweet_clicks_enable_source", + default = false + ) + + object ClickMinDwellTimeType + extends FSEnumParam[ClickMinDwellTimeParam.type]( + name = "signal_good_tweet_clicks_min_dwelltime_type_id", + default = ClickMinDwellTimeParam.TotalDwellTime2s, + enum = ClickMinDwellTimeParam + ) + + object MaxSignalNumParam + extends FSBoundedParam[Int]( + name = "signal_good_tweet_clicks_max_signal_num", + default = 15, + min = 0, + max = 15 + ) + + val AllParams: Seq[Param[_] with FSName] = + Seq(EnableSourceParam, ClickMinDwellTimeType, MaxSignalNumParam) + + lazy val config: BaseConfig = { + val booleanOverrides = FeatureSwitchOverrideUtil.getBooleanFSOverrides( + EnableSourceParam + ) + + val enumOverrides = FeatureSwitchOverrideUtil.getEnumFSOverrides( + NullStatsReceiver, + Logger(getClass), + ClickMinDwellTimeType + ) + + val intOverrides = FeatureSwitchOverrideUtil.getBoundedIntFSOverrides( + MaxSignalNumParam + ) + + BaseConfigBuilder() + .set(booleanOverrides: _*) + .set(enumOverrides: _*) + .set(intOverrides: _*) + .build() + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/InterestedInParams.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/InterestedInParams.scala new file mode 100644 index 0000000000..503469ac37 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/InterestedInParams.scala @@ -0,0 +1,213 @@ +package com.twitter.cr_mixer.param + +import com.twitter.finagle.stats.NullStatsReceiver +import com.twitter.logging.Logger +import com.twitter.simclusters_v2.thriftscala.{EmbeddingType => SimClustersEmbeddingType} +import com.twitter.timelines.configapi.BaseConfig +import com.twitter.timelines.configapi.BaseConfigBuilder +import com.twitter.timelines.configapi.FSBoundedParam +import com.twitter.timelines.configapi.FSEnumParam +import com.twitter.timelines.configapi.FSName +import com.twitter.timelines.configapi.FSParam +import com.twitter.timelines.configapi.FeatureSwitchOverrideUtil +import com.twitter.timelines.configapi.Param + +object InterestedInParams { + + object SourceEmbedding extends Enumeration { + protected case class EmbeddingType(embeddingType: SimClustersEmbeddingType) extends super.Val + import scala.language.implicitConversions + implicit def valueToEmbeddingtype(x: Value): EmbeddingType = x.asInstanceOf[EmbeddingType] + + val UserInterestedIn: Value = EmbeddingType(SimClustersEmbeddingType.FilteredUserInterestedIn) + val UnfilteredUserInterestedIn: Value = EmbeddingType( + SimClustersEmbeddingType.UnfilteredUserInterestedIn) + val FromProducerEmbedding: Value = EmbeddingType( + SimClustersEmbeddingType.FilteredUserInterestedInFromPE) + val LogFavBasedUserInterestedInFromAPE: Value = EmbeddingType( + SimClustersEmbeddingType.LogFavBasedUserInterestedInFromAPE) + val FollowBasedUserInterestedInFromAPE: Value = EmbeddingType( + SimClustersEmbeddingType.FollowBasedUserInterestedInFromAPE) + val UserNextInterestedIn: Value = EmbeddingType(SimClustersEmbeddingType.UserNextInterestedIn) + // AddressBook based InterestedIn + val LogFavBasedUserInterestedAverageAddressBookFromIIAPE: Value = EmbeddingType( + SimClustersEmbeddingType.LogFavBasedUserInterestedAverageAddressBookFromIIAPE) + val LogFavBasedUserInterestedMaxpoolingAddressBookFromIIAPE: Value = EmbeddingType( + SimClustersEmbeddingType.LogFavBasedUserInterestedMaxpoolingAddressBookFromIIAPE) + val LogFavBasedUserInterestedBooktypeMaxpoolingAddressBookFromIIAPE: Value = EmbeddingType( + SimClustersEmbeddingType.LogFavBasedUserInterestedBooktypeMaxpoolingAddressBookFromIIAPE) + val LogFavBasedUserInterestedLargestDimMaxpoolingAddressBookFromIIAPE: Value = EmbeddingType( + SimClustersEmbeddingType.LogFavBasedUserInterestedLargestDimMaxpoolingAddressBookFromIIAPE) + val LogFavBasedUserInterestedLouvainMaxpoolingAddressBookFromIIAPE: Value = EmbeddingType( + SimClustersEmbeddingType.LogFavBasedUserInterestedLouvainMaxpoolingAddressBookFromIIAPE) + val LogFavBasedUserInterestedConnectedMaxpoolingAddressBookFromIIAPE: Value = EmbeddingType( + SimClustersEmbeddingType.LogFavBasedUserInterestedConnectedMaxpoolingAddressBookFromIIAPE) + } + + object EnableSourceParam + extends FSParam[Boolean]( + name = "twistly_interestedin_enable_source", + default = true + ) + + object InterestedInEmbeddingIdParam + extends FSEnumParam[SourceEmbedding.type]( + name = "twistly_interestedin_embedding_id", + default = SourceEmbedding.UnfilteredUserInterestedIn, + enum = SourceEmbedding + ) + + object MinScoreParam + extends FSBoundedParam[Double]( + name = "twistly_interestedin_min_score", + default = 0.072, + min = 0.0, + max = 1.0 + ) + + object EnableSourceSequentialModelParam + extends FSParam[Boolean]( + name = "twistly_interestedin_sequential_model_enable_source", + default = false + ) + + object NextInterestedInEmbeddingIdParam + extends FSEnumParam[SourceEmbedding.type]( + name = "twistly_interestedin_sequential_model_embedding_id", + default = SourceEmbedding.UserNextInterestedIn, + enum = SourceEmbedding + ) + + object MinScoreSequentialModelParam + extends FSBoundedParam[Double]( + name = "twistly_interestedin_sequential_model_min_score", + default = 0.0, + min = 0.0, + max = 1.0 + ) + + object EnableSourceAddressBookParam + extends FSParam[Boolean]( + name = "twistly_interestedin_addressbook_enable_source", + default = false + ) + + object AddressBookInterestedInEmbeddingIdParam + extends FSEnumParam[SourceEmbedding.type]( + name = "twistly_interestedin_addressbook_embedding_id", + default = SourceEmbedding.LogFavBasedUserInterestedLouvainMaxpoolingAddressBookFromIIAPE, + enum = SourceEmbedding + ) + + object MinScoreAddressBookParam + extends FSBoundedParam[Double]( + name = "twistly_interestedin_addressbook_min_score", + default = 0.0, + min = 0.0, + max = 1.0 + ) + + // Prod SimClusters ANN param + // This is used to enable/disable querying of production SANN service. Useful when experimenting + // with replacements to it. + object EnableProdSimClustersANNParam + extends FSParam[Boolean]( + name = "twistly_interestedin_enable_prod_simclusters_ann", + default = true + ) + + // Experimental SimClusters ANN params + object EnableExperimentalSimClustersANNParam + extends FSParam[Boolean]( + name = "twistly_interestedin_enable_experimental_simclusters_ann", + default = false + ) + + // SimClusters ANN 1 cluster params + object EnableSimClustersANN1Param + extends FSParam[Boolean]( + name = "twistly_interestedin_enable_simclusters_ann_1", + default = false + ) + + // SimClusters ANN 2 cluster params + object EnableSimClustersANN2Param + extends FSParam[Boolean]( + name = "twistly_interestedin_enable_simclusters_ann_2", + default = false + ) + + // SimClusters ANN 3 cluster params + object EnableSimClustersANN3Param + extends FSParam[Boolean]( + name = "twistly_interestedin_enable_simclusters_ann_3", + default = false + ) + + // SimClusters ANN 5 cluster params + object EnableSimClustersANN5Param + extends FSParam[Boolean]( + name = "twistly_interestedin_enable_simclusters_ann_5", + default = false + ) + + // SimClusters ANN 4 cluster params + object EnableSimClustersANN4Param + extends FSParam[Boolean]( + name = "twistly_interestedin_enable_simclusters_ann_4", + default = false + ) + val AllParams: Seq[Param[_] with FSName] = Seq( + EnableSourceParam, + EnableSourceSequentialModelParam, + EnableSourceAddressBookParam, + EnableProdSimClustersANNParam, + EnableExperimentalSimClustersANNParam, + EnableSimClustersANN1Param, + EnableSimClustersANN2Param, + EnableSimClustersANN3Param, + EnableSimClustersANN5Param, + EnableSimClustersANN4Param, + MinScoreParam, + MinScoreSequentialModelParam, + MinScoreAddressBookParam, + InterestedInEmbeddingIdParam, + NextInterestedInEmbeddingIdParam, + AddressBookInterestedInEmbeddingIdParam, + ) + + lazy val config: BaseConfig = { + + val booleanOverrides = FeatureSwitchOverrideUtil.getBooleanFSOverrides( + EnableSourceParam, + EnableSourceSequentialModelParam, + EnableSourceAddressBookParam, + EnableProdSimClustersANNParam, + EnableExperimentalSimClustersANNParam, + EnableSimClustersANN1Param, + EnableSimClustersANN2Param, + EnableSimClustersANN3Param, + EnableSimClustersANN5Param, + EnableSimClustersANN4Param + ) + + val doubleOverrides = FeatureSwitchOverrideUtil.getBoundedDoubleFSOverrides( + MinScoreParam, + MinScoreSequentialModelParam, + MinScoreAddressBookParam) + + val enumOverrides = FeatureSwitchOverrideUtil.getEnumFSOverrides( + NullStatsReceiver, + Logger(getClass), + InterestedInEmbeddingIdParam, + NextInterestedInEmbeddingIdParam, + AddressBookInterestedInEmbeddingIdParam + ) + + BaseConfigBuilder() + .set(booleanOverrides: _*) + .set(doubleOverrides: _*) + .set(enumOverrides: _*) + .build() + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/ProducerBasedCandidateGenerationParams.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/ProducerBasedCandidateGenerationParams.scala new file mode 100644 index 0000000000..e9ae7feaaf --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/ProducerBasedCandidateGenerationParams.scala @@ -0,0 +1,143 @@ +package com.twitter.cr_mixer.param + +import com.twitter.finagle.stats.NullStatsReceiver +import com.twitter.logging.Logger +import com.twitter.timelines.configapi.BaseConfig +import com.twitter.timelines.configapi.BaseConfigBuilder +import com.twitter.timelines.configapi.FSBoundedParam +import com.twitter.timelines.configapi.FSEnumParam +import com.twitter.timelines.configapi.FSName +import com.twitter.timelines.configapi.FSParam +import com.twitter.timelines.configapi.FeatureSwitchOverrideUtil +import com.twitter.timelines.configapi.Param + +object ProducerBasedCandidateGenerationParams { + // Source params. Not being used. It is always set to true in prod + object EnableSourceParam + extends FSParam[Boolean]( + name = "producer_based_candidate_generation_enable_source", + default = false + ) + + object UtgCombinationMethodParam + extends FSEnumParam[UnifiedSETweetCombinationMethod.type]( + name = "producer_based_candidate_generation_utg_combination_method_id", + default = UnifiedSETweetCombinationMethod.Frontload, + enum = UnifiedSETweetCombinationMethod + ) + + // UTG params + object EnableUTGParam + extends FSParam[Boolean]( + name = "producer_based_candidate_generation_enable_utg", + default = false + ) + + object EnableUAGParam + extends FSParam[Boolean]( + name = "producer_based_candidate_generation_enable_uag", + default = false + ) + + // SimClusters params + object EnableSimClustersANNParam + extends FSParam[Boolean]( + name = "producer_based_candidate_generation_enable_simclusters", + default = true + ) + + // Filter params + object SimClustersMinScoreParam + extends FSBoundedParam[Double]( + name = "producer_based_candidate_generation_filter_simclusters_min_score", + default = 0.7, + min = 0.0, + max = 1.0 + ) + + // Experimental SimClusters ANN params + object EnableExperimentalSimClustersANNParam + extends FSParam[Boolean]( + name = "producer_based_candidate_generation_enable_experimental_simclusters_ann", + default = false + ) + + // SimClusters ANN cluster 1 params + object EnableSimClustersANN1Param + extends FSParam[Boolean]( + name = "producer_based_candidate_generation_enable_simclusters_ann_1", + default = false + ) + + // SimClusters ANN cluster 2 params + object EnableSimClustersANN2Param + extends FSParam[Boolean]( + name = "producer_based_candidate_generation_enable_simclusters_ann_2", + default = false + ) + + // SimClusters ANN cluster 3 params + object EnableSimClustersANN3Param + extends FSParam[Boolean]( + name = "producer_based_candidate_generation_enable_simclusters_ann_3", + default = false + ) + + // SimClusters ANN cluster 5 params + object EnableSimClustersANN5Param + extends FSParam[Boolean]( + name = "producer_based_candidate_generation_enable_simclusters_ann_5", + default = false + ) + + object EnableSimClustersANN4Param + extends FSParam[Boolean]( + name = "producer_based_candidate_generation_enable_simclusters_ann_4", + default = false + ) + val AllParams: Seq[Param[_] with FSName] = Seq( + EnableSourceParam, + EnableUAGParam, + EnableUTGParam, + EnableSimClustersANNParam, + EnableSimClustersANN1Param, + EnableSimClustersANN2Param, + EnableSimClustersANN3Param, + EnableSimClustersANN5Param, + EnableSimClustersANN4Param, + EnableExperimentalSimClustersANNParam, + SimClustersMinScoreParam, + UtgCombinationMethodParam + ) + + lazy val config: BaseConfig = { + + val booleanOverrides = FeatureSwitchOverrideUtil.getBooleanFSOverrides( + EnableSourceParam, + EnableUAGParam, + EnableUTGParam, + EnableSimClustersANNParam, + EnableSimClustersANN1Param, + EnableSimClustersANN2Param, + EnableSimClustersANN3Param, + EnableSimClustersANN5Param, + EnableSimClustersANN4Param, + EnableExperimentalSimClustersANNParam + ) + + val enumOverrides = FeatureSwitchOverrideUtil.getEnumFSOverrides( + NullStatsReceiver, + Logger(getClass), + UtgCombinationMethodParam, + ) + + val doubleOverrides = + FeatureSwitchOverrideUtil.getBoundedDoubleFSOverrides(SimClustersMinScoreParam) + + BaseConfigBuilder() + .set(booleanOverrides: _*) + .set(doubleOverrides: _*) + .set(enumOverrides: _*) + .build() + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/ProducerBasedUserAdGraphParams.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/ProducerBasedUserAdGraphParams.scala new file mode 100644 index 0000000000..197db074a1 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/ProducerBasedUserAdGraphParams.scala @@ -0,0 +1,53 @@ +package com.twitter.cr_mixer.param + +import com.twitter.timelines.configapi.BaseConfig +import com.twitter.timelines.configapi.BaseConfigBuilder +import com.twitter.timelines.configapi.FSBoundedParam +import com.twitter.timelines.configapi.FSName +import com.twitter.timelines.configapi.FeatureSwitchOverrideUtil +import com.twitter.timelines.configapi.Param + +object ProducerBasedUserAdGraphParams { + + object MinCoOccurrenceParam + extends FSBoundedParam[Int]( + name = "producer_based_user_ad_graph_min_co_occurrence", + default = 2, + min = 0, + max = 500 + ) + + object MinScoreParam + extends FSBoundedParam[Double]( + name = "producer_based_user_ad_graph_min_score", + default = 3.0, + min = 0.0, + max = 10.0 + ) + + object MaxNumFollowersParam + extends FSBoundedParam[Int]( + name = "producer_based_user_ad_graph_max_num_followers", + default = 500, + min = 100, + max = 1000 + ) + + val AllParams: Seq[Param[_] with FSName] = + Seq(MinCoOccurrenceParam, MaxNumFollowersParam, MinScoreParam) + + lazy val config: BaseConfig = { + + val intOverrides = FeatureSwitchOverrideUtil.getBoundedIntFSOverrides( + MinCoOccurrenceParam, + MaxNumFollowersParam, + ) + + val doubleOverrides = FeatureSwitchOverrideUtil.getBoundedDoubleFSOverrides(MinScoreParam) + + BaseConfigBuilder() + .set(intOverrides: _*) + .set(doubleOverrides: _*) + .build() + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/ProducerBasedUserTweetGraphParams.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/ProducerBasedUserTweetGraphParams.scala new file mode 100644 index 0000000000..0747d0afdd --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/ProducerBasedUserTweetGraphParams.scala @@ -0,0 +1,53 @@ +package com.twitter.cr_mixer.param + +import com.twitter.timelines.configapi.BaseConfig +import com.twitter.timelines.configapi.BaseConfigBuilder +import com.twitter.timelines.configapi.FSBoundedParam +import com.twitter.timelines.configapi.FSName +import com.twitter.timelines.configapi.FeatureSwitchOverrideUtil +import com.twitter.timelines.configapi.Param + +object ProducerBasedUserTweetGraphParams { + + object MinCoOccurrenceParam + extends FSBoundedParam[Int]( + name = "producer_based_user_tweet_graph_min_co_occurrence", + default = 4, + min = 0, + max = 500 + ) + + object MinScoreParam + extends FSBoundedParam[Double]( + name = "producer_based_user_tweet_graph_min_score", + default = 3.0, + min = 0.0, + max = 10.0 + ) + + object MaxNumFollowersParam + extends FSBoundedParam[Int]( + name = "producer_based_user_tweet_graph_max_num_followers", + default = 500, + min = 100, + max = 1000 + ) + + val AllParams: Seq[Param[_] with FSName] = + Seq(MinCoOccurrenceParam, MaxNumFollowersParam, MinScoreParam) + + lazy val config: BaseConfig = { + + val intOverrides = FeatureSwitchOverrideUtil.getBoundedIntFSOverrides( + MinCoOccurrenceParam, + MaxNumFollowersParam, + ) + + val doubleOverrides = FeatureSwitchOverrideUtil.getBoundedDoubleFSOverrides(MinScoreParam) + + BaseConfigBuilder() + .set(intOverrides: _*) + .set(doubleOverrides: _*) + .build() + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/RankerParams.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/RankerParams.scala new file mode 100644 index 0000000000..e7785ffb8c --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/RankerParams.scala @@ -0,0 +1,59 @@ +package com.twitter.cr_mixer.param + +import com.twitter.finagle.stats.NullStatsReceiver +import com.twitter.logging.Logger +import com.twitter.timelines.configapi.BaseConfig +import com.twitter.timelines.configapi.BaseConfigBuilder +import com.twitter.timelines.configapi.FSBoundedParam +import com.twitter.timelines.configapi.FSName +import com.twitter.timelines.configapi.FSParam +import com.twitter.timelines.configapi.FeatureSwitchOverrideUtil +import com.twitter.timelines.configapi.Param + +object RankerParams { + + object MaxCandidatesToRank + extends FSBoundedParam[Int]( + name = "twistly_core_max_candidates_to_rank", + default = 2000, + min = 0, + max = 9999 + ) + + object EnableBlueVerifiedTopK + extends FSParam[Boolean]( + name = "twistly_core_blue_verified_top_k", + default = true + ) + + val AllParams: Seq[Param[_] with FSName] = Seq( + MaxCandidatesToRank, + EnableBlueVerifiedTopK + ) + + lazy val config: BaseConfig = { + + val booleanOverrides = FeatureSwitchOverrideUtil.getBooleanFSOverrides(EnableBlueVerifiedTopK) + + val boundedDurationFSOverrides = + FeatureSwitchOverrideUtil.getBoundedDurationFSOverrides() + + val intOverrides = FeatureSwitchOverrideUtil.getBoundedIntFSOverrides( + MaxCandidatesToRank + ) + + val enumOverrides = FeatureSwitchOverrideUtil.getEnumFSOverrides( + NullStatsReceiver, + Logger(getClass), + ) + val stringFSOverrides = FeatureSwitchOverrideUtil.getStringFSOverrides() + + BaseConfigBuilder() + .set(booleanOverrides: _*) + .set(boundedDurationFSOverrides: _*) + .set(intOverrides: _*) + .set(enumOverrides: _*) + .set(stringFSOverrides: _*) + .build() + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/RealGraphInParams.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/RealGraphInParams.scala new file mode 100644 index 0000000000..7614ca0eb8 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/RealGraphInParams.scala @@ -0,0 +1,25 @@ +package com.twitter.cr_mixer.param + +import com.twitter.timelines.configapi._ + +object RealGraphInParams { + object EnableSourceGraphParam + extends FSParam[Boolean]( + name = "graph_realgraphin_enable_source", + default = false + ) + + val AllParams: Seq[Param[_] with FSName] = Seq( + EnableSourceGraphParam, + ) + + lazy val config: BaseConfig = { + val booleanOverrides = FeatureSwitchOverrideUtil.getBooleanFSOverrides( + EnableSourceGraphParam + ) + + BaseConfigBuilder() + .set(booleanOverrides: _*) + .build() + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/RealGraphOonParams.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/RealGraphOonParams.scala new file mode 100644 index 0000000000..8b303c55b7 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/RealGraphOonParams.scala @@ -0,0 +1,51 @@ +package com.twitter.cr_mixer.param + +import com.twitter.timelines.configapi.BaseConfig +import com.twitter.timelines.configapi.BaseConfigBuilder +import com.twitter.timelines.configapi.FSBoundedParam +import com.twitter.timelines.configapi.FSName +import com.twitter.timelines.configapi.FSParam +import com.twitter.timelines.configapi.FeatureSwitchOverrideUtil +import com.twitter.timelines.configapi.Param + +object RealGraphOonParams { + object EnableSourceParam + extends FSParam[Boolean]( + name = "signal_realgraphoon_enable_source", + default = false + ) + + object EnableSourceGraphParam + extends FSParam[Boolean]( + name = "graph_realgraphoon_enable_source", + default = false + ) + + object MaxConsumerSeedsNumParam + extends FSBoundedParam[Int]( + name = "graph_realgraphoon_max_user_seeds_num", + default = 200, + min = 0, + max = 1000 + ) + + val AllParams: Seq[Param[_] with FSName] = Seq( + EnableSourceParam, + EnableSourceGraphParam, + MaxConsumerSeedsNumParam + ) + + lazy val config: BaseConfig = { + val booleanOverrides = FeatureSwitchOverrideUtil.getBooleanFSOverrides( + EnableSourceParam, + EnableSourceGraphParam + ) + + val intOverrides = FeatureSwitchOverrideUtil.getBoundedIntFSOverrides(MaxConsumerSeedsNumParam) + + BaseConfigBuilder() + .set(booleanOverrides: _*) + .set(intOverrides: _*) + .build() + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/RecentFollowsParams.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/RecentFollowsParams.scala new file mode 100644 index 0000000000..ecb75c82f3 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/RecentFollowsParams.scala @@ -0,0 +1,27 @@ +package com.twitter.cr_mixer.param + +import com.twitter.timelines.configapi.BaseConfig +import com.twitter.timelines.configapi.BaseConfigBuilder +import com.twitter.timelines.configapi.FSName +import com.twitter.timelines.configapi.FSParam +import com.twitter.timelines.configapi.FeatureSwitchOverrideUtil +import com.twitter.timelines.configapi.Param + +object RecentFollowsParams { + object EnableSourceParam + extends FSParam[Boolean]( + name = "twistly_recentfollows_enable_source", + default = true + ) + + val AllParams: Seq[Param[_] with FSName] = Seq(EnableSourceParam) + lazy val config: BaseConfig = { + val booleanOverrides = FeatureSwitchOverrideUtil.getBooleanFSOverrides( + EnableSourceParam + ) + + BaseConfigBuilder() + .set(booleanOverrides: _*) + .build() + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/RecentNegativeSignalParams.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/RecentNegativeSignalParams.scala new file mode 100644 index 0000000000..429d6daba7 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/RecentNegativeSignalParams.scala @@ -0,0 +1,39 @@ +package com.twitter.cr_mixer.param + +import com.twitter.finagle.stats.NullStatsReceiver +import com.twitter.logging.Logger +import com.twitter.timelines.configapi.BaseConfig +import com.twitter.timelines.configapi.BaseConfigBuilder +import com.twitter.timelines.configapi.FSName +import com.twitter.timelines.configapi.FSParam +import com.twitter.timelines.configapi.FeatureSwitchOverrideUtil +import com.twitter.timelines.configapi.Param + +object RecentNegativeSignalParams { + object EnableSourceParam + extends FSParam[Boolean]( + name = "twistly_recentnegativesignals_enable_source", + default = false + ) + + val AllParams: Seq[Param[_] with FSName] = Seq( + EnableSourceParam + ) + + lazy val config: BaseConfig = { + val enumOverrides = FeatureSwitchOverrideUtil.getEnumFSOverrides( + NullStatsReceiver, + Logger(getClass), + ) + + val booleanOverrides = FeatureSwitchOverrideUtil.getBooleanFSOverrides( + EnableSourceParam + ) + + val doubleOverrides = + FeatureSwitchOverrideUtil.getBoundedDoubleFSOverrides() + + BaseConfigBuilder() + .set(booleanOverrides: _*).set(doubleOverrides: _*).set(enumOverrides: _*).build() + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/RecentNotificationsParams.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/RecentNotificationsParams.scala new file mode 100644 index 0000000000..641118a05e --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/RecentNotificationsParams.scala @@ -0,0 +1,28 @@ +package com.twitter.cr_mixer.param + +import com.twitter.timelines.configapi.BaseConfig +import com.twitter.timelines.configapi.BaseConfigBuilder +import com.twitter.timelines.configapi.FSName +import com.twitter.timelines.configapi.FSParam +import com.twitter.timelines.configapi.FeatureSwitchOverrideUtil +import com.twitter.timelines.configapi.Param + +object RecentNotificationsParams { + object EnableSourceParam + extends FSParam[Boolean]( + name = "twistly_recentnotifications_enable_source", + default = false + ) + + val AllParams: Seq[Param[_] with FSName] = Seq(EnableSourceParam) + + lazy val config: BaseConfig = { + val booleanOverrides = FeatureSwitchOverrideUtil.getBooleanFSOverrides( + EnableSourceParam + ) + + BaseConfigBuilder() + .set(booleanOverrides: _*) + .build() + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/RecentOriginalTweetsParams.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/RecentOriginalTweetsParams.scala new file mode 100644 index 0000000000..5b485e61f3 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/RecentOriginalTweetsParams.scala @@ -0,0 +1,28 @@ +package com.twitter.cr_mixer.param + +import com.twitter.timelines.configapi.BaseConfig +import com.twitter.timelines.configapi.BaseConfigBuilder +import com.twitter.timelines.configapi.FSName +import com.twitter.timelines.configapi.FSParam +import com.twitter.timelines.configapi.FeatureSwitchOverrideUtil +import com.twitter.timelines.configapi.Param + +object RecentOriginalTweetsParams { + + // Source params + object EnableSourceParam + extends FSParam[Boolean]( + name = "twistly_recentoriginaltweets_enable_source", + default = false + ) + + val AllParams: Seq[Param[_] with FSName] = Seq(EnableSourceParam) + + lazy val config: BaseConfig = { + val booleanOverrides = FeatureSwitchOverrideUtil.getBooleanFSOverrides(EnableSourceParam) + + BaseConfigBuilder() + .set(booleanOverrides: _*) + .build() + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/RecentReplyTweetsParams.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/RecentReplyTweetsParams.scala new file mode 100644 index 0000000000..7e6617c6db --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/RecentReplyTweetsParams.scala @@ -0,0 +1,27 @@ +package com.twitter.cr_mixer.param + +import com.twitter.timelines.configapi.BaseConfig +import com.twitter.timelines.configapi.BaseConfigBuilder +import com.twitter.timelines.configapi.FSName +import com.twitter.timelines.configapi.FSParam +import com.twitter.timelines.configapi.FeatureSwitchOverrideUtil +import com.twitter.timelines.configapi.Param + +object RecentReplyTweetsParams { + // Source params + object EnableSourceParam + extends FSParam[Boolean]( + name = "twistly_recentreplytweets_enable_source", + default = false + ) + + val AllParams: Seq[Param[_] with FSName] = Seq(EnableSourceParam) + + lazy val config: BaseConfig = { + val booleanOverrides = FeatureSwitchOverrideUtil.getBooleanFSOverrides(EnableSourceParam) + + BaseConfigBuilder() + .set(booleanOverrides: _*) + .build() + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/RecentRetweetsParams.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/RecentRetweetsParams.scala new file mode 100644 index 0000000000..93c1fe356c --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/RecentRetweetsParams.scala @@ -0,0 +1,30 @@ +package com.twitter.cr_mixer.param + +import com.twitter.timelines.configapi.BaseConfig +import com.twitter.timelines.configapi.BaseConfigBuilder +import com.twitter.timelines.configapi.FSName +import com.twitter.timelines.configapi.FSParam +import com.twitter.timelines.configapi.FeatureSwitchOverrideUtil +import com.twitter.timelines.configapi.Param + +object RecentRetweetsParams { + + // Source params + object EnableSourceParam + extends FSParam[Boolean]( + name = "twistly_recentretweets_enable_source", + default = false + ) + + val AllParams: Seq[Param[_] with FSName] = Seq(EnableSourceParam) + + lazy val config: BaseConfig = { + val booleanOverrides = FeatureSwitchOverrideUtil.getBooleanFSOverrides( + EnableSourceParam + ) + + BaseConfigBuilder() + .set(booleanOverrides: _*) + .build() + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/RecentTweetFavoritesParams.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/RecentTweetFavoritesParams.scala new file mode 100644 index 0000000000..22d0d6a709 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/RecentTweetFavoritesParams.scala @@ -0,0 +1,29 @@ +package com.twitter.cr_mixer.param + +import com.twitter.timelines.configapi.BaseConfig +import com.twitter.timelines.configapi.BaseConfigBuilder +import com.twitter.timelines.configapi.FSName +import com.twitter.timelines.configapi.FSParam +import com.twitter.timelines.configapi.FeatureSwitchOverrideUtil +import com.twitter.timelines.configapi.Param + +object RecentTweetFavoritesParams { + // Source params + object EnableSourceParam + extends FSParam[Boolean]( + name = "twistly_recenttweetfavorites_enable_source", + default = true + ) + + val AllParams: Seq[Param[_] with FSName] = Seq(EnableSourceParam) + + lazy val config: BaseConfig = { + val booleanOverrides = FeatureSwitchOverrideUtil.getBooleanFSOverrides( + EnableSourceParam + ) + + BaseConfigBuilder() + .set(booleanOverrides: _*) + .build() + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/RelatedTweetGlobalParams.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/RelatedTweetGlobalParams.scala new file mode 100644 index 0000000000..f5e5ee21e2 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/RelatedTweetGlobalParams.scala @@ -0,0 +1,32 @@ +package com.twitter.cr_mixer.param + +import com.twitter.timelines.configapi.BaseConfig +import com.twitter.timelines.configapi.BaseConfigBuilder +import com.twitter.timelines.configapi.FSBoundedParam +import com.twitter.timelines.configapi.FSName +import com.twitter.timelines.configapi.FeatureSwitchOverrideUtil +import com.twitter.timelines.configapi.Param + +object RelatedTweetGlobalParams { + + object MaxCandidatesPerRequestParam + extends FSBoundedParam[Int]( + name = "related_tweet_core_max_candidates_per_request", + default = 100, + min = 0, + max = 500 + ) + + val AllParams: Seq[Param[_] with FSName] = Seq(MaxCandidatesPerRequestParam) + + lazy val config: BaseConfig = { + + val intOverrides = FeatureSwitchOverrideUtil.getBoundedIntFSOverrides( + MaxCandidatesPerRequestParam + ) + + BaseConfigBuilder() + .set(intOverrides: _*) + .build() + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/RelatedTweetProducerBasedParams.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/RelatedTweetProducerBasedParams.scala new file mode 100644 index 0000000000..3851f61448 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/RelatedTweetProducerBasedParams.scala @@ -0,0 +1,111 @@ +package com.twitter.cr_mixer.param + +import com.twitter.timelines.configapi.BaseConfig +import com.twitter.timelines.configapi.BaseConfigBuilder +import com.twitter.timelines.configapi.FSBoundedParam +import com.twitter.timelines.configapi.FSName +import com.twitter.timelines.configapi.FSParam +import com.twitter.timelines.configapi.FeatureSwitchOverrideUtil +import com.twitter.timelines.configapi.Param + +object RelatedTweetProducerBasedParams { + + // UTG params + object EnableUTGParam + extends FSParam[Boolean]( + name = "related_tweet_producer_based_enable_utg", + default = false + ) + + // SimClusters params + object EnableSimClustersANNParam + extends FSParam[Boolean]( + name = "related_tweet_producer_based_enable_simclusters", + default = true + ) + + // Filter params + object SimClustersMinScoreParam + extends FSBoundedParam[Double]( + name = "related_tweet_producer_based_filter_simclusters_min_score", + default = 0.0, + min = 0.0, + max = 1.0 + ) + + // Experimental SimClusters ANN params + object EnableExperimentalSimClustersANNParam + extends FSParam[Boolean]( + name = "related_tweet_producer_based_enable_experimental_simclusters_ann", + default = false + ) + + // SimClusters ANN cluster 1 params + object EnableSimClustersANN1Param + extends FSParam[Boolean]( + name = "related_tweet_producer_based_enable_simclusters_ann_1", + default = false + ) + + // SimClusters ANN cluster 2 params + object EnableSimClustersANN2Param + extends FSParam[Boolean]( + name = "related_tweet_producer_based_enable_simclusters_ann_2", + default = false + ) + + // SimClusters ANN cluster 3 params + object EnableSimClustersANN3Param + extends FSParam[Boolean]( + name = "related_tweet_producer_based_enable_simclusters_ann_3", + default = false + ) + + // SimClusters ANN cluster 3 params + object EnableSimClustersANN5Param + extends FSParam[Boolean]( + name = "related_tweet_producer_based_enable_simclusters_ann_5", + default = false + ) + + // SimClusters ANN cluster 4 params + object EnableSimClustersANN4Param + extends FSParam[Boolean]( + name = "related_tweet_producer_based_enable_simclusters_ann_4", + default = false + ) + val AllParams: Seq[Param[_] with FSName] = Seq( + EnableUTGParam, + EnableSimClustersANNParam, + EnableSimClustersANN1Param, + EnableSimClustersANN2Param, + EnableSimClustersANN3Param, + EnableSimClustersANN5Param, + EnableSimClustersANN4Param, + EnableExperimentalSimClustersANNParam, + SimClustersMinScoreParam + ) + + lazy val config: BaseConfig = { + + val booleanOverrides = FeatureSwitchOverrideUtil.getBooleanFSOverrides( + EnableUTGParam, + EnableSimClustersANNParam, + EnableSimClustersANN1Param, + EnableSimClustersANN2Param, + EnableSimClustersANN3Param, + EnableSimClustersANN5Param, + EnableSimClustersANN4Param, + EnableExperimentalSimClustersANNParam + ) + + val doubleOverrides = FeatureSwitchOverrideUtil.getBoundedDoubleFSOverrides( + SimClustersMinScoreParam + ) + + BaseConfigBuilder() + .set(booleanOverrides: _*) + .set(doubleOverrides: _*) + .build() + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/RelatedTweetTweetBasedParams.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/RelatedTweetTweetBasedParams.scala new file mode 100644 index 0000000000..10d01a5d14 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/RelatedTweetTweetBasedParams.scala @@ -0,0 +1,141 @@ +package com.twitter.cr_mixer.param + +import com.twitter.timelines.configapi.BaseConfig +import com.twitter.timelines.configapi.BaseConfigBuilder +import com.twitter.timelines.configapi.FSBoundedParam +import com.twitter.timelines.configapi.FSName +import com.twitter.timelines.configapi.FSParam +import com.twitter.timelines.configapi.FeatureSwitchOverrideUtil +import com.twitter.timelines.configapi.Param + +object RelatedTweetTweetBasedParams { + + // UTG params + object EnableUTGParam + extends FSParam[Boolean]( + name = "related_tweet_tweet_based_enable_utg", + default = false + ) + + // UVG params + object EnableUVGParam + extends FSParam[Boolean]( + name = "related_tweet_tweet_based_enable_uvg", + default = false + ) + + // UAG params + object EnableUAGParam + extends FSParam[Boolean]( + name = "related_tweet_tweet_based_enable_uag", + default = false + ) + + // SimClusters params + object EnableSimClustersANNParam + extends FSParam[Boolean]( + name = "related_tweet_tweet_based_enable_simclusters", + default = true + ) + + // Experimental SimClusters ANN params + object EnableExperimentalSimClustersANNParam + extends FSParam[Boolean]( + name = "related_tweet_tweet_based_enable_experimental_simclusters_ann", + default = false + ) + + // SimClusters ANN cluster 1 params + object EnableSimClustersANN1Param + extends FSParam[Boolean]( + name = "related_tweet_tweet_based_enable_simclusters_ann_1", + default = false + ) + + // SimClusters ANN cluster 2 params + object EnableSimClustersANN2Param + extends FSParam[Boolean]( + name = "related_tweet_tweet_based_enable_simclusters_ann_2", + default = false + ) + + // SimClusters ANN cluster 3 params + object EnableSimClustersANN3Param + extends FSParam[Boolean]( + name = "related_tweet_tweet_based_enable_simclusters_ann_3", + default = false + ) + + // SimClusters ANN cluster 5 params + object EnableSimClustersANN5Param + extends FSParam[Boolean]( + name = "related_tweet_tweet_based_enable_simclusters_ann_5", + default = false + ) + + object EnableSimClustersANN4Param + extends FSParam[Boolean]( + name = "related_tweet_tweet_based_enable_simclusters_ann_4", + default = false + ) + // TwHIN params + object EnableTwHINParam + extends FSParam[Boolean]( + name = "related_tweet_tweet_based_enable_twhin", + default = false + ) + + // QIG params + object EnableQigSimilarTweetsParam + extends FSParam[Boolean]( + name = "related_tweet_tweet_based_enable_qig_similar_tweets", + default = false + ) + + // Filter params + object SimClustersMinScoreParam + extends FSBoundedParam[Double]( + name = "related_tweet_tweet_based_filter_simclusters_min_score", + default = 0.3, + min = 0.0, + max = 1.0 + ) + + val AllParams: Seq[Param[_] with FSName] = Seq( + EnableTwHINParam, + EnableQigSimilarTweetsParam, + EnableUTGParam, + EnableUVGParam, + EnableSimClustersANNParam, + EnableSimClustersANN2Param, + EnableSimClustersANN3Param, + EnableSimClustersANN5Param, + EnableSimClustersANN4Param, + EnableExperimentalSimClustersANNParam, + SimClustersMinScoreParam + ) + + lazy val config: BaseConfig = { + + val booleanOverrides = FeatureSwitchOverrideUtil.getBooleanFSOverrides( + EnableTwHINParam, + EnableQigSimilarTweetsParam, + EnableUTGParam, + EnableUVGParam, + EnableSimClustersANNParam, + EnableSimClustersANN2Param, + EnableSimClustersANN3Param, + EnableSimClustersANN5Param, + EnableSimClustersANN4Param, + EnableExperimentalSimClustersANNParam + ) + + val doubleOverrides = + FeatureSwitchOverrideUtil.getBoundedDoubleFSOverrides(SimClustersMinScoreParam) + + BaseConfigBuilder() + .set(booleanOverrides: _*) + .set(doubleOverrides: _*) + .build() + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/RelatedVideoTweetGlobalParams.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/RelatedVideoTweetGlobalParams.scala new file mode 100644 index 0000000000..eeed18e6cf --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/RelatedVideoTweetGlobalParams.scala @@ -0,0 +1,32 @@ +package com.twitter.cr_mixer.param + +import com.twitter.timelines.configapi.BaseConfig +import com.twitter.timelines.configapi.BaseConfigBuilder +import com.twitter.timelines.configapi.FSBoundedParam +import com.twitter.timelines.configapi.FSName +import com.twitter.timelines.configapi.FeatureSwitchOverrideUtil +import com.twitter.timelines.configapi.Param + +object RelatedVideoTweetGlobalParams { + + object MaxCandidatesPerRequestParam + extends FSBoundedParam[Int]( + name = "related_video_tweet_core_max_candidates_per_request", + default = 100, + min = 0, + max = 500 + ) + + val AllParams: Seq[Param[_] with FSName] = Seq(MaxCandidatesPerRequestParam) + + lazy val config: BaseConfig = { + + val intOverrides = FeatureSwitchOverrideUtil.getBoundedIntFSOverrides( + MaxCandidatesPerRequestParam + ) + + BaseConfigBuilder() + .set(intOverrides: _*) + .build() + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/RelatedVideoTweetTweetBasedParams.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/RelatedVideoTweetTweetBasedParams.scala new file mode 100644 index 0000000000..3b40653bc2 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/RelatedVideoTweetTweetBasedParams.scala @@ -0,0 +1,134 @@ +package com.twitter.cr_mixer.param + +import com.twitter.timelines.configapi.BaseConfig +import com.twitter.timelines.configapi.BaseConfigBuilder +import com.twitter.timelines.configapi.FSBoundedParam +import com.twitter.timelines.configapi.FSName +import com.twitter.timelines.configapi.FSParam +import com.twitter.timelines.configapi.FeatureSwitchOverrideUtil +import com.twitter.timelines.configapi.Param + +object RelatedVideoTweetTweetBasedParams { + + // UTG params + object EnableUTGParam + extends FSParam[Boolean]( + name = "related_video_tweet_tweet_based_enable_utg", + default = false + ) + + // SimClusters params + object EnableSimClustersANNParam + extends FSParam[Boolean]( + name = "related_video_tweet_tweet_based_enable_simclusters", + default = true + ) + + // Experimental SimClusters ANN params + object EnableExperimentalSimClustersANNParam + extends FSParam[Boolean]( + name = "related_video_tweet_tweet_based_enable_experimental_simclusters_ann", + default = false + ) + + // SimClusters ANN cluster 1 params + object EnableSimClustersANN1Param + extends FSParam[Boolean]( + name = "related_video_tweet_tweet_based_enable_simclusters_ann_1", + default = false + ) + + // SimClusters ANN cluster 2 params + object EnableSimClustersANN2Param + extends FSParam[Boolean]( + name = "related_video_tweet_tweet_based_enable_simclusters_ann_2", + default = false + ) + + // SimClusters ANN cluster 3 params + object EnableSimClustersANN3Param + extends FSParam[Boolean]( + name = "related_video_tweet_tweet_based_enable_simclusters_ann_3", + default = false + ) + + // SimClusters ANN cluster 5 params + object EnableSimClustersANN5Param + extends FSParam[Boolean]( + name = "related_video_tweet_tweet_based_enable_simclusters_ann_5", + default = false + ) + + // SimClusters ANN cluster 4 params + object EnableSimClustersANN4Param + extends FSParam[Boolean]( + name = "related_video_tweet_tweet_based_enable_simclusters_ann_4", + default = false + ) + // TwHIN params + object EnableTwHINParam + extends FSParam[Boolean]( + name = "related_video_tweet_tweet_based_enable_twhin", + default = false + ) + + // QIG params + object EnableQigSimilarTweetsParam + extends FSParam[Boolean]( + name = "related_video_tweet_tweet_based_enable_qig_similar_tweets", + default = false + ) + + // Filter params + object SimClustersMinScoreParam + extends FSBoundedParam[Double]( + name = "related_video_tweet_tweet_based_filter_simclusters_min_score", + default = 0.3, + min = 0.0, + max = 1.0 + ) + + object EnableUVGParam + extends FSParam[Boolean]( + name = "related_video_tweet_tweet_based_enable_uvg", + default = false + ) + + val AllParams: Seq[Param[_] with FSName] = Seq( + EnableTwHINParam, + EnableQigSimilarTweetsParam, + EnableUTGParam, + EnableUVGParam, + EnableSimClustersANNParam, + EnableSimClustersANN2Param, + EnableSimClustersANN3Param, + EnableSimClustersANN5Param, + EnableSimClustersANN4Param, + EnableExperimentalSimClustersANNParam, + SimClustersMinScoreParam + ) + + lazy val config: BaseConfig = { + + val booleanOverrides = FeatureSwitchOverrideUtil.getBooleanFSOverrides( + EnableTwHINParam, + EnableQigSimilarTweetsParam, + EnableUTGParam, + EnableUVGParam, + EnableSimClustersANNParam, + EnableSimClustersANN2Param, + EnableSimClustersANN3Param, + EnableSimClustersANN5Param, + EnableSimClustersANN4Param, + EnableExperimentalSimClustersANNParam + ) + + val doubleOverrides = + FeatureSwitchOverrideUtil.getBoundedDoubleFSOverrides(SimClustersMinScoreParam) + + BaseConfigBuilder() + .set(booleanOverrides: _*) + .set(doubleOverrides: _*) + .build() + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/RepeatedProfileVisitsParams.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/RepeatedProfileVisitsParams.scala new file mode 100644 index 0000000000..4cb205de94 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/RepeatedProfileVisitsParams.scala @@ -0,0 +1,72 @@ +package com.twitter.cr_mixer.param + +import com.twitter.finagle.stats.NullStatsReceiver +import com.twitter.logging.Logger +import com.twitter.usersignalservice.thriftscala.SignalType +import com.twitter.timelines.configapi.BaseConfig +import com.twitter.timelines.configapi.BaseConfigBuilder +import com.twitter.timelines.configapi.FSBoundedParam +import com.twitter.timelines.configapi.FSEnumParam +import com.twitter.timelines.configapi.FSName +import com.twitter.timelines.configapi.FSParam +import com.twitter.timelines.configapi.FeatureSwitchOverrideUtil +import com.twitter.timelines.configapi.Param + +object RepeatedProfileVisitsParams { + object ProfileMinVisitParam extends Enumeration { + protected case class SignalTypeValue(signalType: SignalType) extends super.Val + import scala.language.implicitConversions + implicit def valueToSignalTypeValue(x: Value): SignalTypeValue = + x.asInstanceOf[SignalTypeValue] + + val TotalVisitsInPast180Days = SignalTypeValue(SignalType.RepeatedProfileVisit180dMinVisit6V1) + val TotalVisitsInPast90Days = SignalTypeValue(SignalType.RepeatedProfileVisit90dMinVisit6V1) + val TotalVisitsInPast14Days = SignalTypeValue(SignalType.RepeatedProfileVisit14dMinVisit2V1) + val TotalVisitsInPast180DaysNoNegative = SignalTypeValue( + SignalType.RepeatedProfileVisit180dMinVisit6V1NoNegative) + val TotalVisitsInPast90DaysNoNegative = SignalTypeValue( + SignalType.RepeatedProfileVisit90dMinVisit6V1NoNegative) + val TotalVisitsInPast14DaysNoNegative = SignalTypeValue( + SignalType.RepeatedProfileVisit14dMinVisit2V1NoNegative) + } + + object EnableSourceParam + extends FSParam[Boolean]( + name = "twistly_repeatedprofilevisits_enable_source", + default = true + ) + + object MinScoreParam + extends FSBoundedParam[Double]( + name = "twistly_repeatedprofilevisits_min_score", + default = 0.5, + min = 0.0, + max = 1.0 + ) + + object ProfileMinVisitType + extends FSEnumParam[ProfileMinVisitParam.type]( + name = "twistly_repeatedprofilevisits_min_visit_type_id", + default = ProfileMinVisitParam.TotalVisitsInPast14Days, + enum = ProfileMinVisitParam + ) + + val AllParams: Seq[Param[_] with FSName] = Seq(EnableSourceParam, ProfileMinVisitType) + + lazy val config: BaseConfig = { + val booleanOverrides = FeatureSwitchOverrideUtil.getBooleanFSOverrides( + EnableSourceParam + ) + + val enumOverrides = FeatureSwitchOverrideUtil.getEnumFSOverrides( + NullStatsReceiver, + Logger(getClass), + ProfileMinVisitType + ) + + BaseConfigBuilder() + .set(booleanOverrides: _*) + .set(enumOverrides: _*) + .build() + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/SimClustersANNParams.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/SimClustersANNParams.scala new file mode 100644 index 0000000000..b650d5123b --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/SimClustersANNParams.scala @@ -0,0 +1,76 @@ +package com.twitter.cr_mixer.param + +import com.twitter.timelines.configapi.BaseConfig +import com.twitter.timelines.configapi.BaseConfigBuilder +import com.twitter.timelines.configapi.FSName +import com.twitter.timelines.configapi.FSParam +import com.twitter.timelines.configapi.FeatureSwitchOverrideUtil +import com.twitter.timelines.configapi.Param + +object SimClustersANNParams { + + // Different SimClusters ANN cluster has its own config id (model slot) + object SimClustersANNConfigId + extends FSParam[String]( + name = "similarity_simclusters_ann_simclusters_ann_config_id", + default = "Default" + ) + + object SimClustersANN1ConfigId + extends FSParam[String]( + name = "similarity_simclusters_ann_simclusters_ann_1_config_id", + default = "20220810" + ) + + object SimClustersANN2ConfigId + extends FSParam[String]( + name = "similarity_simclusters_ann_simclusters_ann_2_config_id", + default = "20220818" + ) + + object SimClustersANN3ConfigId + extends FSParam[String]( + name = "similarity_simclusters_ann_simclusters_ann_3_config_id", + default = "20220819" + ) + + object SimClustersANN5ConfigId + extends FSParam[String]( + name = "similarity_simclusters_ann_simclusters_ann_5_config_id", + default = "20221221" + ) + object SimClustersANN4ConfigId + extends FSParam[String]( + name = "similarity_simclusters_ann_simclusters_ann_4_config_id", + default = "20221220" + ) + object ExperimentalSimClustersANNConfigId + extends FSParam[String]( + name = "similarity_simclusters_ann_experimental_simclusters_ann_config_id", + default = "20220801" + ) + + val AllParams: Seq[Param[_] with FSName] = Seq( + SimClustersANNConfigId, + SimClustersANN1ConfigId, + SimClustersANN2ConfigId, + SimClustersANN3ConfigId, + SimClustersANN5ConfigId, + ExperimentalSimClustersANNConfigId + ) + + lazy val config: BaseConfig = { + val stringOverrides = FeatureSwitchOverrideUtil.getStringFSOverrides( + SimClustersANNConfigId, + SimClustersANN1ConfigId, + SimClustersANN2ConfigId, + SimClustersANN3ConfigId, + SimClustersANN5ConfigId, + ExperimentalSimClustersANNConfigId + ) + + BaseConfigBuilder() + .set(stringOverrides: _*) + .build() + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/TopicTweetParams.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/TopicTweetParams.scala new file mode 100644 index 0000000000..3ef683f527 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/TopicTweetParams.scala @@ -0,0 +1,115 @@ +package com.twitter.cr_mixer.param + +import com.twitter.conversions.DurationOps._ +import com.twitter.finagle.stats.NullStatsReceiver +import com.twitter.logging.Logger +import com.twitter.timelines.configapi.BaseConfig +import com.twitter.timelines.configapi.BaseConfigBuilder +import com.twitter.timelines.configapi.DurationConversion +import com.twitter.timelines.configapi.FSBoundedParam +import com.twitter.timelines.configapi.FSName +import com.twitter.timelines.configapi.FSParam +import com.twitter.timelines.configapi.FeatureSwitchOverrideUtil +import com.twitter.timelines.configapi.HasDurationConversion +import com.twitter.timelines.configapi.Param +import com.twitter.util.Duration + +object TopicTweetParams { + object MaxTweetAge + extends FSBoundedParam[Duration]( + name = "topic_tweet_candidate_generation_max_tweet_age_hours", + default = 24.hours, + min = 12.hours, + max = 48.hours + ) + with HasDurationConversion { + override val durationConversion: DurationConversion = DurationConversion.FromHours + } + + object MaxTopicTweetCandidatesParam + extends FSBoundedParam[Int]( + name = "topic_tweet_max_candidates_num", + default = 200, + min = 0, + max = 1000 + ) + + object MaxSkitTfgCandidatesParam + extends FSBoundedParam[Int]( + name = "topic_tweet_skit_tfg_max_candidates_num", + default = 100, + min = 0, + max = 1000 + ) + + object MaxSkitHighPrecisionCandidatesParam + extends FSBoundedParam[Int]( + name = "topic_tweet_skit_high_precision_max_candidates_num", + default = 100, + min = 0, + max = 1000 + ) + + object MaxCertoCandidatesParam + extends FSBoundedParam[Int]( + name = "topic_tweet_certo_max_candidates_num", + default = 100, + min = 0, + max = 1000 + ) + + // The min prod score for Certo L2-normalized cosine candidates + object CertoScoreThresholdParam + extends FSBoundedParam[Double]( + name = "topic_tweet_certo_score_threshold", + default = 0.015, + min = 0, + max = 1 + ) + + object SemanticCoreVersionIdParam + extends FSParam[Long]( + name = "semantic_core_version_id", + default = 1380520918896713735L + ) + + val AllParams: Seq[Param[_] with FSName] = Seq( + CertoScoreThresholdParam, + MaxTopicTweetCandidatesParam, + MaxTweetAge, + MaxCertoCandidatesParam, + MaxSkitTfgCandidatesParam, + MaxSkitHighPrecisionCandidatesParam, + SemanticCoreVersionIdParam + ) + + lazy val config: BaseConfig = { + val booleanOverrides = FeatureSwitchOverrideUtil.getBooleanFSOverrides() + + val doubleOverrides = + FeatureSwitchOverrideUtil.getBoundedDoubleFSOverrides(CertoScoreThresholdParam) + + val intOverrides = FeatureSwitchOverrideUtil.getBoundedIntFSOverrides( + MaxCertoCandidatesParam, + MaxSkitTfgCandidatesParam, + MaxSkitHighPrecisionCandidatesParam, + MaxTopicTweetCandidatesParam + ) + + val longOverrides = FeatureSwitchOverrideUtil.getLongFSOverrides(SemanticCoreVersionIdParam) + + val durationFSOverrides = FeatureSwitchOverrideUtil.getDurationFSOverrides(MaxTweetAge) + + val enumOverrides = + FeatureSwitchOverrideUtil.getEnumFSOverrides(NullStatsReceiver, Logger(getClass)) + + BaseConfigBuilder() + .set(booleanOverrides: _*) + .set(doubleOverrides: _*) + .set(intOverrides: _*) + .set(longOverrides: _*) + .set(enumOverrides: _*) + .set(durationFSOverrides: _*) + .build() + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/TweetBasedCandidateGenerationParams.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/TweetBasedCandidateGenerationParams.scala new file mode 100644 index 0000000000..7f94d2e416 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/TweetBasedCandidateGenerationParams.scala @@ -0,0 +1,189 @@ +package com.twitter.cr_mixer.param + +import com.twitter.finagle.stats.NullStatsReceiver +import com.twitter.logging.Logger +import com.twitter.timelines.configapi.BaseConfig +import com.twitter.timelines.configapi.BaseConfigBuilder +import com.twitter.timelines.configapi.FSBoundedParam +import com.twitter.timelines.configapi.FSName +import com.twitter.timelines.configapi.FSParam +import com.twitter.timelines.configapi.FeatureSwitchOverrideUtil +import com.twitter.timelines.configapi.Param + +object TweetBasedCandidateGenerationParams { + + // Source params. Not being used. It is always set to true in prod + object EnableSourceParam + extends FSParam[Boolean]( + name = "tweet_based_candidate_generation_enable_source", + default = false + ) + + // UTG params + object EnableUTGParam + extends FSParam[Boolean]( + name = "tweet_based_candidate_generation_enable_utg", + default = true + ) + + // SimClusters params + object EnableSimClustersANNParam + extends FSParam[Boolean]( + name = "tweet_based_candidate_generation_enable_simclusters", + default = true + ) + + // Experimental SimClusters ANN params + object EnableExperimentalSimClustersANNParam + extends FSParam[Boolean]( + name = "tweet_based_candidate_generation_enable_experimental_simclusters_ann", + default = false + ) + + // SimClusters ANN cluster 1 params + object EnableSimClustersANN1Param + extends FSParam[Boolean]( + name = "tweet_based_candidate_generation_enable_simclusters_ann_1", + default = false + ) + + // SimClusters ANN cluster 2 params + object EnableSimClustersANN2Param + extends FSParam[Boolean]( + name = "tweet_based_candidate_generation_enable_simclusters_ann_2", + default = false + ) + + // SimClusters ANN cluster 3 params + object EnableSimClustersANN3Param + extends FSParam[Boolean]( + name = "tweet_based_candidate_generation_enable_simclusters_ann_3", + default = false + ) + + // SimClusters ANN cluster 3 params + object EnableSimClustersANN5Param + extends FSParam[Boolean]( + name = "tweet_based_candidate_generation_enable_simclusters_ann_5", + default = false + ) + + // SimClusters ANN cluster 4 params + object EnableSimClustersANN4Param + extends FSParam[Boolean]( + name = "tweet_based_candidate_generation_enable_simclusters_ann_4", + default = false + ) + // TwHIN params + object EnableTwHINParam + extends FSParam[Boolean]( + name = "tweet_based_candidate_generation_enable_twhin", + default = false + ) + + // QIG params + object EnableQigSimilarTweetsParam + extends FSParam[Boolean]( + name = "tweet_based_candidate_generation_enable_qig_similar_tweets", + default = false + ) + + object QigMaxNumSimilarTweetsParam + extends FSBoundedParam[Int]( + name = "tweet_based_candidate_generation_qig_max_num_similar_tweets", + default = 100, + min = 10, + max = 100 + ) + + // UVG params + object EnableUVGParam + extends FSParam[Boolean]( + name = "tweet_based_candidate_generation_enable_uvg", + default = false + ) + + // UAG params + object EnableUAGParam + extends FSParam[Boolean]( + name = "tweet_based_candidate_generation_enable_uag", + default = false + ) + + // Filter params + object SimClustersMinScoreParam + extends FSBoundedParam[Double]( + name = "tweet_based_candidate_generation_filter_simclusters_min_score", + default = 0.5, + min = 0.0, + max = 1.0 + ) + + // for learning DDG that has a higher threshold for video based SANN + object SimClustersVideoBasedMinScoreParam + extends FSBoundedParam[Double]( + name = "tweet_based_candidate_generation_filter_simclusters_video_based_min_score", + default = 0.5, + min = 0.0, + max = 1.0 + ) + + val AllParams: Seq[Param[_] with FSName] = Seq( + EnableSourceParam, + EnableTwHINParam, + EnableQigSimilarTweetsParam, + EnableUTGParam, + EnableUVGParam, + EnableUAGParam, + EnableSimClustersANNParam, + EnableSimClustersANN1Param, + EnableSimClustersANN2Param, + EnableSimClustersANN3Param, + EnableSimClustersANN5Param, + EnableSimClustersANN4Param, + EnableExperimentalSimClustersANNParam, + SimClustersMinScoreParam, + SimClustersVideoBasedMinScoreParam, + QigMaxNumSimilarTweetsParam, + ) + + lazy val config: BaseConfig = { + + val booleanOverrides = FeatureSwitchOverrideUtil.getBooleanFSOverrides( + EnableSourceParam, + EnableTwHINParam, + EnableQigSimilarTweetsParam, + EnableUTGParam, + EnableUVGParam, + EnableUAGParam, + EnableSimClustersANNParam, + EnableSimClustersANN1Param, + EnableSimClustersANN2Param, + EnableSimClustersANN3Param, + EnableSimClustersANN5Param, + EnableSimClustersANN4Param, + EnableExperimentalSimClustersANNParam, + ) + + val doubleOverrides = + FeatureSwitchOverrideUtil.getBoundedDoubleFSOverrides( + SimClustersMinScoreParam, + SimClustersVideoBasedMinScoreParam) + + val enumOverrides = FeatureSwitchOverrideUtil.getEnumFSOverrides( + NullStatsReceiver, + Logger(getClass), + ) + + val intOverrides = FeatureSwitchOverrideUtil.getBoundedIntFSOverrides( + QigMaxNumSimilarTweetsParam + ) + + BaseConfigBuilder() + .set(booleanOverrides: _*) + .set(doubleOverrides: _*) + .set(enumOverrides: _*) + .set(intOverrides: _*) + .build() + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/TweetBasedTwHINParams.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/TweetBasedTwHINParams.scala new file mode 100644 index 0000000000..c4ecfc6fb9 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/TweetBasedTwHINParams.scala @@ -0,0 +1,30 @@ +package com.twitter.cr_mixer.param + +import com.twitter.cr_mixer.model.ModelConfig +import com.twitter.timelines.configapi.BaseConfig +import com.twitter.timelines.configapi.BaseConfigBuilder +import com.twitter.timelines.configapi.FSName +import com.twitter.timelines.configapi.FSParam +import com.twitter.timelines.configapi.FeatureSwitchOverrideUtil +import com.twitter.timelines.configapi.Param + +object TweetBasedTwHINParams { + object ModelIdParam + extends FSParam[String]( + name = "tweet_based_twhin_model_id", + default = ModelConfig.TweetBasedTwHINRegularUpdateAll20221024, + ) + + val AllParams: Seq[Param[_] with FSName] = Seq(ModelIdParam) + + lazy val config: BaseConfig = { + val stringFSOverrides = + FeatureSwitchOverrideUtil.getStringFSOverrides( + ModelIdParam + ) + + BaseConfigBuilder() + .set(stringFSOverrides: _*) + .build() + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/TweetBasedUserAdGraphParams.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/TweetBasedUserAdGraphParams.scala new file mode 100644 index 0000000000..9e994b16b2 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/TweetBasedUserAdGraphParams.scala @@ -0,0 +1,58 @@ +package com.twitter.cr_mixer.param + +import com.twitter.timelines.configapi.BaseConfig +import com.twitter.timelines.configapi.BaseConfigBuilder +import com.twitter.timelines.configapi.FSBoundedParam +import com.twitter.timelines.configapi.FSName +import com.twitter.timelines.configapi.FeatureSwitchOverrideUtil +import com.twitter.timelines.configapi.Param + +object TweetBasedUserAdGraphParams { + + object MinCoOccurrenceParam + extends FSBoundedParam[Int]( + name = "tweet_based_user_ad_graph_min_co_occurrence", + default = 1, + min = 0, + max = 500 + ) + + object ConsumersBasedMinScoreParam + extends FSBoundedParam[Double]( + name = "tweet_based_user_ad_graph_consumers_based_min_score", + default = 0.0, + min = 0.0, + max = 10.0 + ) + + object MaxConsumerSeedsNumParam + extends FSBoundedParam[Int]( + name = "tweet_based_user_ad_graph_max_user_seeds_num", + default = 100, + min = 0, + max = 300 + ) + + val AllParams: Seq[Param[_] with FSName] = Seq( + MinCoOccurrenceParam, + MaxConsumerSeedsNumParam, + ConsumersBasedMinScoreParam + ) + + lazy val config: BaseConfig = { + + val intOverrides = FeatureSwitchOverrideUtil.getBoundedIntFSOverrides( + MinCoOccurrenceParam, + MaxConsumerSeedsNumParam + ) + + val doubleOverrides = + FeatureSwitchOverrideUtil.getBoundedDoubleFSOverrides(ConsumersBasedMinScoreParam) + + BaseConfigBuilder() + .set(intOverrides: _*) + .set(doubleOverrides: _*) + .build() + } + +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/TweetBasedUserTweetGraphParams.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/TweetBasedUserTweetGraphParams.scala new file mode 100644 index 0000000000..8cc42f81f2 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/TweetBasedUserTweetGraphParams.scala @@ -0,0 +1,89 @@ +package com.twitter.cr_mixer.param + +import com.twitter.timelines.configapi.BaseConfig +import com.twitter.timelines.configapi.BaseConfigBuilder +import com.twitter.timelines.configapi.FSBoundedParam +import com.twitter.timelines.configapi.FSName +import com.twitter.timelines.configapi.FSParam +import com.twitter.timelines.configapi.FeatureSwitchOverrideUtil +import com.twitter.timelines.configapi.Param + +object TweetBasedUserTweetGraphParams { + + object MinCoOccurrenceParam + extends FSBoundedParam[Int]( + name = "tweet_based_user_tweet_graph_min_co_occurrence", + default = 3, + min = 0, + max = 500 + ) + + object TweetBasedMinScoreParam + extends FSBoundedParam[Double]( + name = "tweet_based_user_tweet_graph_tweet_based_min_score", + default = 0.5, + min = 0.0, + max = 10.0 + ) + + object ConsumersBasedMinScoreParam + extends FSBoundedParam[Double]( + name = "tweet_based_user_tweet_graph_consumers_based_min_score", + default = 4.0, + min = 0.0, + max = 10.0 + ) + object MaxConsumerSeedsNumParam + extends FSBoundedParam[Int]( + name = "tweet_based_user_tweet_graph_max_user_seeds_num", + default = 100, + min = 0, + max = 300 + ) + + object EnableCoverageExpansionOldTweetParam + extends FSParam[Boolean]( + name = "tweet_based_user_tweet_graph_enable_coverage_expansion_old_tweet", + default = false + ) + + object EnableCoverageExpansionAllTweetParam + extends FSParam[Boolean]( + name = "tweet_based_user_tweet_graph_enable_coverage_expansion_all_tweet", + default = false + ) + + val AllParams: Seq[Param[_] with FSName] = Seq( + EnableCoverageExpansionAllTweetParam, + EnableCoverageExpansionOldTweetParam, + MinCoOccurrenceParam, + MaxConsumerSeedsNumParam, + TweetBasedMinScoreParam, + ConsumersBasedMinScoreParam + ) + + lazy val config: BaseConfig = { + + val booleanOverrides = FeatureSwitchOverrideUtil.getBooleanFSOverrides( + EnableCoverageExpansionAllTweetParam, + EnableCoverageExpansionOldTweetParam + ) + + val intOverrides = FeatureSwitchOverrideUtil.getBoundedIntFSOverrides( + MinCoOccurrenceParam, + MaxConsumerSeedsNumParam + ) + + val doubleOverrides = + FeatureSwitchOverrideUtil.getBoundedDoubleFSOverrides( + TweetBasedMinScoreParam, + ConsumersBasedMinScoreParam) + + BaseConfigBuilder() + .set(booleanOverrides: _*) + .set(intOverrides: _*) + .set(doubleOverrides: _*) + .build() + } + +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/TweetBasedUserVideoGraphParams.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/TweetBasedUserVideoGraphParams.scala new file mode 100644 index 0000000000..0de5d2df79 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/TweetBasedUserVideoGraphParams.scala @@ -0,0 +1,81 @@ +package com.twitter.cr_mixer.param + +import com.twitter.timelines.configapi.BaseConfig +import com.twitter.timelines.configapi.BaseConfigBuilder +import com.twitter.timelines.configapi.FSBoundedParam +import com.twitter.timelines.configapi.FSName +import com.twitter.timelines.configapi.FSParam +import com.twitter.timelines.configapi.FeatureSwitchOverrideUtil +import com.twitter.timelines.configapi.Param + +object TweetBasedUserVideoGraphParams { + + object MinCoOccurrenceParam + extends FSBoundedParam[Int]( + name = "tweet_based_user_video_graph_min_co_occurrence", + default = 5, + min = 0, + max = 500 + ) + + object TweetBasedMinScoreParam + extends FSBoundedParam[Double]( + name = "tweet_based_user_video_graph_tweet_based_min_score", + default = 0.0, + min = 0.0, + max = 100.0 + ) + + object ConsumersBasedMinScoreParam + extends FSBoundedParam[Double]( + name = "tweet_based_user_video_graph_consumers_based_min_score", + default = 4.0, + min = 0.0, + max = 10.0 + ) + + object MaxConsumerSeedsNumParam + extends FSBoundedParam[Int]( + name = "tweet_based_user_video_graph_max_user_seeds_num", + default = 200, + min = 0, + max = 500 + ) + + object EnableCoverageExpansionOldTweetParam + extends FSParam[Boolean]( + name = "tweet_based_user_video_graph_enable_coverage_expansion_old_tweet", + default = false + ) + + object EnableCoverageExpansionAllTweetParam + extends FSParam[Boolean]( + name = "tweet_based_user_video_graph_enable_coverage_expansion_all_tweet", + default = false + ) + + val AllParams: Seq[Param[_] with FSName] = Seq( + MinCoOccurrenceParam, + MaxConsumerSeedsNumParam, + TweetBasedMinScoreParam, + EnableCoverageExpansionOldTweetParam, + EnableCoverageExpansionAllTweetParam + ) + + lazy val config: BaseConfig = { + + val intOverrides = FeatureSwitchOverrideUtil.getBoundedIntFSOverrides( + MinCoOccurrenceParam, + MaxConsumerSeedsNumParam + ) + + val doubleOverrides = + FeatureSwitchOverrideUtil.getBoundedDoubleFSOverrides(TweetBasedMinScoreParam) + + BaseConfigBuilder() + .set(intOverrides: _*) + .set(doubleOverrides: _*) + .build() + } + +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/TweetSharesParams.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/TweetSharesParams.scala new file mode 100644 index 0000000000..1602441b06 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/TweetSharesParams.scala @@ -0,0 +1,29 @@ +package com.twitter.cr_mixer.param + +import com.twitter.timelines.configapi.BaseConfig +import com.twitter.timelines.configapi.BaseConfigBuilder +import com.twitter.timelines.configapi.FSName +import com.twitter.timelines.configapi.FSParam +import com.twitter.timelines.configapi.FeatureSwitchOverrideUtil +import com.twitter.timelines.configapi.Param + +object TweetSharesParams { + object EnableSourceParam + extends FSParam[Boolean]( + name = "twistly_tweetshares_enable_source", + default = false + ) + + val AllParams: Seq[Param[_] with FSName] = Seq(EnableSourceParam) + + lazy val config: BaseConfig = { + val booleanOverrides = FeatureSwitchOverrideUtil.getBooleanFSOverrides( + EnableSourceParam, + ) + + BaseConfigBuilder() + .set(booleanOverrides: _*) + .build() + } + +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/UnifiedSETweetCombinationMethod.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/UnifiedSETweetCombinationMethod.scala new file mode 100644 index 0000000000..f5b92f1384 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/UnifiedSETweetCombinationMethod.scala @@ -0,0 +1,15 @@ +package com.twitter.cr_mixer.param + +import scala.language.implicitConversions + +object UnifiedSETweetCombinationMethod extends Enumeration { + + protected case class CombinationType(s: String) extends super.Val + + implicit def valueToCombinationType(x: Value): CombinationType = x.asInstanceOf[CombinationType] + + val Default: Value = CombinationType("") + val Interleave: Value = CombinationType("Interleave") + val Frontload: Value = CombinationType("Frontload") + val Backfill: Value = CombinationType("Backfill") +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/UnifiedUSSSignalParams.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/UnifiedUSSSignalParams.scala new file mode 100644 index 0000000000..071cabc0c9 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/UnifiedUSSSignalParams.scala @@ -0,0 +1,121 @@ +package com.twitter.cr_mixer.param +import com.twitter.finagle.stats.NullStatsReceiver +import com.twitter.logging.Logger +import com.twitter.timelines.configapi.BaseConfig +import com.twitter.timelines.configapi.BaseConfigBuilder +import com.twitter.timelines.configapi.FSBoundedParam +import com.twitter.timelines.configapi.FSEnumParam +import com.twitter.timelines.configapi.FSName +import com.twitter.timelines.configapi.FSParam +import com.twitter.timelines.configapi.FeatureSwitchOverrideUtil +import com.twitter.timelines.configapi.Param +import com.twitter.usersignalservice.thriftscala.SignalType +import scala.language.implicitConversions + +object UnifiedUSSSignalParams { + + object TweetAggregationTypeParam extends Enumeration { + protected case class SignalTypeValue(signalType: SignalType) extends super.Val + + implicit def valueToSignalTypeValue(x: Value): SignalTypeValue = + x.asInstanceOf[SignalTypeValue] + + val UniformAggregation = SignalTypeValue(SignalType.TweetBasedUnifiedUniformSignal) + val EngagementAggregation = SignalTypeValue( + SignalType.TweetBasedUnifiedEngagementWeightedSignal) + } + + object ProducerAggregationTypeParam extends Enumeration { + protected case class SignalTypeValue(signalType: SignalType) extends super.Val + + import scala.language.implicitConversions + + implicit def valueToSignalTypeValue(x: Value): SignalTypeValue = + x.asInstanceOf[SignalTypeValue] + + val UniformAggregation = SignalTypeValue(SignalType.ProducerBasedUnifiedUniformSignal) + val EngagementAggregation = SignalTypeValue( + SignalType.ProducerBasedUnifiedEngagementWeightedSignal) + + } + + object ReplaceIndividualUSSSourcesParam + extends FSParam[Boolean]( + name = "twistly_agg_replace_enable_source", + default = false + ) + + object EnableTweetAggSourceParam + extends FSParam[Boolean]( + name = "twistly_agg_tweet_agg_enable_source", + default = false + ) + + object TweetAggTypeParam + extends FSEnumParam[TweetAggregationTypeParam.type]( + name = "twistly_agg_tweet_agg_type_id", + default = TweetAggregationTypeParam.EngagementAggregation, + enum = TweetAggregationTypeParam + ) + + object UnifiedTweetSourceNumberParam + extends FSBoundedParam[Int]( + name = "twistly_agg_tweet_agg_source_number", + default = 0, + min = 0, + max = 100, + ) + + object EnableProducerAggSourceParam + extends FSParam[Boolean]( + name = "twistly_agg_producer_agg_enable_source", + default = false + ) + + object ProducerAggTypeParam + extends FSEnumParam[ProducerAggregationTypeParam.type]( + name = "twistly_agg_producer_agg_type_id", + default = ProducerAggregationTypeParam.EngagementAggregation, + enum = ProducerAggregationTypeParam + ) + + object UnifiedProducerSourceNumberParam + extends FSBoundedParam[Int]( + name = "twistly_agg_producer_agg_source_number", + default = 0, + min = 0, + max = 100, + ) + + val AllParams: Seq[Param[_] with FSName] = Seq( + EnableTweetAggSourceParam, + EnableProducerAggSourceParam, + TweetAggTypeParam, + ProducerAggTypeParam, + UnifiedTweetSourceNumberParam, + UnifiedProducerSourceNumberParam, + ReplaceIndividualUSSSourcesParam + ) + lazy val config: BaseConfig = { + val booleanOverrides = FeatureSwitchOverrideUtil.getBooleanFSOverrides( + EnableTweetAggSourceParam, + EnableProducerAggSourceParam, + ReplaceIndividualUSSSourcesParam, + ) + val intOverrides = FeatureSwitchOverrideUtil.getBoundedIntFSOverrides( + UnifiedProducerSourceNumberParam, + UnifiedTweetSourceNumberParam) + val enumOverrides = FeatureSwitchOverrideUtil.getEnumFSOverrides( + NullStatsReceiver, + Logger(getClass), + TweetAggTypeParam, + ProducerAggTypeParam + ) + + BaseConfigBuilder() + .set(booleanOverrides: _*) + .set(intOverrides: _*) + .set(enumOverrides: _*) + .build() + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/UtegTweetGlobalParams.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/UtegTweetGlobalParams.scala new file mode 100644 index 0000000000..29f5a78182 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/UtegTweetGlobalParams.scala @@ -0,0 +1,94 @@ +package com.twitter.cr_mixer.param + +import com.twitter.conversions.DurationOps._ +import com.twitter.timelines.configapi.BaseConfig +import com.twitter.timelines.configapi.BaseConfigBuilder +import com.twitter.timelines.configapi.DurationConversion +import com.twitter.timelines.configapi.FSBoundedParam +import com.twitter.timelines.configapi.FSName +import com.twitter.timelines.configapi.FSParam +import com.twitter.timelines.configapi.FeatureSwitchOverrideUtil +import com.twitter.timelines.configapi.HasDurationConversion +import com.twitter.timelines.configapi.Param +import com.twitter.util.Duration + +object UtegTweetGlobalParams { + + object MaxUtegCandidatesToRequestParam + extends FSBoundedParam[Int]( + name = "max_uteg_candidates_to_request", + default = 800, + min = 10, + max = 200 + ) + + object CandidateRefreshSinceTimeOffsetHoursParam + extends FSBoundedParam[Duration]( + name = "candidate_refresh_since_time_offset_hours", + default = 48.hours, + min = 1.hours, + max = 96.hours + ) + with HasDurationConversion { + override val durationConversion: DurationConversion = DurationConversion.FromHours + } + + object EnableTLRHealthFilterParam + extends FSParam[Boolean]( + name = "enable_uteg_tlr_health_filter", + default = true + ) + + object EnableRepliesToNonFollowedUsersFilterParam + extends FSParam[Boolean]( + name = "enable_uteg_replies_to_non_followed_users_filter", + default = false + ) + + object EnableRetweetFilterParam + extends FSParam[Boolean]( + name = "enable_uteg_retweet_filter", + default = true + ) + + object EnableInNetworkFilterParam + extends FSParam[Boolean]( + name = "enable_uteg_in_network_filter", + default = true + ) + + val AllParams: Seq[Param[_] with FSName] = + Seq( + MaxUtegCandidatesToRequestParam, + CandidateRefreshSinceTimeOffsetHoursParam, + EnableTLRHealthFilterParam, + EnableRepliesToNonFollowedUsersFilterParam, + EnableRetweetFilterParam, + EnableInNetworkFilterParam + ) + + lazy val config: BaseConfig = { + + val intOverrides = FeatureSwitchOverrideUtil.getBoundedIntFSOverrides( + MaxUtegCandidatesToRequestParam + ) + + val durationFSOverrides = + FeatureSwitchOverrideUtil.getDurationFSOverrides( + CandidateRefreshSinceTimeOffsetHoursParam + ) + + val booleanOverrides = FeatureSwitchOverrideUtil.getBooleanFSOverrides( + EnableTLRHealthFilterParam, + EnableRepliesToNonFollowedUsersFilterParam, + EnableRetweetFilterParam, + EnableInNetworkFilterParam + ) + + BaseConfigBuilder() + .set(intOverrides: _*) + .set(durationFSOverrides: _*) + .set(booleanOverrides: _*) + .build() + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/VideoTweetFilterParams.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/VideoTweetFilterParams.scala new file mode 100644 index 0000000000..3a93d0a1a6 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/VideoTweetFilterParams.scala @@ -0,0 +1,31 @@ +package com.twitter.cr_mixer.param + +import com.twitter.timelines.configapi.BaseConfig +import com.twitter.timelines.configapi.BaseConfigBuilder +import com.twitter.timelines.configapi.FeatureSwitchOverrideUtil +import com.twitter.timelines.configapi.FSName +import com.twitter.timelines.configapi.FSParam +import com.twitter.timelines.configapi.Param + +object VideoTweetFilterParams { + + object EnableVideoTweetFilterParam + extends FSParam[Boolean]( + name = "video_tweet_filter_enable_filter", + default = false + ) + + val AllParams: Seq[Param[_] with FSName] = Seq( + EnableVideoTweetFilterParam + ) + + lazy val config: BaseConfig = { + + val booleanOverrides = + FeatureSwitchOverrideUtil.getBooleanFSOverrides(EnableVideoTweetFilterParam) + + BaseConfigBuilder() + .set(booleanOverrides: _*) + .build() + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/VideoViewTweetsParams.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/VideoViewTweetsParams.scala new file mode 100644 index 0000000000..44f508d89d --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/VideoViewTweetsParams.scala @@ -0,0 +1,64 @@ +package com.twitter.cr_mixer.param + +import com.twitter.finagle.stats.NullStatsReceiver +import com.twitter.logging.Logger +import com.twitter.timelines.configapi.BaseConfig +import com.twitter.timelines.configapi.BaseConfigBuilder +import com.twitter.timelines.configapi.FSEnumParam +import com.twitter.timelines.configapi.FSName +import com.twitter.timelines.configapi.FSParam +import com.twitter.timelines.configapi.FeatureSwitchOverrideUtil +import com.twitter.timelines.configapi.Param +import com.twitter.usersignalservice.thriftscala.SignalType + +object VideoViewTweetsParams { + object EnableSourceParam + extends FSParam[Boolean]( + name = "signal_videoviewtweets_enable_source", + default = false + ) + + object EnableSourceImpressionParam + extends FSParam[Boolean]( + name = "signal_videoviewtweets_enableimpression_source", + default = false + ) + + object VideoViewTweetType extends Enumeration { + protected case class SignalTypeValue(signalType: SignalType) extends super.Val + import scala.language.implicitConversions + implicit def valueToSignalTypeValue(x: Value): SignalTypeValue = + x.asInstanceOf[SignalTypeValue] + + val VideoTweetQualityView: SignalTypeValue = SignalTypeValue(SignalType.VideoView90dQualityV1) + val VideoTweetPlayback50: SignalTypeValue = SignalTypeValue(SignalType.VideoView90dPlayback50V1) + } + + object VideoViewTweetTypeParam + extends FSEnumParam[VideoViewTweetType.type]( + name = "signal_videoviewtweets_videoviewtype_id", + default = VideoViewTweetType.VideoTweetQualityView, + enum = VideoViewTweetType + ) + + val AllParams: Seq[Param[_] with FSName] = + Seq(EnableSourceParam, EnableSourceImpressionParam, VideoViewTweetTypeParam) + + lazy val config: BaseConfig = { + val booleanOverrides = FeatureSwitchOverrideUtil.getBooleanFSOverrides( + EnableSourceParam, + EnableSourceImpressionParam, + ) + val enumOverrides = + FeatureSwitchOverrideUtil.getEnumFSOverrides( + NullStatsReceiver, + Logger(getClass), + VideoViewTweetTypeParam) + + BaseConfigBuilder() + .set(booleanOverrides: _*) + .set(enumOverrides: _*) + .build() + } + +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/decider/BUILD b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/decider/BUILD new file mode 100644 index 0000000000..730986d64a --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/decider/BUILD @@ -0,0 +1,16 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + strict_deps = True, + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/javax/inject:javax.inject", + "decider/src/main/scala", + "finagle/finagle-base-http/src/main", + "finagle/finagle-core/src/main", + "finagle/finagle-http/src/main/scala", + "servo/decider", + "src/scala/com/twitter/simclusters_v2/common", + ], +) diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/decider/CrMixerDecider.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/decider/CrMixerDecider.scala new file mode 100644 index 0000000000..8c909ca05d --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/decider/CrMixerDecider.scala @@ -0,0 +1,39 @@ +package com.twitter.cr_mixer.param.decider + +import com.twitter.decider.Decider +import com.twitter.decider.RandomRecipient +import com.twitter.decider.Recipient +import com.twitter.decider.SimpleRecipient +import com.twitter.simclusters_v2.common.DeciderGateBuilderWithIdHashing +import javax.inject.Inject + +case class CrMixerDecider @Inject() (decider: Decider) { + + def isAvailable(feature: String, recipient: Option[Recipient]): Boolean = { + decider.isAvailable(feature, recipient) + } + + lazy val deciderGateBuilder = new DeciderGateBuilderWithIdHashing(decider) + + /** + * When useRandomRecipient is set to false, the decider is either completely on or off. + * When useRandomRecipient is set to true, the decider is on for the specified % of traffic. + */ + def isAvailable(feature: String, useRandomRecipient: Boolean = true): Boolean = { + if (useRandomRecipient) isAvailable(feature, Some(RandomRecipient)) + else isAvailable(feature, None) + } + + /*** + * Decide whether the decider is available for a specific id using SimpleRecipient(id). + */ + def isAvailableForId( + id: Long, + deciderConstants: String + ): Boolean = { + // Note: SimpleRecipient does expose a `val isUser = true` field which is not correct if the Id is not a user Id. + // However this field does not appear to be used anywhere in source. + decider.isAvailable(deciderConstants, Some(SimpleRecipient(id))) + } + +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/decider/DeciderKey.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/decider/DeciderKey.scala new file mode 100644 index 0000000000..518ea53db3 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/decider/DeciderKey.scala @@ -0,0 +1,67 @@ +package com.twitter.cr_mixer.param.decider + +import com.twitter.servo.decider.DeciderKeyEnum + +object DeciderConstants { + val enableHealthSignalsScoreDeciderKey = "enable_tweet_health_score" + val enableUTGRealTimeTweetEngagementScoreDeciderKey = "enable_utg_realtime_tweet_engagement_score" + val enableUserAgathaScoreDeciderKey = "enable_user_agatha_score" + val enableUserTweetEntityGraphTrafficDeciderKey = "enable_user_tweet_entity_graph_traffic" + val enableUserTweetGraphTrafficDeciderKey = "enable_user_tweet_graph_traffic" + val enableUserVideoGraphTrafficDeciderKey = "enable_user_video_graph_traffic" + val enableUserAdGraphTrafficDeciderKey = "enable_user_ad_graph_traffic" + val enableSimClustersANN2DarkTrafficDeciderKey = "enable_simclusters_ann_2_dark_traffic" + val enableQigSimilarTweetsTrafficDeciderKey = "enable_qig_similar_tweets_traffic" + val enableFRSTrafficDeciderKey = "enable_frs_traffic" + val upperFunnelPerStepScribeRate = "upper_funnel_per_step_scribe_rate" + val kafkaMessageScribeSampleRate = "kafka_message_scribe_sample_rate" + val enableRealGraphMhStoreDeciderKey = "enable_real_graph_mh_store" + val topLevelApiDdgMetricsScribeRate = "top_level_api_ddg_metrics_scribe_rate" + val adsRecommendationsPerExperimentScribeRate = "ads_recommendations_per_experiment_scribe_rate" + val enableScribeForBlueVerifiedTweetCandidates = + "enable_scribe_for_blue_verified_tweet_candidates" + + val enableUserStateStoreDeciderKey = "enable_user_state_store" + val enableUserMediaRepresentationStoreDeciderKey = + "enable_user_media_representation_store" + val enableMagicRecsRealTimeAggregatesStoreDeciderKey = + "enable_magic_recs_real_time_aggregates_store" + + val enableEarlybirdTrafficDeciderKey = "enable_earlybird_traffic" + + val enableTopicTweetTrafficDeciderKey = "enable_topic_tweet_traffic" + + val getTweetRecommendationsCacheRate = "get_tweet_recommendations_cache_rate" +} + +object DeciderKey extends DeciderKeyEnum { + + val enableHealthSignalsScoreDeciderKey: Value = Value( + DeciderConstants.enableHealthSignalsScoreDeciderKey + ) + + val enableUtgRealTimeTweetEngagementScoreDeciderKey: Value = Value( + DeciderConstants.enableUTGRealTimeTweetEngagementScoreDeciderKey + ) + val enableUserAgathaScoreDeciderKey: Value = Value( + DeciderConstants.enableUserAgathaScoreDeciderKey + ) + val enableUserMediaRepresentationStoreDeciderKey: Value = Value( + DeciderConstants.enableUserMediaRepresentationStoreDeciderKey + ) + + val enableMagicRecsRealTimeAggregatesStore: Value = Value( + DeciderConstants.enableMagicRecsRealTimeAggregatesStoreDeciderKey + ) + + val enableUserStateStoreDeciderKey: Value = Value( + DeciderConstants.enableUserStateStoreDeciderKey + ) + + val enableRealGraphMhStoreDeciderKey: Value = Value( + DeciderConstants.enableRealGraphMhStoreDeciderKey + ) + + val enableEarlybirdTrafficDeciderKey: Value = Value( + DeciderConstants.enableEarlybirdTrafficDeciderKey) +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/decider/EndpointLoadShedder.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/decider/EndpointLoadShedder.scala new file mode 100644 index 0000000000..a53e629a99 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/decider/EndpointLoadShedder.scala @@ -0,0 +1,57 @@ +package com.twitter.cr_mixer.param.decider + +import com.twitter.decider.Decider +import com.twitter.decider.RandomRecipient +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.util.Future +import javax.inject.Inject +import scala.util.control.NoStackTrace + +/* + Provides deciders-controlled load shedding for a given Product from a given endpoint. + The format of the decider keys is: + + enable_loadshedding__ + E.g.: + enable_loadshedding_getTweetRecommendations_Notifications + + Deciders are fractional, so a value of 50.00 will drop 50% of responses. If a decider key is not + defined for a particular endpoint/product combination, those requests will always be + served. + + We should therefore aim to define keys for the endpoints/product we care most about in decider.yml, + so that we can control them during incidents. + */ +case class EndpointLoadShedder @Inject() ( + decider: Decider, + statsReceiver: StatsReceiver) { + import EndpointLoadShedder._ + + // Fall back to False for any undefined key + private val deciderWithFalseFallback: Decider = decider.orElse(Decider.False) + private val keyPrefix = "enable_loadshedding" + private val scopedStats = statsReceiver.scope("EndpointLoadShedder") + + def apply[T](endpointName: String, product: String)(serve: => Future[T]): Future[T] = { + /* + Checks if either per-product or top-level load shedding is enabled + If both are enabled at different percentages, load shedding will not be perfectly calculable due + to salting of hash (i.e. 25% load shed for Product x + 25% load shed for overall does not + result in 50% load shed for x) + */ + val keyTyped = s"${keyPrefix}_${endpointName}_$product" + val keyTopLevel = s"${keyPrefix}_${endpointName}" + + if (deciderWithFalseFallback.isAvailable(keyTopLevel, recipient = Some(RandomRecipient))) { + scopedStats.counter(keyTopLevel).incr + Future.exception(LoadSheddingException) + } else if (deciderWithFalseFallback.isAvailable(keyTyped, recipient = Some(RandomRecipient))) { + scopedStats.counter(keyTyped).incr + Future.exception(LoadSheddingException) + } else serve + } +} + +object EndpointLoadShedder { + object LoadSheddingException extends Exception with NoStackTrace +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/ranker/BUILD b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/ranker/BUILD new file mode 100644 index 0000000000..139ecd4c76 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/ranker/BUILD @@ -0,0 +1,30 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + strict_deps = True, + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/com/twitter/storehaus:core", + "3rdparty/jvm/javax/inject:javax.inject", + "configapi/configapi-core", + "content-recommender/thrift/src/main/thrift:content-recommender-common-scala", + "cr-mixer/server/src/main/scala/com/twitter/cr_mixer/config", + "cr-mixer/server/src/main/scala/com/twitter/cr_mixer/model", + "cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param", + "cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/decider", + "cr-mixer/server/src/main/scala/com/twitter/cr_mixer/util", + "cr-mixer/thrift/src/main/thrift:thrift-scala", + "decider/src/main/scala", + "frigate/frigate-common:base", + "frigate/frigate-common/src/main/scala/com/twitter/frigate/common/base", + "frigate/frigate-common/src/main/scala/com/twitter/frigate/common/util:stats_util", + "hydra/common/libraries/src/main/scala/com/twitter/hydra/common/model_config", + "hydra/partition/thrift/src/main/thrift:thrift-scala", + "hydra/root/thrift/src/main/thrift:thrift-scala", + "product-mixer/core/src/main/thrift/com/twitter/product_mixer/core:thrift-scala", + "src/scala/com/twitter/simclusters_v2/common", + "src/thrift/com/twitter/core_workflows/user_model:user_model-scala", + "src/thrift/com/twitter/ml/api:data-scala", + "src/thrift/com/twitter/simclusters_v2:simclusters_v2-thrift-scala", + ], +) diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/ranker/DefaultRanker.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/ranker/DefaultRanker.scala new file mode 100644 index 0000000000..2ae91642b7 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/ranker/DefaultRanker.scala @@ -0,0 +1,23 @@ +package com.twitter.cr_mixer.ranker + +import com.twitter.cr_mixer.model.BlendedCandidate +import com.twitter.cr_mixer.model.RankedCandidate +import com.twitter.util.Future +import javax.inject.Singleton + +/** + * Keep the same order as the input. + */ +@Singleton +class DefaultRanker() { + def rank( + candidates: Seq[BlendedCandidate], + ): Future[Seq[RankedCandidate]] = { + val candidateSize = candidates.size + val rankedCandidates = candidates.zipWithIndex.map { + case (candidate, index) => + candidate.toRankedCandidate((candidateSize - index).toDouble) + } + Future.value(rankedCandidates) + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/ranker/SwitchRanker.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/ranker/SwitchRanker.scala new file mode 100644 index 0000000000..da44f664ea --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/ranker/SwitchRanker.scala @@ -0,0 +1,46 @@ +package com.twitter.cr_mixer.ranker + +import com.twitter.cr_mixer.model.BlendedCandidate +import com.twitter.cr_mixer.model.CrCandidateGeneratorQuery +import com.twitter.cr_mixer.model.RankedCandidate +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.util.Future +import com.twitter.util.JavaTimer +import com.twitter.util.Time +import com.twitter.util.Timer +import javax.inject.Inject +import javax.inject.Singleton + +/** + * CR-Mixer internal ranker + */ +@Singleton +class SwitchRanker @Inject() ( + defaultRanker: DefaultRanker, + globalStats: StatsReceiver) { + private val stats: StatsReceiver = globalStats.scope(this.getClass.getCanonicalName) + implicit val timer: Timer = new JavaTimer(true) + + def rank( + query: CrCandidateGeneratorQuery, + candidates: Seq[BlendedCandidate], + ): Future[Seq[RankedCandidate]] = { + defaultRanker.rank(candidates) + } + +} + +object SwitchRanker { + + /** Prefers candidates generated from sources with the latest timestamps. + * The newer the source signal, the higher a candidate ranks. + * This ordering biases against consumer-based candidates because their timestamp defaults to 0 + */ + val TimestampOrder: Ordering[RankedCandidate] = + math.Ordering + .by[RankedCandidate, Time]( + _.reasonChosen.sourceInfoOpt + .flatMap(_.sourceEventTime) + .getOrElse(Time.fromMilliseconds(0L))) + .reverse +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/scribe/BUILD b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/scribe/BUILD new file mode 100644 index 0000000000..8e6ae50494 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/scribe/BUILD @@ -0,0 +1,22 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + strict_deps = True, + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/javax/inject:javax.inject", + "configapi/configapi-core", + "cr-mixer/thrift/src/main/thrift:thrift-scala", + "decider/src/main/scala", + "finagle-internal/mtls/src/main/scala/com/twitter/finagle/mtls/authentication", + "finagle/finagle-core/src/main", + "frigate/frigate-common:base", + "frigate/frigate-common:util", + "frigate/frigate-common/src/main/scala/com/twitter/frigate/common/base", + "product-mixer/core/src/main/thrift/com/twitter/product_mixer/core:thrift-scala", + "scrooge/scrooge-serializer/src/main/scala", + "src/scala/com/twitter/simclusters_v2/common", + "src/thrift/com/twitter/simclusters_v2:simclusters_v2-thrift-scala", + "util-internal/scribe/src/main/scala/com/twitter/logging", + ], +) diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/scribe/ScribeCategory.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/scribe/ScribeCategory.scala new file mode 100644 index 0000000000..b86c9174f2 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/scribe/ScribeCategory.scala @@ -0,0 +1,64 @@ +package com.twitter.cr_mixer.scribe + +/** + * Categories define scribe categories used in cr-mixer service. + */ +object ScribeCategories { + lazy val AllCategories = + List(AbDecider, TopLevelApiDdgMetrics, TweetsRecs) + + /** + * AbDecider represents scribe logs for experiments + */ + lazy val AbDecider: ScribeCategory = ScribeCategory( + "abdecider_scribe", + "client_event" + ) + + /** + * Top-Level Client event scribe logs, to record changes in system metrics (e.g. latency, + * candidates returned, empty rate ) per experiment bucket, and store them in DDG metric group + */ + lazy val TopLevelApiDdgMetrics: ScribeCategory = ScribeCategory( + "top_level_api_ddg_metrics_scribe", + "client_event" + ) + + lazy val TweetsRecs: ScribeCategory = ScribeCategory( + "get_tweets_recommendations_scribe", + "cr_mixer_get_tweets_recommendations" + ) + + lazy val VITTweetsRecs: ScribeCategory = ScribeCategory( + "get_vit_tweets_recommendations_scribe", + "cr_mixer_get_vit_tweets_recommendations" + ) + + lazy val RelatedTweets: ScribeCategory = ScribeCategory( + "get_related_tweets_scribe", + "cr_mixer_get_related_tweets" + ) + + lazy val UtegTweets: ScribeCategory = ScribeCategory( + "get_uteg_tweets_scribe", + "cr_mixer_get_uteg_tweets" + ) + + lazy val AdsRecommendations: ScribeCategory = ScribeCategory( + "get_ads_recommendations_scribe", + "cr_mixer_get_ads_recommendations" + ) +} + +/** + * Category represents each scribe log data. + * + * @param loggerFactoryNode loggerFactory node name in cr-mixer associated with this scribe category + * @param scribeCategory scribe category name (globally unique at Twitter) + */ +case class ScribeCategory( + loggerFactoryNode: String, + scribeCategory: String) { + def getProdLoggerFactoryNode: String = loggerFactoryNode + def getStagingLoggerFactoryNode: String = "staging_" + loggerFactoryNode +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/service/BUILD.bazel b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/service/BUILD.bazel new file mode 100644 index 0000000000..8fa46c7720 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/service/BUILD.bazel @@ -0,0 +1,15 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + strict_deps = True, + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/javax/inject:javax.inject", + "configapi/configapi-core", + "cr-mixer/thrift/src/main/thrift:thrift-scala", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/product", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/product", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/product/registry", + "stitch/stitch-core", + ], +) diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/service/CrMixerAlertNotificationConfig.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/service/CrMixerAlertNotificationConfig.scala new file mode 100644 index 0000000000..df0572ef1a --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/service/CrMixerAlertNotificationConfig.scala @@ -0,0 +1,26 @@ +package com.twitter.cr_mixer.service + +import com.twitter.product_mixer.core.functional_component.common.alert.Destination +import com.twitter.product_mixer.core.functional_component.common.alert.NotificationGroup + +/** + * Notifications (email, pagerduty, etc) can be specific per-alert but it is common for multiple + * products to share notification configuration. + * + * Our configuration uses only email notifications because SampleMixer is a demonstration service + * with neither internal nor customer-facing users. You will likely want to use a PagerDuty + * destination instead. For example: + * {{{ + * critical = Destination(pagerDutyKey = Some("your-pagerduty-key")) + * }}} + * + * + * For more information about how to get a PagerDuty key, see: + * https://docbird.twitter.biz/mon/how-to-guides.html?highlight=notificationgroup#set-up-email-pagerduty-and-slack-notifications + */ +object CrMixerAlertNotificationConfig { + val DefaultNotificationGroup: NotificationGroup = NotificationGroup( + warn = Destination(emails = Seq("no-reply@twitter.com")), + critical = Destination(emails = Seq("no-reply@twitter.com")) + ) +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/BUILD b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/BUILD new file mode 100644 index 0000000000..c7ae7c7522 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/BUILD @@ -0,0 +1,74 @@ +scala_library( + sources = [ + "*.scala", + ], + compiler_option_sets = ["fatal_warnings"], + strict_deps = True, + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/com/twitter/storehaus:core", + "3rdparty/jvm/com/twitter/storehaus:memcache", + "3rdparty/jvm/io/grpc:grpc-api", + "3rdparty/jvm/io/grpc:grpc-auth", + "3rdparty/jvm/io/grpc:grpc-core", + "3rdparty/jvm/io/grpc:grpc-netty", + "3rdparty/jvm/io/grpc:grpc-protobuf", + "3rdparty/jvm/io/grpc:grpc-stub", + "3rdparty/jvm/io/opil:tensorflow-serving-client", + "3rdparty/jvm/javax/inject:javax.inject", + "3rdparty/src/jvm/com/twitter/storehaus:core", + "ann/src/main/scala/com/twitter/ann/hnsw", + "ann/src/main/thrift/com/twitter/ann/common:ann-common-scala", + "configapi/configapi-core", + "content-recommender/thrift/src/main/thrift:thrift-scala", + "cr-mixer/server/src/main/scala/com/twitter/cr_mixer/config", + "cr-mixer/server/src/main/scala/com/twitter/cr_mixer/exception", + "cr-mixer/server/src/main/scala/com/twitter/cr_mixer/model", + "cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param", + "cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/decider", + "cr-mixer/server/src/main/scala/com/twitter/cr_mixer/source_signal", + "cr-mixer/server/src/main/scala/com/twitter/cr_mixer/util", + "cr-mixer/thrift/src/main/thrift:thrift-scala", + "decider/src/main/scala", + "finagle-internal/finagle-grpc/src/main/scala", + "finagle-internal/mtls/src/main/scala/com/twitter/finagle/mtls/client", + "finatra-internal/mtls-thriftmux/src/main/scala", + "finatra/inject/inject-core/src/main/scala", + "follow-recommendations-service/thrift/src/main/thrift:thrift-scala", + "frigate/frigate-common:base", + "frigate/frigate-common/src/main/scala/com/twitter/frigate/common/base", + "frigate/frigate-common/src/main/scala/com/twitter/frigate/common/candidate", + "frigate/frigate-common/src/main/scala/com/twitter/frigate/common/store/strato", + "frigate/frigate-common/src/main/scala/com/twitter/frigate/common/util", + "frigate/frigate-common/src/main/scala/com/twitter/frigate/common/util:stats_util", + "hermit/hermit-core/src/main/scala/com/twitter/hermit/store/common", + "mediaservices/commons/src/main/scala:futuretracker", + "product-mixer/core/src/main/thrift/com/twitter/product_mixer/core:thrift-scala", + "qig-ranker/thrift/src/main/thrift:thrift-scala", + "relevance-platform/src/main/scala/com/twitter/relevance_platform/common/injection", + "relevance-platform/src/main/scala/com/twitter/relevance_platform/simclustersann/multicluster", + "simclusters-ann/thrift/src/main/thrift:thrift-scala", + "snowflake/src/main/scala/com/twitter/snowflake/id", + "src/java/com/twitter/search/common/schema/base", + "src/java/com/twitter/search/common/schema/earlybird", + "src/java/com/twitter/search/queryparser/query:core-query-nodes", + "src/java/com/twitter/search/queryparser/query/search:search-query-nodes", + "src/scala/com/twitter/cortex/ml/embeddings/common:Helpers", + "src/scala/com/twitter/ml/featurestore/lib", + "src/scala/com/twitter/simclusters_v2/common", + "src/thrift/com/twitter/ml/api:embedding-scala", + "src/thrift/com/twitter/recos:recos-common-scala", + "src/thrift/com/twitter/recos/user_ad_graph:user_ad_graph-scala", + "src/thrift/com/twitter/recos/user_tweet_entity_graph:user_tweet_entity_graph-scala", + "src/thrift/com/twitter/recos/user_tweet_graph:user_tweet_graph-scala", + "src/thrift/com/twitter/recos/user_video_graph:user_video_graph-scala", + "src/thrift/com/twitter/search:earlybird-scala", + "src/thrift/com/twitter/search/common:ranking-scala", + "src/thrift/com/twitter/search/query_interaction_graph/service:qig-service-scala", + "src/thrift/com/twitter/simclusters_v2:simclusters_v2-thrift-scala", + "src/thrift/com/twitter/topic_recos:topic_recos-thrift-scala", + "src/thrift/com/twitter/trends/trip_v1:trip-tweets-thrift-scala", + "src/thrift/com/twitter/twistly:twistly-scala", + "strato/src/main/scala/com/twitter/strato/client", + ], +) diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/CertoTopicTweetSimilarityEngine.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/CertoTopicTweetSimilarityEngine.scala new file mode 100644 index 0000000000..a57085d0fd --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/CertoTopicTweetSimilarityEngine.scala @@ -0,0 +1,94 @@ +package com.twitter.cr_mixer.similarity_engine + +import com.google.inject.Inject +import com.google.inject.Singleton +import com.google.inject.name.Named +import com.twitter.cr_mixer.model.ModuleNames +import com.twitter.cr_mixer.model.TopicTweetWithScore +import com.twitter.cr_mixer.param.TopicTweetParams +import com.twitter.cr_mixer.similarity_engine.CertoTopicTweetSimilarityEngine._ +import com.twitter.cr_mixer.thriftscala.SimilarityEngineType +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.frigate.common.util.StatsUtil +import com.twitter.simclusters_v2.thriftscala.TopicId +import com.twitter.storehaus.ReadableStore +import com.twitter.timelines.configapi +import com.twitter.topic_recos.thriftscala._ +import com.twitter.util.Future + +@Singleton +case class CertoTopicTweetSimilarityEngine @Inject() ( + @Named(ModuleNames.CertoStratoStoreName) certoStratoStore: ReadableStore[ + TopicId, + Seq[TweetWithScores] + ], + statsReceiver: StatsReceiver) + extends ReadableStore[EngineQuery[Query], Seq[TopicTweetWithScore]] { + + private val name: String = this.getClass.getSimpleName + private val stats = statsReceiver.scope(name) + + override def get(query: EngineQuery[Query]): Future[Option[Seq[TopicTweetWithScore]]] = { + StatsUtil.trackOptionItemsStats(stats) { + topTweetsByFollowerL2NormalizedScore.get(query).map { + _.map { topicTopTweets => + topicTopTweets.map { topicTweet => + TopicTweetWithScore( + tweetId = topicTweet.tweetId, + score = topicTweet.scores.followerL2NormalizedCosineSimilarity8HrHalfLife, + similarityEngineType = SimilarityEngineType.CertoTopicTweet + ) + } + } + } + } + } + + private val topTweetsByFollowerL2NormalizedScore: ReadableStore[EngineQuery[Query], Seq[ + TweetWithScores + ]] = { + ReadableStore.fromFnFuture { query: EngineQuery[Query] => + StatsUtil.trackOptionItemsStats(stats) { + for { + topKTweetsWithScores <- certoStratoStore.get(query.storeQuery.topicId) + } yield { + topKTweetsWithScores.map( + _.filter( + _.scores.followerL2NormalizedCosineSimilarity8HrHalfLife >= query.storeQuery.certoScoreTheshold) + .take(query.storeQuery.maxCandidates)) + } + } + } + } +} + +object CertoTopicTweetSimilarityEngine { + + // Query is used as a cache key. Do not add any user level information in this. + case class Query( + topicId: TopicId, + maxCandidates: Int, + certoScoreTheshold: Double) + + def fromParams( + topicId: TopicId, + isVideoOnly: Boolean, + params: configapi.Params, + ): EngineQuery[Query] = { + + val maxCandidates = if (isVideoOnly) { + params(TopicTweetParams.MaxCertoCandidatesParam) * 2 + } else { + params(TopicTweetParams.MaxCertoCandidatesParam) + } + + EngineQuery( + Query( + topicId = topicId, + maxCandidates = maxCandidates, + certoScoreTheshold = params(TopicTweetParams.CertoScoreThresholdParam) + ), + params + ) + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/ConsumerBasedWalsSimilarityEngine.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/ConsumerBasedWalsSimilarityEngine.scala new file mode 100644 index 0000000000..599704fa79 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/ConsumerBasedWalsSimilarityEngine.scala @@ -0,0 +1,246 @@ +package com.twitter.cr_mixer.similarity_engine + +import com.twitter.cr_mixer.model.SimilarityEngineInfo +import com.twitter.cr_mixer.model.SourceInfo +import com.twitter.cr_mixer.model.TweetWithScore +import com.twitter.cr_mixer.param.ConsumerBasedWalsParams +import com.twitter.cr_mixer.similarity_engine.ConsumerBasedWalsSimilarityEngine.Query +import com.twitter.cr_mixer.thriftscala.SimilarityEngineType +import com.twitter.cr_mixer.thriftscala.SourceType +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.simclusters_v2.thriftscala.InternalId +import com.twitter.storehaus.ReadableStore +import com.twitter.timelines.configapi +import com.twitter.util.Future +import io.grpc.ManagedChannel +import tensorflow.serving.Predict.PredictRequest +import tensorflow.serving.Predict.PredictResponse +import tensorflow.serving.PredictionServiceGrpc +import org.tensorflow.example.Feature +import org.tensorflow.example.Int64List +import org.tensorflow.example.FloatList +import org.tensorflow.example.Features +import org.tensorflow.example.Example +import tensorflow.serving.Model +import org.tensorflow.framework.TensorProto +import org.tensorflow.framework.DataType +import org.tensorflow.framework.TensorShapeProto +import com.twitter.finagle.grpc.FutureConverters +import java.util.ArrayList +import java.lang +import com.twitter.util.Return +import com.twitter.util.Throw +import java.util.concurrent.ConcurrentHashMap +import scala.jdk.CollectionConverters._ + +// Stats object maintain a set of stats that are specific to the Wals Engine. +case class WalsStats(scope: String, scopedStats: StatsReceiver) { + + val requestStat = scopedStats.scope(scope) + val inputSignalSize = requestStat.stat("input_signal_size") + + val latency = requestStat.stat("latency_ms") + val latencyOnError = requestStat.stat("error_latency_ms") + val latencyOnSuccess = requestStat.stat("success_latency_ms") + + val requests = requestStat.counter("requests") + val success = requestStat.counter("success") + val failures = requestStat.scope("failures") + + def onFailure(t: Throwable, startTimeMs: Long) { + val duration = System.currentTimeMillis() - startTimeMs + latency.add(duration) + latencyOnError.add(duration) + failures.counter(t.getClass.getName).incr() + } + + def onSuccess(startTimeMs: Long) { + val duration = System.currentTimeMillis() - startTimeMs + latency.add(duration) + latencyOnSuccess.add(duration) + success.incr() + } +} + +// StatsMap maintains a mapping from Model's input signature to a stats receiver +// The Wals model suports multiple input signature which can run different graphs internally and +// can have a different performance profile. +// Invoking StatsReceiver.stat() on each request can create a new stat object and can be expensive +// in performance critical paths. +object WalsStatsMap { + val mapping = new ConcurrentHashMap[String, WalsStats]() + + def get(scope: String, scopedStats: StatsReceiver): WalsStats = { + mapping.computeIfAbsent(scope, (scope) => WalsStats(scope, scopedStats)) + } +} + +case class ConsumerBasedWalsSimilarityEngine( + homeNaviGRPCClient: ManagedChannel, + adsFavedNaviGRPCClient: ManagedChannel, + adsMonetizableNaviGRPCClient: ManagedChannel, + statsReceiver: StatsReceiver) + extends ReadableStore[ + Query, + Seq[TweetWithScore] + ] { + + override def get( + query: ConsumerBasedWalsSimilarityEngine.Query + ): Future[Option[Seq[TweetWithScore]]] = { + val startTimeMs = System.currentTimeMillis() + val stats = + WalsStatsMap.get( + query.wilyNsName + "/" + query.modelSignatureName, + statsReceiver.scope("NaviPredictionService") + ) + stats.requests.incr() + stats.inputSignalSize.add(query.sourceIds.size) + try { + // avoid inference calls is source signals are empty + if (query.sourceIds.isEmpty) { + Future.value(Some(Seq.empty)) + } else { + val grpcClient = query.wilyNsName match { + case "navi-wals-recommended-tweets-home-client" => homeNaviGRPCClient + case "navi-wals-ads-faved-tweets" => adsFavedNaviGRPCClient + case "navi-wals-ads-monetizable-tweets" => adsFavedNaviGRPCClient + // default to homeNaviGRPCClient + case _ => homeNaviGRPCClient + } + val stub = PredictionServiceGrpc.newFutureStub(grpcClient) + val inferRequest = getModelInput(query) + + FutureConverters + .RichListenableFuture(stub.predict(inferRequest)).toTwitter + .transform { + case Return(resp) => + stats.onSuccess(startTimeMs) + Future.value(Some(getModelOutput(query, resp))) + case Throw(e) => + stats.onFailure(e, startTimeMs) + Future.exception(e) + } + } + } catch { + case e: Throwable => Future.exception(e) + } + } + + def getFeaturesForRecommendations(query: ConsumerBasedWalsSimilarityEngine.Query): Example = { + val tweetIds = new ArrayList[lang.Long]() + val tweetFaveWeight = new ArrayList[lang.Float]() + + query.sourceIds.foreach { sourceInfo => + val weight = sourceInfo.sourceType match { + case SourceType.TweetFavorite | SourceType.Retweet => 1.0f + // currently no-op - as we do not get negative signals + case SourceType.TweetDontLike | SourceType.TweetReport | SourceType.AccountMute | + SourceType.AccountBlock => + 0.0f + case _ => 0.0f + } + sourceInfo.internalId match { + case InternalId.TweetId(tweetId) => + tweetIds.add(tweetId) + tweetFaveWeight.add(weight) + case _ => + throw new IllegalArgumentException( + s"Invalid InternalID - does not contain TweetId for Source Signal: ${sourceInfo}") + } + } + + val tweetIdsFeature = + Feature + .newBuilder().setInt64List( + Int64List + .newBuilder().addAllValue(tweetIds).build() + ).build() + + val tweetWeightsFeature = Feature + .newBuilder().setFloatList( + FloatList.newBuilder().addAllValue(tweetFaveWeight).build()).build() + + val features = Features + .newBuilder() + .putFeature("tweet_ids", tweetIdsFeature) + .putFeature("tweet_weights", tweetWeightsFeature) + .build() + Example.newBuilder().setFeatures(features).build() + } + + def getModelInput(query: ConsumerBasedWalsSimilarityEngine.Query): PredictRequest = { + val tfExample = getFeaturesForRecommendations(query) + + val inferenceRequest = PredictRequest + .newBuilder() + .setModelSpec( + Model.ModelSpec + .newBuilder() + .setName(query.modelName) + .setSignatureName(query.modelSignatureName)) + .putInputs( + query.modelInputName, + TensorProto + .newBuilder() + .setDtype(DataType.DT_STRING) + .setTensorShape(TensorShapeProto + .newBuilder() + .addDim(TensorShapeProto.Dim.newBuilder().setSize(1))) + .addStringVal(tfExample.toByteString) + .build() + ).build() + inferenceRequest + } + + def getModelOutput(query: Query, response: PredictResponse): Seq[TweetWithScore] = { + val outputName = query.modelOutputName + if (response.containsOutputs(outputName)) { + val tweetList = response.getOutputsMap + .get(outputName) + .getInt64ValList.asScala + tweetList.zip(tweetList.size to 1 by -1).map { (tweetWithScore) => + TweetWithScore(tweetWithScore._1, tweetWithScore._2.toLong) + } + } else { + Seq.empty + } + } +} + +object ConsumerBasedWalsSimilarityEngine { + case class Query( + sourceIds: Seq[SourceInfo], + modelName: String, + modelInputName: String, + modelOutputName: String, + modelSignatureName: String, + wilyNsName: String, + ) + + def fromParams( + sourceIds: Seq[SourceInfo], + params: configapi.Params, + ): EngineQuery[Query] = { + EngineQuery( + Query( + sourceIds, + params(ConsumerBasedWalsParams.ModelNameParam), + params(ConsumerBasedWalsParams.ModelInputNameParam), + params(ConsumerBasedWalsParams.ModelOutputNameParam), + params(ConsumerBasedWalsParams.ModelSignatureNameParam), + params(ConsumerBasedWalsParams.WilyNsNameParam), + ), + params + ) + } + + def toSimilarityEngineInfo( + score: Double + ): SimilarityEngineInfo = { + SimilarityEngineInfo( + similarityEngineType = SimilarityEngineType.ConsumerBasedWalsANN, + modelId = None, + score = Some(score)) + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/ConsumerEmbeddingBasedTripSimilarityEngine.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/ConsumerEmbeddingBasedTripSimilarityEngine.scala new file mode 100644 index 0000000000..82a0742087 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/ConsumerEmbeddingBasedTripSimilarityEngine.scala @@ -0,0 +1,118 @@ +package com.twitter.cr_mixer.similarity_engine + +import com.twitter.cr_mixer.model.TripTweetWithScore +import com.twitter.cr_mixer.param.ConsumerEmbeddingBasedTripParams +import com.twitter.cr_mixer.util.InterleaveUtil +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.frigate.common.util.StatsUtil +import com.twitter.simclusters_v2.common.ClusterId +import com.twitter.simclusters_v2.common.SimClustersEmbedding +import com.twitter.simclusters_v2.common.UserId +import com.twitter.simclusters_v2.thriftscala.InternalId +import com.twitter.storehaus.ReadableStore +import com.twitter.timelines.configapi +import com.twitter.timelines.configapi.Params +import com.twitter.trends.trip_v1.trip_tweets.thriftscala.Cluster +import com.twitter.trends.trip_v1.trip_tweets.thriftscala.ClusterDomain +import com.twitter.trends.trip_v1.trip_tweets.thriftscala.TripTweet +import com.twitter.trends.trip_v1.trip_tweets.thriftscala.TripDomain +import com.twitter.util.Future + +case class TripEngineQuery( + modelId: String, + sourceId: InternalId, + tripSourceId: String, + maxResult: Int, + params: Params) + +case class ConsumerEmbeddingBasedTripSimilarityEngine( + embeddingStoreLookUpMap: Map[String, ReadableStore[UserId, SimClustersEmbedding]], + tripCandidateSource: ReadableStore[TripDomain, Seq[TripTweet]], + statsReceiver: StatsReceiver, +) extends ReadableStore[TripEngineQuery, Seq[TripTweetWithScore]] { + import ConsumerEmbeddingBasedTripSimilarityEngine._ + + private val scopedStats = statsReceiver.scope(name) + private def fetchTopClusters(query: TripEngineQuery): Future[Option[Seq[ClusterId]]] = { + query.sourceId match { + case InternalId.UserId(userId) => + val embeddingStore = embeddingStoreLookUpMap.getOrElse( + query.modelId, + throw new IllegalArgumentException( + s"${this.getClass.getSimpleName}: " + + s"ModelId ${query.modelId} does not exist for embeddingStore" + ) + ) + embeddingStore.get(userId).map(_.map(_.topClusterIds(MaxClusters))) + case _ => + Future.None + } + } + private def fetchCandidates( + topClusters: Seq[ClusterId], + tripSourceId: String + ): Future[Seq[Seq[TripTweetWithScore]]] = { + Future + .collect { + topClusters.map { clusterId => + tripCandidateSource + .get( + TripDomain( + sourceId = tripSourceId, + clusterDomain = Some( + ClusterDomain(simCluster = Some(Cluster(clusterIntId = Some(clusterId))))))).map { + _.map { + _.collect { + case TripTweet(tweetId, score) => + TripTweetWithScore(tweetId, score) + } + }.getOrElse(Seq.empty).take(MaxNumResultsPerCluster) + } + } + } + } + + override def get(engineQuery: TripEngineQuery): Future[Option[Seq[TripTweetWithScore]]] = { + val fetchTopClustersStat = scopedStats.scope(engineQuery.modelId).scope("fetchTopClusters") + val fetchCandidatesStat = scopedStats.scope(engineQuery.modelId).scope("fetchCandidates") + + for { + topClustersOpt <- StatsUtil.trackOptionStats(fetchTopClustersStat) { + fetchTopClusters(engineQuery) + } + candidates <- StatsUtil.trackItemsStats(fetchCandidatesStat) { + topClustersOpt match { + case Some(topClusters) => fetchCandidates(topClusters, engineQuery.tripSourceId) + case None => Future.Nil + } + } + } yield { + val interleavedTweets = InterleaveUtil.interleave(candidates) + val dedupCandidates = interleavedTweets + .groupBy(_.tweetId).flatMap { + case (_, tweetWithScoreSeq) => tweetWithScoreSeq.sortBy(-_.score).take(1) + }.toSeq.take(engineQuery.maxResult) + Some(dedupCandidates) + } + } +} + +object ConsumerEmbeddingBasedTripSimilarityEngine { + private val MaxClusters: Int = 8 + private val MaxNumResultsPerCluster: Int = 25 + private val name: String = this.getClass.getSimpleName + + def fromParams( + modelId: String, + sourceId: InternalId, + params: configapi.Params + ): TripEngineQuery = { + TripEngineQuery( + modelId = modelId, + sourceId = sourceId, + tripSourceId = params(ConsumerEmbeddingBasedTripParams.SourceIdParam), + maxResult = params(ConsumerEmbeddingBasedTripParams.MaxNumCandidatesParam), + params = params + ) + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/ConsumerEmbeddingBasedTwHINSimilarityEngine.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/ConsumerEmbeddingBasedTwHINSimilarityEngine.scala new file mode 100644 index 0000000000..ed722f3eb6 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/ConsumerEmbeddingBasedTwHINSimilarityEngine.scala @@ -0,0 +1,18 @@ +package com.twitter.cr_mixer.similarity_engine + +import com.twitter.cr_mixer.param.ConsumerEmbeddingBasedTwHINParams +import com.twitter.simclusters_v2.thriftscala.InternalId +import com.twitter.timelines.configapi + +object ConsumerEmbeddingBasedTwHINSimilarityEngine { + def fromParams( + sourceId: InternalId, + params: configapi.Params, + ): HnswANNEngineQuery = { + HnswANNEngineQuery( + sourceId = sourceId, + modelId = params(ConsumerEmbeddingBasedTwHINParams.ModelIdParam), + params = params + ) + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/ConsumerEmbeddingBasedTwoTowerSimilarityEngine.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/ConsumerEmbeddingBasedTwoTowerSimilarityEngine.scala new file mode 100644 index 0000000000..c63d517d68 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/ConsumerEmbeddingBasedTwoTowerSimilarityEngine.scala @@ -0,0 +1,18 @@ +package com.twitter.cr_mixer.similarity_engine + +import com.twitter.cr_mixer.param.ConsumerEmbeddingBasedTwoTowerParams +import com.twitter.simclusters_v2.thriftscala.InternalId +import com.twitter.timelines.configapi + +object ConsumerEmbeddingBasedTwoTowerSimilarityEngine { + def fromParams( + sourceId: InternalId, + params: configapi.Params, + ): HnswANNEngineQuery = { + HnswANNEngineQuery( + sourceId = sourceId, + modelId = params(ConsumerEmbeddingBasedTwoTowerParams.ModelIdParam), + params = params + ) + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/ConsumersBasedUserAdGraphSimilarityEngine.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/ConsumersBasedUserAdGraphSimilarityEngine.scala new file mode 100644 index 0000000000..585edc584b --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/ConsumersBasedUserAdGraphSimilarityEngine.scala @@ -0,0 +1,90 @@ +package com.twitter.cr_mixer.similarity_engine + +import com.twitter.cr_mixer.model.SimilarityEngineInfo +import com.twitter.cr_mixer.model.TweetWithScore +import com.twitter.cr_mixer.param.ConsumersBasedUserAdGraphParams +import com.twitter.cr_mixer.param.GlobalParams +import com.twitter.cr_mixer.thriftscala.SimilarityEngineType +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.recos.user_ad_graph.thriftscala.ConsumersBasedRelatedAdRequest +import com.twitter.recos.user_ad_graph.thriftscala.RelatedAdResponse +import com.twitter.simclusters_v2.common.UserId +import com.twitter.storehaus.ReadableStore +import com.twitter.timelines.configapi +import com.twitter.util.Future +import javax.inject.Singleton + +/** + * This store uses the graph based input (a list of userIds) + * to query consumersBasedUserAdGraph and get their top engaged ad tweets + */ +@Singleton +case class ConsumersBasedUserAdGraphSimilarityEngine( + consumersBasedUserAdGraphStore: ReadableStore[ + ConsumersBasedRelatedAdRequest, + RelatedAdResponse + ], + statsReceiver: StatsReceiver) + extends ReadableStore[ + ConsumersBasedUserAdGraphSimilarityEngine.Query, + Seq[TweetWithScore] + ] { + + override def get( + query: ConsumersBasedUserAdGraphSimilarityEngine.Query + ): Future[Option[Seq[TweetWithScore]]] = { + val consumersBasedRelatedAdRequest = + ConsumersBasedRelatedAdRequest( + query.seedWithScores.keySet.toSeq, + maxResults = Some(query.maxResults), + minCooccurrence = Some(query.minCooccurrence), + minScore = Some(query.minScore), + maxTweetAgeInHours = Some(query.maxTweetAgeInHours) + ) + consumersBasedUserAdGraphStore + .get(consumersBasedRelatedAdRequest) + .map { relatedAdResponseOpt => + relatedAdResponseOpt.map { relatedAdResponse => + relatedAdResponse.adTweets.map { tweet => + TweetWithScore(tweet.adTweetId, tweet.score) + } + } + } + } +} + +object ConsumersBasedUserAdGraphSimilarityEngine { + + case class Query( + seedWithScores: Map[UserId, Double], + maxResults: Int, + minCooccurrence: Int, + minScore: Double, + maxTweetAgeInHours: Int) + + def toSimilarityEngineInfo( + score: Double + ): SimilarityEngineInfo = { + SimilarityEngineInfo( + similarityEngineType = SimilarityEngineType.ConsumersBasedUserAdGraph, + modelId = None, + score = Some(score)) + } + + def fromParams( + seedWithScores: Map[UserId, Double], + params: configapi.Params, + ): EngineQuery[Query] = { + + EngineQuery( + Query( + seedWithScores = seedWithScores, + maxResults = params(GlobalParams.MaxCandidateNumPerSourceKeyParam), + minCooccurrence = params(ConsumersBasedUserAdGraphParams.MinCoOccurrenceParam), + minScore = params(ConsumersBasedUserAdGraphParams.MinScoreParam), + maxTweetAgeInHours = params(GlobalParams.MaxTweetAgeHoursParam).inHours, + ), + params + ) + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/ConsumersBasedUserVideoGraphSimilarityEngine.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/ConsumersBasedUserVideoGraphSimilarityEngine.scala new file mode 100644 index 0000000000..633f5ee6dd --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/ConsumersBasedUserVideoGraphSimilarityEngine.scala @@ -0,0 +1,91 @@ +package com.twitter.cr_mixer.similarity_engine + +import com.twitter.cr_mixer.model.SimilarityEngineInfo +import com.twitter.cr_mixer.model.TweetWithScore +import com.twitter.cr_mixer.param.ConsumersBasedUserVideoGraphParams +import com.twitter.cr_mixer.param.GlobalParams +import com.twitter.cr_mixer.thriftscala.SimilarityEngineType +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.recos.user_video_graph.thriftscala.ConsumersBasedRelatedTweetRequest +import com.twitter.recos.user_video_graph.thriftscala.RelatedTweetResponse +import com.twitter.simclusters_v2.common.UserId +import com.twitter.storehaus.ReadableStore +import com.twitter.timelines.configapi +import com.twitter.util.Future +import javax.inject.Singleton + +/** + * This store uses the graph based input (a list of userIds) + * to query consumersBasedUserVideoGraph and get their top engaged tweets + */ +@Singleton +case class ConsumersBasedUserVideoGraphSimilarityEngine( + consumersBasedUserVideoGraphStore: ReadableStore[ + ConsumersBasedRelatedTweetRequest, + RelatedTweetResponse + ], + statsReceiver: StatsReceiver) + extends ReadableStore[ + ConsumersBasedUserVideoGraphSimilarityEngine.Query, + Seq[TweetWithScore] + ] { + + override def get( + query: ConsumersBasedUserVideoGraphSimilarityEngine.Query + ): Future[Option[Seq[TweetWithScore]]] = { + val consumersBasedRelatedTweetRequest = + ConsumersBasedRelatedTweetRequest( + query.seedWithScores.keySet.toSeq, + maxResults = Some(query.maxResults), + minCooccurrence = Some(query.minCooccurrence), + minScore = Some(query.minScore), + maxTweetAgeInHours = Some(query.maxTweetAgeInHours) + ) + consumersBasedUserVideoGraphStore + .get(consumersBasedRelatedTweetRequest) + .map { relatedTweetResponseOpt => + relatedTweetResponseOpt.map { relatedTweetResponse => + relatedTweetResponse.tweets.map { tweet => + TweetWithScore(tweet.tweetId, tweet.score) + } + } + } + } +} + +object ConsumersBasedUserVideoGraphSimilarityEngine { + + case class Query( + seedWithScores: Map[UserId, Double], + maxResults: Int, + minCooccurrence: Int, + minScore: Double, + maxTweetAgeInHours: Int) + + def toSimilarityEngineInfo( + score: Double + ): SimilarityEngineInfo = { + SimilarityEngineInfo( + similarityEngineType = SimilarityEngineType.ConsumersBasedUserVideoGraph, + modelId = None, + score = Some(score)) + } + + def fromParamsForRealGraphIn( + seedWithScores: Map[UserId, Double], + params: configapi.Params, + ): EngineQuery[Query] = { + + EngineQuery( + Query( + seedWithScores = seedWithScores, + maxResults = params(GlobalParams.MaxCandidateNumPerSourceKeyParam), + minCooccurrence = + params(ConsumersBasedUserVideoGraphParams.RealGraphInMinCoOccurrenceParam), + minScore = params(ConsumersBasedUserVideoGraphParams.RealGraphInMinScoreParam), + maxTweetAgeInHours = params(GlobalParams.MaxTweetAgeHoursParam).inHours + ), + params + ) + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/DiffusionBasedSimilarityEngine.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/DiffusionBasedSimilarityEngine.scala new file mode 100644 index 0000000000..a1bc0e2487 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/DiffusionBasedSimilarityEngine.scala @@ -0,0 +1,73 @@ +package com.twitter.cr_mixer.similarity_engine + +import com.twitter.cr_mixer.model.SimilarityEngineInfo +import com.twitter.simclusters_v2.thriftscala.TweetsWithScore +import com.twitter.simclusters_v2.thriftscala.InternalId +import com.twitter.cr_mixer.model.TweetWithScore +import com.twitter.cr_mixer.thriftscala.SimilarityEngineType +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.simclusters_v2.thriftscala.InternalId +import com.twitter.storehaus.ReadableStore +import com.twitter.timelines.configapi +import com.twitter.util.Future +import javax.inject.Singleton + +@Singleton +case class DiffusionBasedSimilarityEngine( + retweetBasedDiffusionRecsMhStore: ReadableStore[Long, TweetsWithScore], + statsReceiver: StatsReceiver) + extends ReadableStore[ + DiffusionBasedSimilarityEngine.Query, + Seq[TweetWithScore] + ] { + + override def get( + query: DiffusionBasedSimilarityEngine.Query + ): Future[Option[Seq[TweetWithScore]]] = { + + query.sourceId match { + case InternalId.UserId(userId) => + retweetBasedDiffusionRecsMhStore.get(userId).map { + _.map { tweetsWithScore => + { + tweetsWithScore.tweets + .map(tweet => TweetWithScore(tweet.tweetId, tweet.score)) + } + } + } + case _ => + Future.None + } + } +} + +object DiffusionBasedSimilarityEngine { + + val defaultScore: Double = 0.0 + + case class Query( + sourceId: InternalId, + ) + + def toSimilarityEngineInfo( + query: LookupEngineQuery[Query], + score: Double + ): SimilarityEngineInfo = { + SimilarityEngineInfo( + similarityEngineType = SimilarityEngineType.DiffusionBasedTweet, + modelId = Some(query.lookupKey), + score = Some(score)) + } + + def fromParams( + sourceId: InternalId, + modelId: String, + params: configapi.Params, + ): LookupEngineQuery[Query] = { + LookupEngineQuery( + Query(sourceId = sourceId), + modelId, + params + ) + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/EarlybirdModelBasedSimilarityEngine.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/EarlybirdModelBasedSimilarityEngine.scala new file mode 100644 index 0000000000..da82b4eb14 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/EarlybirdModelBasedSimilarityEngine.scala @@ -0,0 +1,92 @@ +package com.twitter.cr_mixer.similarity_engine + +import com.twitter.cr_mixer.config.TimeoutConfig +import com.twitter.cr_mixer.similarity_engine.EarlybirdModelBasedSimilarityEngine.EarlybirdModelBasedSearchQuery +import com.twitter.cr_mixer.similarity_engine.EarlybirdSimilarityEngineBase._ +import com.twitter.cr_mixer.util.EarlybirdSearchUtil.EarlybirdClientId +import com.twitter.cr_mixer.util.EarlybirdSearchUtil.FacetsToFetch +import com.twitter.cr_mixer.util.EarlybirdSearchUtil.MetadataOptions +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.finagle.tracing.Trace +import com.twitter.search.common.ranking.thriftscala.ThriftRankingParams +import com.twitter.search.common.ranking.thriftscala.ThriftScoringFunctionType +import com.twitter.search.earlybird.thriftscala.EarlybirdRequest +import com.twitter.search.earlybird.thriftscala.EarlybirdService +import com.twitter.search.earlybird.thriftscala.ThriftSearchQuery +import com.twitter.search.earlybird.thriftscala.ThriftSearchRankingMode +import com.twitter.search.earlybird.thriftscala.ThriftSearchRelevanceOptions +import com.twitter.simclusters_v2.common.UserId +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +case class EarlybirdModelBasedSimilarityEngine @Inject() ( + earlybirdSearchClient: EarlybirdService.MethodPerEndpoint, + timeoutConfig: TimeoutConfig, + stats: StatsReceiver) + extends EarlybirdSimilarityEngineBase[EarlybirdModelBasedSearchQuery] { + import EarlybirdModelBasedSimilarityEngine._ + override val statsReceiver: StatsReceiver = stats.scope(this.getClass.getSimpleName) + override def getEarlybirdRequest( + query: EarlybirdModelBasedSearchQuery + ): Option[EarlybirdRequest] = + if (query.seedUserIds.nonEmpty) + Some( + EarlybirdRequest( + searchQuery = getThriftSearchQuery(query), + clientId = Some(EarlybirdClientId), + timeoutMs = timeoutConfig.earlybirdServerTimeout.inMilliseconds.intValue(), + clientRequestID = Some(s"${Trace.id.traceId}"), + )) + else None +} + +object EarlybirdModelBasedSimilarityEngine { + case class EarlybirdModelBasedSearchQuery( + seedUserIds: Seq[UserId], + maxNumTweets: Int, + oldestTweetTimestampInSec: Option[UserId], + frsUserToScoresForScoreAdjustment: Option[Map[UserId, Double]]) + extends EarlybirdSearchQuery + + /** + * Used by Push Service + */ + val RealGraphScoringModel = "frigate_unified_engagement_rg" + val MaxHitsToProcess = 1000 + val MaxConsecutiveSameUser = 1 + + private def getModelBasedRankingParams( + authorSpecificScoreAdjustments: Map[Long, Double] + ): ThriftRankingParams = ThriftRankingParams( + `type` = Some(ThriftScoringFunctionType.ModelBased), + selectedModels = Some(Map(RealGraphScoringModel -> 1.0)), + applyBoosts = false, + authorSpecificScoreAdjustments = Some(authorSpecificScoreAdjustments) + ) + + private def getRelevanceOptions( + authorSpecificScoreAdjustments: Map[Long, Double], + ): ThriftSearchRelevanceOptions = { + ThriftSearchRelevanceOptions( + maxConsecutiveSameUser = Some(MaxConsecutiveSameUser), + rankingParams = Some(getModelBasedRankingParams(authorSpecificScoreAdjustments)), + maxHitsToProcess = Some(MaxHitsToProcess), + orderByRelevance = true + ) + } + + private def getThriftSearchQuery(query: EarlybirdModelBasedSearchQuery): ThriftSearchQuery = + ThriftSearchQuery( + serializedQuery = Some(f"(* [since_time ${query.oldestTweetTimestampInSec.getOrElse(0)}])"), + fromUserIDFilter64 = Some(query.seedUserIds), + numResults = query.maxNumTweets, + maxHitsToProcess = MaxHitsToProcess, + rankingMode = ThriftSearchRankingMode.Relevance, + relevanceOptions = + Some(getRelevanceOptions(query.frsUserToScoresForScoreAdjustment.getOrElse(Map.empty))), + facetFieldNames = Some(FacetsToFetch), + resultMetadataOptions = Some(MetadataOptions), + searcherId = None + ) +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/EarlybirdRecencyBasedSimilarityEngine.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/EarlybirdRecencyBasedSimilarityEngine.scala new file mode 100644 index 0000000000..988d666a43 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/EarlybirdRecencyBasedSimilarityEngine.scala @@ -0,0 +1,86 @@ +package com.twitter.cr_mixer.similarity_engine +import com.twitter.cr_mixer.config.TimeoutConfig +import com.twitter.cr_mixer.model.ModuleNames +import com.twitter.cr_mixer.model.TweetWithAuthor +import com.twitter.cr_mixer.similarity_engine.EarlybirdRecencyBasedSimilarityEngine.EarlybirdRecencyBasedSearchQuery +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.frigate.common.util.StatsUtil +import com.twitter.simclusters_v2.common.TweetId +import com.twitter.simclusters_v2.common.UserId +import com.twitter.snowflake.id.SnowflakeId +import com.twitter.storehaus.ReadableStore +import com.twitter.util.Duration +import com.twitter.util.Future +import com.twitter.util.Time +import javax.inject.Inject +import javax.inject.Named +import javax.inject.Singleton + +@Singleton +case class EarlybirdRecencyBasedSimilarityEngine @Inject() ( + @Named(ModuleNames.EarlybirdRecencyBasedWithoutRetweetsRepliesTweetsCache) + earlybirdRecencyBasedWithoutRetweetsRepliesTweetsCacheStore: ReadableStore[ + UserId, + Seq[TweetId] + ], + @Named(ModuleNames.EarlybirdRecencyBasedWithRetweetsRepliesTweetsCache) + earlybirdRecencyBasedWithRetweetsRepliesTweetsCacheStore: ReadableStore[ + UserId, + Seq[TweetId] + ], + timeoutConfig: TimeoutConfig, + stats: StatsReceiver) + extends ReadableStore[EarlybirdRecencyBasedSearchQuery, Seq[TweetWithAuthor]] { + import EarlybirdRecencyBasedSimilarityEngine._ + val statsReceiver: StatsReceiver = stats.scope(this.getClass.getSimpleName) + + override def get( + query: EarlybirdRecencyBasedSearchQuery + ): Future[Option[Seq[TweetWithAuthor]]] = { + Future + .collect { + if (query.filterOutRetweetsAndReplies) { + query.seedUserIds.map { seedUserId => + StatsUtil.trackOptionItemsStats(statsReceiver.scope("WithoutRetweetsAndReplies")) { + earlybirdRecencyBasedWithoutRetweetsRepliesTweetsCacheStore + .get(seedUserId).map(_.map(_.map(tweetId => + TweetWithAuthor(tweetId = tweetId, authorId = seedUserId)))) + } + } + } else { + query.seedUserIds.map { seedUserId => + StatsUtil.trackOptionItemsStats(statsReceiver.scope("WithRetweetsAndReplies")) { + earlybirdRecencyBasedWithRetweetsRepliesTweetsCacheStore + .get(seedUserId) + .map(_.map(_.map(tweetId => + TweetWithAuthor(tweetId = tweetId, authorId = seedUserId)))) + } + } + } + } + .map { tweetWithAuthorList => + val earliestTweetId = SnowflakeId.firstIdFor(Time.now - query.maxTweetAge) + tweetWithAuthorList + .flatMap(_.getOrElse(Seq.empty)) + .filter(tweetWithAuthor => + tweetWithAuthor.tweetId >= earliestTweetId // tweet age filter + && !query.excludedTweetIds + .contains(tweetWithAuthor.tweetId)) // excluded tweet filter + .sortBy(tweetWithAuthor => + -SnowflakeId.unixTimeMillisFromId(tweetWithAuthor.tweetId)) // sort by recency + .take(query.maxNumTweets) // take most recent N tweets + } + .map(result => Some(result)) + } + +} + +object EarlybirdRecencyBasedSimilarityEngine { + case class EarlybirdRecencyBasedSearchQuery( + seedUserIds: Seq[UserId], + maxNumTweets: Int, + excludedTweetIds: Set[TweetId], + maxTweetAge: Duration, + filterOutRetweetsAndReplies: Boolean) + +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/EarlybirdSimilarityEngine.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/EarlybirdSimilarityEngine.scala new file mode 100644 index 0000000000..be23134eb1 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/EarlybirdSimilarityEngine.scala @@ -0,0 +1,32 @@ +package com.twitter.cr_mixer.similarity_engine + +import com.twitter.cr_mixer.model.TweetWithAuthor +import com.twitter.cr_mixer.similarity_engine.SimilarityEngine.SimilarityEngineConfig +import com.twitter.cr_mixer.thriftscala.SimilarityEngineType +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.storehaus.ReadableStore +import com.twitter.util.Future + +class EarlybirdSimilarityEngine[ + Query, + EarlybirdSimilarityEngineStore <: ReadableStore[Query, Seq[TweetWithAuthor]] +]( + implementingStore: EarlybirdSimilarityEngineStore, + override val identifier: SimilarityEngineType, + globalStats: StatsReceiver, + engineConfig: SimilarityEngineConfig, +) extends SimilarityEngine[EngineQuery[Query], TweetWithAuthor] { + private val scopedStats = globalStats.scope("similarityEngine", identifier.toString) + + def getScopedStats: StatsReceiver = scopedStats + + def getCandidates(query: EngineQuery[Query]): Future[Option[Seq[TweetWithAuthor]]] = { + SimilarityEngine.getFromFn( + implementingStore.get, + query.storeQuery, + engineConfig, + query.params, + scopedStats + ) + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/EarlybirdSimilarityEngineBase.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/EarlybirdSimilarityEngineBase.scala new file mode 100644 index 0000000000..ab4eb408e2 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/EarlybirdSimilarityEngineBase.scala @@ -0,0 +1,56 @@ +package com.twitter.cr_mixer.similarity_engine + +import com.twitter.cr_mixer.model.TweetWithAuthor +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.search.earlybird.thriftscala.EarlybirdRequest +import com.twitter.search.earlybird.thriftscala.EarlybirdResponseCode +import com.twitter.search.earlybird.thriftscala.EarlybirdService +import com.twitter.simclusters_v2.common.UserId +import com.twitter.storehaus.ReadableStore +import com.twitter.util.Future + +/** + * This trait is a base trait for Earlybird similarity engines. All Earlybird similarity + * engines extend from it and override the construction method for EarlybirdRequest + */ +trait EarlybirdSimilarityEngineBase[EarlybirdSearchQuery] + extends ReadableStore[EarlybirdSearchQuery, Seq[TweetWithAuthor]] { + def earlybirdSearchClient: EarlybirdService.MethodPerEndpoint + + def statsReceiver: StatsReceiver + + def getEarlybirdRequest(query: EarlybirdSearchQuery): Option[EarlybirdRequest] + + override def get(query: EarlybirdSearchQuery): Future[Option[Seq[TweetWithAuthor]]] = { + getEarlybirdRequest(query) + .map { earlybirdRequest => + earlybirdSearchClient + .search(earlybirdRequest).map { response => + response.responseCode match { + case EarlybirdResponseCode.Success => + val earlybirdSearchResult = + response.searchResults + .map( + _.results + .map(searchResult => + TweetWithAuthor( + searchResult.id, + // fromUserId should be there since MetadataOptions.getFromUserId = true + searchResult.metadata.map(_.fromUserId).getOrElse(0))).toSeq) + statsReceiver.scope("result").stat("size").add(earlybirdSearchResult.size) + earlybirdSearchResult + case e => + statsReceiver.scope("failures").counter(e.getClass.getSimpleName).incr() + Some(Seq.empty) + } + } + }.getOrElse(Future.None) + } +} + +object EarlybirdSimilarityEngineBase { + trait EarlybirdSearchQuery { + def seedUserIds: Seq[UserId] + def maxNumTweets: Int + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/EarlybirdSimilarityEngineRouter.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/EarlybirdSimilarityEngineRouter.scala new file mode 100644 index 0000000000..3237f13f8a --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/EarlybirdSimilarityEngineRouter.scala @@ -0,0 +1,136 @@ +package com.twitter.cr_mixer.similarity_engine + +import com.twitter.cr_mixer.config.TimeoutConfig +import com.twitter.cr_mixer.model.EarlybirdSimilarityEngineType +import com.twitter.cr_mixer.model.EarlybirdSimilarityEngineType_ModelBased +import com.twitter.cr_mixer.model.EarlybirdSimilarityEngineType_RecencyBased +import com.twitter.cr_mixer.model.EarlybirdSimilarityEngineType_TensorflowBased +import com.twitter.cr_mixer.model.TweetWithAuthor +import com.twitter.cr_mixer.param.EarlybirdFrsBasedCandidateGenerationParams +import com.twitter.cr_mixer.param.EarlybirdFrsBasedCandidateGenerationParams.FrsBasedCandidateGenerationEarlybirdSimilarityEngineTypeParam +import com.twitter.cr_mixer.param.FrsParams.FrsBasedCandidateGenerationMaxCandidatesNumParam +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.simclusters_v2.common.TweetId +import com.twitter.simclusters_v2.common.UserId +import com.twitter.snowflake.id.SnowflakeId +import com.twitter.storehaus.ReadableStore +import com.twitter.timelines.configapi +import com.twitter.util.Duration +import com.twitter.util.Future +import com.twitter.util.Time +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +case class EarlybirdSimilarityEngineRouter @Inject() ( + earlybirdRecencyBasedSimilarityEngine: EarlybirdSimilarityEngine[ + EarlybirdRecencyBasedSimilarityEngine.EarlybirdRecencyBasedSearchQuery, + EarlybirdRecencyBasedSimilarityEngine + ], + earlybirdModelBasedSimilarityEngine: EarlybirdSimilarityEngine[ + EarlybirdModelBasedSimilarityEngine.EarlybirdModelBasedSearchQuery, + EarlybirdModelBasedSimilarityEngine + ], + earlybirdTensorflowBasedSimilarityEngine: EarlybirdSimilarityEngine[ + EarlybirdTensorflowBasedSimilarityEngine.EarlybirdTensorflowBasedSearchQuery, + EarlybirdTensorflowBasedSimilarityEngine + ], + timeoutConfig: TimeoutConfig, + statsReceiver: StatsReceiver) + extends ReadableStore[EarlybirdSimilarityEngineRouter.Query, Seq[TweetWithAuthor]] { + import EarlybirdSimilarityEngineRouter._ + + override def get( + k: EarlybirdSimilarityEngineRouter.Query + ): Future[Option[Seq[TweetWithAuthor]]] = { + k.rankingMode match { + case EarlybirdSimilarityEngineType_RecencyBased => + earlybirdRecencyBasedSimilarityEngine.getCandidates(recencyBasedQueryFromParams(k)) + case EarlybirdSimilarityEngineType_ModelBased => + earlybirdModelBasedSimilarityEngine.getCandidates(modelBasedQueryFromParams(k)) + case EarlybirdSimilarityEngineType_TensorflowBased => + earlybirdTensorflowBasedSimilarityEngine.getCandidates(tensorflowBasedQueryFromParams(k)) + } + } +} + +object EarlybirdSimilarityEngineRouter { + case class Query( + searcherUserId: Option[UserId], + seedUserIds: Seq[UserId], + maxNumTweets: Int, + excludedTweetIds: Set[TweetId], + rankingMode: EarlybirdSimilarityEngineType, + frsUserToScoresForScoreAdjustment: Option[Map[UserId, Double]], + maxTweetAge: Duration, + filterOutRetweetsAndReplies: Boolean, + params: configapi.Params) + + def queryFromParams( + searcherUserId: Option[UserId], + seedUserIds: Seq[UserId], + excludedTweetIds: Set[TweetId], + frsUserToScoresForScoreAdjustment: Option[Map[UserId, Double]], + params: configapi.Params + ): Query = + Query( + searcherUserId, + seedUserIds, + maxNumTweets = params(FrsBasedCandidateGenerationMaxCandidatesNumParam), + excludedTweetIds, + rankingMode = + params(FrsBasedCandidateGenerationEarlybirdSimilarityEngineTypeParam).rankingMode, + frsUserToScoresForScoreAdjustment, + maxTweetAge = params( + EarlybirdFrsBasedCandidateGenerationParams.FrsBasedCandidateGenerationEarlybirdMaxTweetAge), + filterOutRetweetsAndReplies = params( + EarlybirdFrsBasedCandidateGenerationParams.FrsBasedCandidateGenerationEarlybirdFilterOutRetweetsAndReplies), + params + ) + + private def recencyBasedQueryFromParams( + query: Query + ): EngineQuery[EarlybirdRecencyBasedSimilarityEngine.EarlybirdRecencyBasedSearchQuery] = + EngineQuery( + EarlybirdRecencyBasedSimilarityEngine.EarlybirdRecencyBasedSearchQuery( + seedUserIds = query.seedUserIds, + maxNumTweets = query.maxNumTweets, + excludedTweetIds = query.excludedTweetIds, + maxTweetAge = query.maxTweetAge, + filterOutRetweetsAndReplies = query.filterOutRetweetsAndReplies + ), + query.params + ) + + private def tensorflowBasedQueryFromParams( + query: Query, + ): EngineQuery[EarlybirdTensorflowBasedSimilarityEngine.EarlybirdTensorflowBasedSearchQuery] = + EngineQuery( + EarlybirdTensorflowBasedSimilarityEngine.EarlybirdTensorflowBasedSearchQuery( + searcherUserId = query.searcherUserId, + seedUserIds = query.seedUserIds, + maxNumTweets = query.maxNumTweets, + // hard code the params below for now. Will move to FS after shipping the ddg + beforeTweetIdExclusive = None, + afterTweetIdExclusive = + Some(SnowflakeId.firstIdFor((Time.now - query.maxTweetAge).inMilliseconds)), + filterOutRetweetsAndReplies = query.filterOutRetweetsAndReplies, + useTensorflowRanking = true, + excludedTweetIds = query.excludedTweetIds, + maxNumHitsPerShard = 1000 + ), + query.params + ) + private def modelBasedQueryFromParams( + query: Query, + ): EngineQuery[EarlybirdModelBasedSimilarityEngine.EarlybirdModelBasedSearchQuery] = + EngineQuery( + EarlybirdModelBasedSimilarityEngine.EarlybirdModelBasedSearchQuery( + seedUserIds = query.seedUserIds, + maxNumTweets = query.maxNumTweets, + oldestTweetTimestampInSec = Some(query.maxTweetAge.ago.inSeconds), + frsUserToScoresForScoreAdjustment = query.frsUserToScoresForScoreAdjustment + ), + query.params + ) +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/EarlybirdTensorflowBasedSimilarityEngine.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/EarlybirdTensorflowBasedSimilarityEngine.scala new file mode 100644 index 0000000000..dd29a067b1 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/EarlybirdTensorflowBasedSimilarityEngine.scala @@ -0,0 +1,171 @@ +package com.twitter.cr_mixer.similarity_engine + +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.search.earlybird.thriftscala.EarlybirdRequest +import com.twitter.search.earlybird.thriftscala.EarlybirdService +import com.twitter.search.earlybird.thriftscala.ThriftSearchQuery +import com.twitter.util.Time +import com.twitter.search.common.query.thriftjava.thriftscala.CollectorParams +import com.twitter.search.common.ranking.thriftscala.ThriftAgeDecayRankingParams +import com.twitter.search.common.ranking.thriftscala.ThriftLinearFeatureRankingParams +import com.twitter.search.common.ranking.thriftscala.ThriftRankingParams +import com.twitter.search.common.ranking.thriftscala.ThriftScoringFunctionType +import com.twitter.search.earlybird.thriftscala.ThriftSearchRelevanceOptions +import javax.inject.Inject +import javax.inject.Singleton +import EarlybirdSimilarityEngineBase._ +import com.twitter.cr_mixer.config.TimeoutConfig +import com.twitter.cr_mixer.similarity_engine.EarlybirdTensorflowBasedSimilarityEngine.EarlybirdTensorflowBasedSearchQuery +import com.twitter.cr_mixer.util.EarlybirdSearchUtil.EarlybirdClientId +import com.twitter.cr_mixer.util.EarlybirdSearchUtil.FacetsToFetch +import com.twitter.cr_mixer.util.EarlybirdSearchUtil.GetCollectorTerminationParams +import com.twitter.cr_mixer.util.EarlybirdSearchUtil.GetEarlybirdQuery +import com.twitter.cr_mixer.util.EarlybirdSearchUtil.MetadataOptions +import com.twitter.cr_mixer.util.EarlybirdSearchUtil.GetNamedDisjunctions +import com.twitter.search.earlybird.thriftscala.ThriftSearchRankingMode +import com.twitter.simclusters_v2.common.TweetId +import com.twitter.simclusters_v2.common.UserId +import com.twitter.util.Duration + +@Singleton +case class EarlybirdTensorflowBasedSimilarityEngine @Inject() ( + earlybirdSearchClient: EarlybirdService.MethodPerEndpoint, + timeoutConfig: TimeoutConfig, + stats: StatsReceiver) + extends EarlybirdSimilarityEngineBase[EarlybirdTensorflowBasedSearchQuery] { + import EarlybirdTensorflowBasedSimilarityEngine._ + override val statsReceiver: StatsReceiver = stats.scope(this.getClass.getSimpleName) + override def getEarlybirdRequest( + query: EarlybirdTensorflowBasedSearchQuery + ): Option[EarlybirdRequest] = { + if (query.seedUserIds.nonEmpty) + Some( + EarlybirdRequest( + searchQuery = getThriftSearchQuery(query, timeoutConfig.earlybirdServerTimeout), + clientHost = None, + clientRequestID = None, + clientId = Some(EarlybirdClientId), + clientRequestTimeMs = Some(Time.now.inMilliseconds), + cachingParams = None, + timeoutMs = timeoutConfig.earlybirdServerTimeout.inMilliseconds.intValue(), + facetRequest = None, + termStatisticsRequest = None, + debugMode = 0, + debugOptions = None, + searchSegmentId = None, + returnStatusType = None, + successfulResponseThreshold = None, + querySource = None, + getOlderResults = Some(false), + followedUserIds = Some(query.seedUserIds), + adjustedProtectedRequestParams = None, + adjustedFullArchiveRequestParams = None, + getProtectedTweetsOnly = Some(false), + retokenizeSerializedQuery = None, + skipVeryRecentTweets = true, + experimentClusterToUse = None + )) + else None + } +} + +object EarlybirdTensorflowBasedSimilarityEngine { + case class EarlybirdTensorflowBasedSearchQuery( + searcherUserId: Option[UserId], + seedUserIds: Seq[UserId], + maxNumTweets: Int, + beforeTweetIdExclusive: Option[TweetId], + afterTweetIdExclusive: Option[TweetId], + filterOutRetweetsAndReplies: Boolean, + useTensorflowRanking: Boolean, + excludedTweetIds: Set[TweetId], + maxNumHitsPerShard: Int) + extends EarlybirdSearchQuery + + private def getThriftSearchQuery( + query: EarlybirdTensorflowBasedSearchQuery, + processingTimeout: Duration + ): ThriftSearchQuery = + ThriftSearchQuery( + serializedQuery = GetEarlybirdQuery( + query.beforeTweetIdExclusive, + query.afterTweetIdExclusive, + query.excludedTweetIds, + query.filterOutRetweetsAndReplies).map(_.serialize), + fromUserIDFilter64 = Some(query.seedUserIds), + numResults = query.maxNumTweets, + // Whether to collect conversation IDs. Remove it for now. + // collectConversationId = Gate.True(), // true for Home + rankingMode = ThriftSearchRankingMode.Relevance, + relevanceOptions = Some(getRelevanceOptions(query.useTensorflowRanking)), + collectorParams = Some( + CollectorParams( + // numResultsToReturn defines how many results each EB shard will return to search root + numResultsToReturn = 1000, + // terminationParams.maxHitsToProcess is used for early terminating per shard results fetching. + terminationParams = + GetCollectorTerminationParams(query.maxNumHitsPerShard, processingTimeout) + )), + facetFieldNames = Some(FacetsToFetch), + resultMetadataOptions = Some(MetadataOptions), + searcherId = query.searcherUserId, + searchStatusIds = None, + namedDisjunctionMap = GetNamedDisjunctions(query.excludedTweetIds) + ) + + // The specific values of recap relevance/reranking options correspond to + // experiment: enable_recap_reranking_2988,timeline_internal_disable_recap_filter + // bucket : enable_rerank,disable_filter + private def getRelevanceOptions(useTensorflowRanking: Boolean): ThriftSearchRelevanceOptions = { + ThriftSearchRelevanceOptions( + proximityScoring = true, + maxConsecutiveSameUser = Some(2), + rankingParams = + if (useTensorflowRanking) Some(getTensorflowBasedRankingParams) + else Some(getLinearRankingParams), + maxHitsToProcess = Some(500), + maxUserBlendCount = Some(3), + proximityPhraseWeight = 9.0, + returnAllResults = Some(true) + ) + } + + private def getTensorflowBasedRankingParams: ThriftRankingParams = { + getLinearRankingParams.copy( + `type` = Some(ThriftScoringFunctionType.TensorflowBased), + selectedTensorflowModel = Some("timelines_rectweet_replica"), + applyBoosts = false, + authorSpecificScoreAdjustments = None + ) + } + + private def getLinearRankingParams: ThriftRankingParams = { + ThriftRankingParams( + `type` = Some(ThriftScoringFunctionType.Linear), + minScore = -1.0e100, + retweetCountParams = Some(ThriftLinearFeatureRankingParams(weight = 20.0)), + replyCountParams = Some(ThriftLinearFeatureRankingParams(weight = 1.0)), + reputationParams = Some(ThriftLinearFeatureRankingParams(weight = 0.2)), + luceneScoreParams = Some(ThriftLinearFeatureRankingParams(weight = 2.0)), + textScoreParams = Some(ThriftLinearFeatureRankingParams(weight = 0.18)), + urlParams = Some(ThriftLinearFeatureRankingParams(weight = 2.0)), + isReplyParams = Some(ThriftLinearFeatureRankingParams(weight = 1.0)), + favCountParams = Some(ThriftLinearFeatureRankingParams(weight = 30.0)), + langEnglishUIBoost = 0.5, + langEnglishTweetBoost = 0.2, + langDefaultBoost = 0.02, + unknownLanguageBoost = 0.05, + offensiveBoost = 0.1, + inTrustedCircleBoost = 3.0, + multipleHashtagsOrTrendsBoost = 0.6, + inDirectFollowBoost = 4.0, + tweetHasTrendBoost = 1.1, + selfTweetBoost = 2.0, + tweetHasImageUrlBoost = 2.0, + tweetHasVideoUrlBoost = 2.0, + useUserLanguageInfo = true, + ageDecayParams = Some(ThriftAgeDecayRankingParams(slope = 0.005, base = 1.0)) + ) + } + +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/FilterUtil.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/FilterUtil.scala new file mode 100644 index 0000000000..4cd94d2bb3 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/FilterUtil.scala @@ -0,0 +1,42 @@ +package com.twitter.cr_mixer.similarity_engine + +import com.twitter.cr_mixer.model.SourceInfo +import com.twitter.cr_mixer.model.TweetWithScore +import com.twitter.simclusters_v2.thriftscala.InternalId +import com.twitter.snowflake.id.SnowflakeId +import com.twitter.util.Duration +import com.twitter.util.Time + +object FilterUtil { + + /** Returns a list of tweets that are generated less than `maxTweetAgeHours` hours ago */ + def tweetAgeFilter( + candidates: Seq[TweetWithScore], + maxTweetAgeHours: Duration + ): Seq[TweetWithScore] = { + // Tweet IDs are approximately chronological (see http://go/snowflake), + // so we are building the earliest tweet id once + // The per-candidate logic here then be candidate.tweetId > earliestPermittedTweetId, which is far cheaper. + // See @cyao's phab on CrMixer generic age filter for reference https://phabricator.twitter.biz/D903188 + val earliestTweetId = SnowflakeId.firstIdFor(Time.now - maxTweetAgeHours) + candidates.filter { candidate => candidate.tweetId >= earliestTweetId } + } + + /** Returns a list of tweet sources that are generated less than `maxTweetAgeHours` hours ago */ + def tweetSourceAgeFilter( + candidates: Seq[SourceInfo], + maxTweetSignalAgeHoursParam: Duration + ): Seq[SourceInfo] = { + // Tweet IDs are approximately chronological (see http://go/snowflake), + // so we are building the earliest tweet id once + // This filter applies to source signals. Some candidate source calls can be avoided if source signals + // can be filtered. + val earliestTweetId = SnowflakeId.firstIdFor(Time.now - maxTweetSignalAgeHoursParam) + candidates.filter { candidate => + candidate.internalId match { + case InternalId.TweetId(tweetId) => tweetId >= earliestTweetId + case _ => false + } + } + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/HnswANNSimilarityEngine.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/HnswANNSimilarityEngine.scala new file mode 100644 index 0000000000..4a1422ce9e --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/HnswANNSimilarityEngine.scala @@ -0,0 +1,187 @@ +package com.twitter.cr_mixer.similarity_engine + +import com.twitter.ann.common.thriftscala.AnnQueryService +import com.twitter.ann.common.thriftscala.Distance +import com.twitter.ann.common.thriftscala.NearestNeighborQuery +import com.twitter.ann.hnsw.HnswCommon +import com.twitter.ann.hnsw.HnswParams +import com.twitter.bijection.Injection +import com.twitter.cortex.ml.embeddings.common.TweetKind +import com.twitter.cr_mixer.model.SimilarityEngineInfo +import com.twitter.cr_mixer.model.TweetWithScore +import com.twitter.cr_mixer.similarity_engine.SimilarityEngine.MemCacheConfig +import com.twitter.cr_mixer.similarity_engine.SimilarityEngine.SimilarityEngineConfig +import com.twitter.cr_mixer.thriftscala.SimilarityEngineType +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.frigate.common.util.StatsUtil +import com.twitter.mediaservices.commons.codec.ArrayByteBufferCodec +import com.twitter.ml.api.thriftscala.{Embedding => ThriftEmbedding} +import com.twitter.ml.featurestore.lib +import com.twitter.simclusters_v2.thriftscala.InternalId +import com.twitter.storehaus.ReadableStore +import com.twitter.timelines.configapi.Params +import com.twitter.util.Future + +case class HnswANNEngineQuery( + modelId: String, + sourceId: InternalId, + params: Params, +) { + val cacheKey: String = s"${modelId}_${sourceId.toString}" +} + +/** + * This Engine looks for tweets whose similarity is close to a Source Dense Embedding. + * Only support Long based embedding lookup. UserId or TweetId. + * + * It provides HNSW specific implementations + * + * @param memCacheConfigOpt If specified, it will wrap the underlying store with a MemCache layer + * You should only enable this for cacheable queries, e.x. TweetIds. + * consumer based UserIds are generally not possible to cache. + */ +class HnswANNSimilarityEngine( + embeddingStoreLookUpMap: Map[String, ReadableStore[InternalId, ThriftEmbedding]], + annServiceLookUpMap: Map[String, AnnQueryService.MethodPerEndpoint], + globalStats: StatsReceiver, + override val identifier: SimilarityEngineType, + engineConfig: SimilarityEngineConfig, + memCacheConfigOpt: Option[MemCacheConfig[HnswANNEngineQuery]] = None) + extends SimilarityEngine[HnswANNEngineQuery, TweetWithScore] { + + private val MaxNumResults: Int = 200 + private val ef: Int = 800 + private val TweetIdByteInjection: Injection[lib.TweetId, Array[Byte]] = TweetKind.byteInjection + + private val scopedStats = globalStats.scope("similarityEngine", identifier.toString) + + def getScopedStats: StatsReceiver = scopedStats + + private def fetchEmbedding( + query: HnswANNEngineQuery, + ): Future[Option[ThriftEmbedding]] = { + val embeddingStore = embeddingStoreLookUpMap.getOrElse( + query.modelId, + throw new IllegalArgumentException( + s"${this.getClass.getSimpleName} ${identifier.toString}: " + + s"ModelId ${query.modelId} does not exist for embeddingStore" + ) + ) + + embeddingStore.get(query.sourceId) + } + + private def fetchCandidates( + query: HnswANNEngineQuery, + embedding: ThriftEmbedding + ): Future[Seq[TweetWithScore]] = { + val annService = annServiceLookUpMap.getOrElse( + query.modelId, + throw new IllegalArgumentException( + s"${this.getClass.getSimpleName} ${identifier.toString}: " + + s"ModelId ${query.modelId} does not exist for annStore" + ) + ) + + val hnswParams = HnswCommon.RuntimeParamsInjection.apply(HnswParams(ef)) + + val annQuery = + NearestNeighborQuery(embedding, withDistance = true, hnswParams, MaxNumResults) + + annService + .query(annQuery) + .map( + _.nearestNeighbors + .map { nearestNeighbor => + val candidateId = TweetIdByteInjection + .invert(ArrayByteBufferCodec.decode(nearestNeighbor.id)) + .toOption + .map(_.tweetId) + (candidateId, nearestNeighbor.distance) + }.collect { + case (Some(candidateId), Some(distance)) => + TweetWithScore(candidateId, toScore(distance)) + }) + } + + // Convert Distance to a score such that higher scores mean more similar. + def toScore(distance: Distance): Double = { + distance match { + case Distance.EditDistance(editDistance) => + // (-Infinite, 0.0] + 0.0 - editDistance.distance + case Distance.L2Distance(l2Distance) => + // (-Infinite, 0.0] + 0.0 - l2Distance.distance + case Distance.CosineDistance(cosineDistance) => + // [0.0 - 1.0] + 1.0 - cosineDistance.distance + case Distance.InnerProductDistance(innerProductDistance) => + // (-Infinite, Infinite) + 1.0 - innerProductDistance.distance + case Distance.UnknownUnionField(_) => + throw new IllegalStateException( + s"${this.getClass.getSimpleName} does not recognize $distance.toString" + ) + } + } + + private[similarity_engine] def getEmbeddingAndCandidates( + query: HnswANNEngineQuery + ): Future[Option[Seq[TweetWithScore]]] = { + + val fetchEmbeddingStat = scopedStats.scope(query.modelId).scope("fetchEmbedding") + val fetchCandidatesStat = scopedStats.scope(query.modelId).scope("fetchCandidates") + + for { + embeddingOpt <- StatsUtil.trackOptionStats(fetchEmbeddingStat) { fetchEmbedding(query) } + candidates <- StatsUtil.trackItemsStats(fetchCandidatesStat) { + + embeddingOpt match { + case Some(embedding) => fetchCandidates(query, embedding) + case None => Future.Nil + } + } + } yield { + Some(candidates) + } + } + + // Add memcache wrapper, if specified + private val store = { + val uncachedStore = ReadableStore.fromFnFuture(getEmbeddingAndCandidates) + + memCacheConfigOpt match { + case Some(config) => + SimilarityEngine.addMemCache( + underlyingStore = uncachedStore, + memCacheConfig = config, + statsReceiver = scopedStats + ) + case _ => uncachedStore + } + } + + def toSimilarityEngineInfo( + query: HnswANNEngineQuery, + score: Double + ): SimilarityEngineInfo = { + SimilarityEngineInfo( + similarityEngineType = this.identifier, + modelId = Some(query.modelId), + score = Some(score)) + } + + override def getCandidates( + engineQuery: HnswANNEngineQuery + ): Future[Option[Seq[TweetWithScore]]] = { + val versionedStats = globalStats.scope(engineQuery.modelId) + SimilarityEngine.getFromFn( + store.get, + engineQuery, + engineConfig, + engineQuery.params, + versionedStats + ) + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/LookupSimilarityEngine.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/LookupSimilarityEngine.scala new file mode 100644 index 0000000000..c4e4698999 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/LookupSimilarityEngine.scala @@ -0,0 +1,78 @@ +package com.twitter.cr_mixer.similarity_engine + +import com.twitter.cr_mixer.similarity_engine.SimilarityEngine.MemCacheConfig +import com.twitter.cr_mixer.similarity_engine.SimilarityEngine.SimilarityEngineConfig +import com.twitter.cr_mixer.thriftscala.SimilarityEngineType +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.storehaus.ReadableStore +import com.twitter.timelines.configapi.Params +import com.twitter.util.Future + +case class LookupEngineQuery[Query]( + storeQuery: Query, // the actual Query type of the underlying store + lookupKey: String, + params: Params, +) + +/** + * This Engine provides a map interface for looking up different model implementations. + * It provides modelId level monitoring for free. + * + * Example use cases include OfflineSimClusters lookup + * + * + * @param versionedStoreMap A mapping from a modelId to a corresponding implementation + * @param memCacheConfigOpt If specified, it will wrap the underlying store with a MemCache layer + * You should only enable this for cacheable queries, e.x. TweetIds. + * consumer based UserIds are generally not possible to cache. + */ +class LookupSimilarityEngine[Query, Candidate <: Serializable]( + versionedStoreMap: Map[String, ReadableStore[Query, Seq[Candidate]]], // key = modelId + override val identifier: SimilarityEngineType, + globalStats: StatsReceiver, + engineConfig: SimilarityEngineConfig, + memCacheConfigOpt: Option[MemCacheConfig[Query]] = None) + extends SimilarityEngine[LookupEngineQuery[Query], Candidate] { + + private val scopedStats = globalStats.scope("similarityEngine", identifier.toString) + + private val underlyingLookupMap = { + memCacheConfigOpt match { + case Some(config) => + versionedStoreMap.map { + case (modelId, store) => + ( + modelId, + SimilarityEngine.addMemCache( + underlyingStore = store, + memCacheConfig = config, + keyPrefix = Some(modelId), + statsReceiver = scopedStats + ) + ) + } + case _ => versionedStoreMap + } + } + + override def getCandidates( + engineQuery: LookupEngineQuery[Query] + ): Future[Option[Seq[Candidate]]] = { + val versionedStore = + underlyingLookupMap + .getOrElse( + engineQuery.lookupKey, + throw new IllegalArgumentException( + s"${this.getClass.getSimpleName} ${identifier.toString}: ModelId ${engineQuery.lookupKey} does not exist" + ) + ) + + SimilarityEngine.getFromFn( + fn = versionedStore.get, + storeQuery = engineQuery.storeQuery, + engineConfig = engineConfig, + params = engineQuery.params, + scopedStats = scopedStats.scope(engineQuery.lookupKey) + ) + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/ModelBasedANNStore.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/ModelBasedANNStore.scala new file mode 100644 index 0000000000..064bb8b1aa --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/ModelBasedANNStore.scala @@ -0,0 +1,136 @@ +package com.twitter.cr_mixer.similarity_engine + +import com.twitter.ann.common.thriftscala.AnnQueryService +import com.twitter.ann.common.thriftscala.Distance +import com.twitter.ann.common.thriftscala.NearestNeighborQuery +import com.twitter.ann.common.thriftscala.NearestNeighborResult +import com.twitter.ann.hnsw.HnswCommon +import com.twitter.ann.hnsw.HnswParams +import com.twitter.bijection.Injection +import com.twitter.conversions.DurationOps._ +import com.twitter.cortex.ml.embeddings.common.TweetKind +import com.twitter.cr_mixer.model.SimilarityEngineInfo +import com.twitter.cr_mixer.model.TweetWithScore +import com.twitter.cr_mixer.thriftscala.SimilarityEngineType +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.frigate.common.util.StatsUtil +import com.twitter.mediaservices.commons.codec.ArrayByteBufferCodec +import com.twitter.ml.api.thriftscala.{Embedding => ThriftEmbedding} +import com.twitter.ml.featurestore.lib +import com.twitter.simclusters_v2.thriftscala.InternalId +import com.twitter.storehaus.ReadableStore +import com.twitter.util.Duration +import com.twitter.util.Future +import javax.inject.Singleton + +/** + * This store looks for tweets whose similarity is close to a Source Dense Embedding. + * Only support Long based embedding lookup. UserId or TweetId + */ +@Singleton +class ModelBasedANNStore( + embeddingStoreLookUpMap: Map[String, ReadableStore[InternalId, ThriftEmbedding]], + annServiceLookUpMap: Map[String, AnnQueryService.MethodPerEndpoint], + globalStats: StatsReceiver) + extends ReadableStore[ + ModelBasedANNStore.Query, + Seq[TweetWithScore] + ] { + + import ModelBasedANNStore._ + + private val stats = globalStats.scope(this.getClass.getSimpleName) + private val fetchEmbeddingStat = stats.scope("fetchEmbedding") + private val fetchCandidatesStat = stats.scope("fetchCandidates") + + override def get(query: Query): Future[Option[Seq[TweetWithScore]]] = { + for { + maybeEmbedding <- StatsUtil.trackOptionStats(fetchEmbeddingStat.scope(query.modelId)) { + fetchEmbedding(query) + } + maybeCandidates <- StatsUtil.trackOptionStats(fetchCandidatesStat.scope(query.modelId)) { + maybeEmbedding match { + case Some(embedding) => + fetchCandidates(query, embedding) + case None => + Future.None + } + } + } yield { + maybeCandidates.map( + _.nearestNeighbors + .map { nearestNeighbor => + val candidateId = TweetIdByteInjection + .invert(ArrayByteBufferCodec.decode(nearestNeighbor.id)) + .toOption + .map(_.tweetId) + (candidateId, nearestNeighbor.distance) + }.collect { + case (Some(candidateId), Some(distance)) => + TweetWithScore(candidateId, toScore(distance)) + }) + } + } + + private def fetchEmbedding(query: Query): Future[Option[ThriftEmbedding]] = { + embeddingStoreLookUpMap.get(query.modelId) match { + case Some(embeddingStore) => + embeddingStore.get(query.sourceId) + case _ => + Future.None + } + } + + private def fetchCandidates( + query: Query, + embedding: ThriftEmbedding + ): Future[Option[NearestNeighborResult]] = { + val hnswParams = HnswCommon.RuntimeParamsInjection.apply(HnswParams(query.ef)) + + annServiceLookUpMap.get(query.modelId) match { + case Some(annService) => + val annQuery = + NearestNeighborQuery(embedding, withDistance = true, hnswParams, MaxNumResults) + annService.query(annQuery).map(v => Some(v)) + case _ => + Future.None + } + } +} + +object ModelBasedANNStore { + + val MaxNumResults: Int = 200 + val MaxTweetCandidateAge: Duration = 1.day + + val TweetIdByteInjection: Injection[lib.TweetId, Array[Byte]] = TweetKind.byteInjection + + // For more information about HNSW algorithm: https://docbird.twitter.biz/ann/hnsw.html + case class Query( + sourceId: InternalId, + modelId: String, + similarityEngineType: SimilarityEngineType, + ef: Int = 800) + + def toScore(distance: Distance): Double = { + distance match { + case Distance.L2Distance(l2Distance) => + // (-Infinite, 0.0] + 0.0 - l2Distance.distance + case Distance.CosineDistance(cosineDistance) => + // [0.0 - 1.0] + 1.0 - cosineDistance.distance + case Distance.InnerProductDistance(innerProductDistance) => + // (-Infinite, Infinite) + 1.0 - innerProductDistance.distance + case _ => + 0.0 + } + } + def toSimilarityEngineInfo(query: Query, score: Double): SimilarityEngineInfo = { + SimilarityEngineInfo( + similarityEngineType = query.similarityEngineType, + modelId = Some(query.modelId), + score = Some(score)) + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/ProducerBasedUnifiedSimilarityEngine.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/ProducerBasedUnifiedSimilarityEngine.scala new file mode 100644 index 0000000000..f782ae0379 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/ProducerBasedUnifiedSimilarityEngine.scala @@ -0,0 +1,641 @@ +package com.twitter.cr_mixer.similarity_engine + +import com.twitter.cr_mixer.model.CandidateGenerationInfo +import com.twitter.cr_mixer.model.ModuleNames +import com.twitter.cr_mixer.model.SimilarityEngineInfo +import com.twitter.cr_mixer.model.SourceInfo +import com.twitter.cr_mixer.model.TweetWithCandidateGenerationInfo +import com.twitter.cr_mixer.model.TweetWithScore +import com.twitter.cr_mixer.param.GlobalParams +import com.twitter.cr_mixer.param.ProducerBasedCandidateGenerationParams +import com.twitter.cr_mixer.param.UnifiedSETweetCombinationMethod +import com.twitter.cr_mixer.param.RelatedTweetProducerBasedParams +import com.twitter.cr_mixer.param.SimClustersANNParams +import com.twitter.cr_mixer.thriftscala.SimilarityEngineType +import com.twitter.cr_mixer.thriftscala.SourceType +import com.twitter.cr_mixer.util.InterleaveUtil +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.frigate.common.util.StatsUtil +import com.twitter.simclusters_v2.common.ModelVersions +import com.twitter.simclusters_v2.thriftscala.EmbeddingType +import com.twitter.simclusters_v2.thriftscala.InternalId +import com.twitter.storehaus.ReadableStore +import com.twitter.timelines.configapi +import com.twitter.util.Duration +import com.twitter.util.Future +import javax.inject.Named +import javax.inject.Singleton +import scala.collection.mutable.ArrayBuffer + +/** + * This store looks for similar tweets from UserTweetGraph for a Source ProducerId + * For a query producerId,User Tweet Graph (UTG), + * lets us find out which tweets the query producer's followers co-engaged + */ +@Singleton +case class ProducerBasedUnifiedSimilarityEngine( + @Named(ModuleNames.ProducerBasedUserTweetGraphSimilarityEngine) + producerBasedUserTweetGraphSimilarityEngine: StandardSimilarityEngine[ + ProducerBasedUserTweetGraphSimilarityEngine.Query, + TweetWithScore + ], + simClustersANNSimilarityEngine: StandardSimilarityEngine[ + SimClustersANNSimilarityEngine.Query, + TweetWithScore + ], + statsReceiver: StatsReceiver) + extends ReadableStore[ProducerBasedUnifiedSimilarityEngine.Query, Seq[ + TweetWithCandidateGenerationInfo + ]] { + + import ProducerBasedUnifiedSimilarityEngine._ + private val stats = statsReceiver.scope(this.getClass.getSimpleName) + private val fetchCandidatesStat = stats.scope("fetchCandidates") + + override def get( + query: Query + ): Future[Option[Seq[TweetWithCandidateGenerationInfo]]] = { + query.sourceInfo.internalId match { + case _: InternalId.UserId => + StatsUtil.trackOptionItemsStats(fetchCandidatesStat) { + val sannCandidatesFut = if (query.enableSimClustersANN) { + simClustersANNSimilarityEngine.getCandidates(query.simClustersANNQuery) + } else Future.None + + val sann1CandidatesFut = + if (query.enableSimClustersANN1) { + simClustersANNSimilarityEngine.getCandidates(query.simClustersANN1Query) + } else Future.None + + val sann2CandidatesFut = + if (query.enableSimClustersANN2) { + simClustersANNSimilarityEngine.getCandidates(query.simClustersANN2Query) + } else Future.None + + val sann3CandidatesFut = + if (query.enableSimClustersANN3) { + simClustersANNSimilarityEngine.getCandidates(query.simClustersANN3Query) + } else Future.None + + val sann4CandidatesFut = + if (query.enableSimClustersANN4) { + simClustersANNSimilarityEngine.getCandidates(query.simClustersANN4Query) + } else Future.None + + val sann5CandidatesFut = + if (query.enableSimClustersANN5) { + simClustersANNSimilarityEngine.getCandidates(query.simClustersANN5Query) + } else Future.None + + val experimentalSANNCandidatesFut = + if (query.enableExperimentalSimClustersANN) { + simClustersANNSimilarityEngine.getCandidates(query.experimentalSimClustersANNQuery) + } else Future.None + + val utgCandidatesFut = if (query.enableUtg) { + producerBasedUserTweetGraphSimilarityEngine.getCandidates(query.utgQuery) + } else Future.None + + Future + .join( + sannCandidatesFut, + sann1CandidatesFut, + sann2CandidatesFut, + sann3CandidatesFut, + sann4CandidatesFut, + sann5CandidatesFut, + experimentalSANNCandidatesFut, + utgCandidatesFut + ).map { + case ( + simClustersAnnCandidates, + simClustersAnn1Candidates, + simClustersAnn2Candidates, + simClustersAnn3Candidates, + simClustersAnn4Candidates, + simClustersAnn5Candidates, + experimentalSANNCandidates, + userTweetGraphCandidates) => + val filteredSANNTweets = simClustersCandidateMinScoreFilter( + simClustersAnnCandidates.toSeq.flatten, + query.simClustersMinScore, + query.simClustersANNQuery.storeQuery.simClustersANNConfigId) + + val filteredExperimentalSANNTweets = simClustersCandidateMinScoreFilter( + experimentalSANNCandidates.toSeq.flatten, + query.simClustersMinScore, + query.experimentalSimClustersANNQuery.storeQuery.simClustersANNConfigId) + + val filteredSANN1Tweets = simClustersCandidateMinScoreFilter( + simClustersAnn1Candidates.toSeq.flatten, + query.simClustersMinScore, + query.simClustersANN1Query.storeQuery.simClustersANNConfigId) + + val filteredSANN2Tweets = simClustersCandidateMinScoreFilter( + simClustersAnn2Candidates.toSeq.flatten, + query.simClustersMinScore, + query.simClustersANN2Query.storeQuery.simClustersANNConfigId) + + val filteredSANN3Tweets = simClustersCandidateMinScoreFilter( + simClustersAnn3Candidates.toSeq.flatten, + query.simClustersMinScore, + query.simClustersANN3Query.storeQuery.simClustersANNConfigId) + + val filteredSANN4Tweets = simClustersCandidateMinScoreFilter( + simClustersAnn4Candidates.toSeq.flatten, + query.simClustersMinScore, + query.simClustersANN4Query.storeQuery.simClustersANNConfigId) + + val filteredSANN5Tweets = simClustersCandidateMinScoreFilter( + simClustersAnn5Candidates.toSeq.flatten, + query.simClustersMinScore, + query.simClustersANN5Query.storeQuery.simClustersANNConfigId) + + val filteredUTGTweets = + userTweetGraphFilter(userTweetGraphCandidates.toSeq.flatten) + + val sannTweetsWithCGInfo = filteredSANNTweets.map { tweetWithScore => + val similarityEngineInfo = SimClustersANNSimilarityEngine + .toSimilarityEngineInfo(query.simClustersANNQuery, tweetWithScore.score) + TweetWithCandidateGenerationInfo( + tweetWithScore.tweetId, + CandidateGenerationInfo( + Some(query.sourceInfo), + similarityEngineInfo, + Seq(similarityEngineInfo) + )) + } + val sann1TweetsWithCGInfo = filteredSANN1Tweets.map { tweetWithScore => + val similarityEngineInfo = SimClustersANNSimilarityEngine + .toSimilarityEngineInfo(query.simClustersANN1Query, tweetWithScore.score) + TweetWithCandidateGenerationInfo( + tweetWithScore.tweetId, + CandidateGenerationInfo( + Some(query.sourceInfo), + similarityEngineInfo, + Seq(similarityEngineInfo) + )) + } + val sann2TweetsWithCGInfo = filteredSANN2Tweets.map { tweetWithScore => + val similarityEngineInfo = SimClustersANNSimilarityEngine + .toSimilarityEngineInfo(query.simClustersANN2Query, tweetWithScore.score) + TweetWithCandidateGenerationInfo( + tweetWithScore.tweetId, + CandidateGenerationInfo( + Some(query.sourceInfo), + similarityEngineInfo, + Seq(similarityEngineInfo) + )) + } + + val sann3TweetsWithCGInfo = filteredSANN3Tweets.map { tweetWithScore => + val similarityEngineInfo = SimClustersANNSimilarityEngine + .toSimilarityEngineInfo(query.simClustersANN3Query, tweetWithScore.score) + TweetWithCandidateGenerationInfo( + tweetWithScore.tweetId, + CandidateGenerationInfo( + Some(query.sourceInfo), + similarityEngineInfo, + Seq(similarityEngineInfo) + )) + } + + val sann4TweetsWithCGInfo = filteredSANN4Tweets.map { tweetWithScore => + val similarityEngineInfo = SimClustersANNSimilarityEngine + .toSimilarityEngineInfo(query.simClustersANN4Query, tweetWithScore.score) + TweetWithCandidateGenerationInfo( + tweetWithScore.tweetId, + CandidateGenerationInfo( + Some(query.sourceInfo), + similarityEngineInfo, + Seq(similarityEngineInfo) + )) + } + + val sann5TweetsWithCGInfo = filteredSANN5Tweets.map { tweetWithScore => + val similarityEngineInfo = SimClustersANNSimilarityEngine + .toSimilarityEngineInfo(query.simClustersANN5Query, tweetWithScore.score) + TweetWithCandidateGenerationInfo( + tweetWithScore.tweetId, + CandidateGenerationInfo( + Some(query.sourceInfo), + similarityEngineInfo, + Seq(similarityEngineInfo) + )) + } + + val experimentalSANNTweetsWithCGInfo = filteredExperimentalSANNTweets.map { + tweetWithScore => + val similarityEngineInfo = SimClustersANNSimilarityEngine + .toSimilarityEngineInfo( + query.experimentalSimClustersANNQuery, + tweetWithScore.score) + TweetWithCandidateGenerationInfo( + tweetWithScore.tweetId, + CandidateGenerationInfo( + Some(query.sourceInfo), + similarityEngineInfo, + Seq(similarityEngineInfo) + )) + } + val utgTweetsWithCGInfo = filteredUTGTweets.map { tweetWithScore => + val similarityEngineInfo = + ProducerBasedUserTweetGraphSimilarityEngine + .toSimilarityEngineInfo(tweetWithScore.score) + TweetWithCandidateGenerationInfo( + tweetWithScore.tweetId, + CandidateGenerationInfo( + Some(query.sourceInfo), + similarityEngineInfo, + Seq(similarityEngineInfo) + )) + } + + val candidateSourcesToBeInterleaved = + ArrayBuffer[Seq[TweetWithCandidateGenerationInfo]]( + sannTweetsWithCGInfo, + sann1TweetsWithCGInfo, + sann2TweetsWithCGInfo, + sann3TweetsWithCGInfo, + sann4TweetsWithCGInfo, + sann5TweetsWithCGInfo, + experimentalSANNTweetsWithCGInfo, + ) + + if (query.utgCombinationMethod == UnifiedSETweetCombinationMethod.Interleave) { + candidateSourcesToBeInterleaved += utgTweetsWithCGInfo + } + + val interleavedCandidates = + InterleaveUtil.interleave(candidateSourcesToBeInterleaved) + + val candidateSourcesToBeOrdered = + ArrayBuffer[Seq[TweetWithCandidateGenerationInfo]](interleavedCandidates) + + if (query.utgCombinationMethod == UnifiedSETweetCombinationMethod.Frontload) + candidateSourcesToBeOrdered.prepend(utgTweetsWithCGInfo) + + val candidatesFromGivenOrderCombination = + SimilaritySourceOrderingUtil.keepGivenOrder(candidateSourcesToBeOrdered) + + val unifiedCandidatesWithUnifiedCGInfo = candidatesFromGivenOrderCombination.map { + candidate => + /*** + * when a candidate was made by interleave/keepGivenOrder, + * then we apply getProducerBasedUnifiedCGInfo() to override with the unified CGInfo + * + * in contributingSE list for interleave. We only have the chosen SE available. + * This is hard to add for interleave, and we plan to add it later after abstraction improvement. + */ + TweetWithCandidateGenerationInfo( + tweetId = candidate.tweetId, + candidateGenerationInfo = getProducerBasedUnifiedCGInfo( + candidate.candidateGenerationInfo.sourceInfoOpt, + candidate.getSimilarityScore, + candidate.candidateGenerationInfo.contributingSimilarityEngines + ) // getSimilarityScore comes from either unifiedScore or single score + ) + } + stats.stat("unified_candidate_size").add(unifiedCandidatesWithUnifiedCGInfo.size) + val truncatedCandidates = + unifiedCandidatesWithUnifiedCGInfo.take(query.maxCandidateNumPerSourceKey) + stats.stat("truncatedCandidates_size").add(truncatedCandidates.size) + + Some(truncatedCandidates) + + } + } + + case _ => + stats.counter("sourceId_is_not_userId_cnt").incr() + Future.None + } + } + + private def simClustersCandidateMinScoreFilter( + simClustersAnnCandidates: Seq[TweetWithScore], + simClustersMinScore: Double, + simClustersANNConfigId: String + ): Seq[TweetWithScore] = { + val filteredCandidates = simClustersAnnCandidates + .filter { candidate => + candidate.score > simClustersMinScore + } + + stats.stat(simClustersANNConfigId, "simClustersAnnCandidates_size").add(filteredCandidates.size) + stats.counter(simClustersANNConfigId, "simClustersAnnRequests").incr() + if (filteredCandidates.isEmpty) + stats.counter(simClustersANNConfigId, "emptyFilteredSimClustersAnnCandidates").incr() + + filteredCandidates.map { candidate => + TweetWithScore(candidate.tweetId, candidate.score) + } + } + + /** A no-op filter as UTG filter already happened at UTG service side */ + private def userTweetGraphFilter( + userTweetGraphCandidates: Seq[TweetWithScore] + ): Seq[TweetWithScore] = { + val filteredCandidates = userTweetGraphCandidates + + stats.stat("userTweetGraphCandidates_size").add(userTweetGraphCandidates.size) + if (filteredCandidates.isEmpty) stats.counter("emptyFilteredUserTweetGraphCandidates").incr() + + filteredCandidates.map { candidate => + TweetWithScore(candidate.tweetId, candidate.score) + } + } + +} +object ProducerBasedUnifiedSimilarityEngine { + + /*** + * Every candidate will have the CG Info with ProducerBasedUnifiedSimilarityEngine + * as they are generated by a composite of Similarity Engines. + * Additionally, we store the contributing SEs (eg., SANN, UTG). + */ + private def getProducerBasedUnifiedCGInfo( + sourceInfoOpt: Option[SourceInfo], + unifiedScore: Double, + contributingSimilarityEngines: Seq[SimilarityEngineInfo] + ): CandidateGenerationInfo = { + CandidateGenerationInfo( + sourceInfoOpt, + SimilarityEngineInfo( + similarityEngineType = SimilarityEngineType.ProducerBasedUnifiedSimilarityEngine, + modelId = None, // We do not assign modelId for a unified similarity engine + score = Some(unifiedScore) + ), + contributingSimilarityEngines + ) + } + + case class Query( + sourceInfo: SourceInfo, + maxCandidateNumPerSourceKey: Int, + maxTweetAgeHours: Duration, + // SimClusters + enableSimClustersANN: Boolean, + simClustersANNQuery: EngineQuery[SimClustersANNSimilarityEngine.Query], + enableExperimentalSimClustersANN: Boolean, + experimentalSimClustersANNQuery: EngineQuery[SimClustersANNSimilarityEngine.Query], + enableSimClustersANN1: Boolean, + simClustersANN1Query: EngineQuery[SimClustersANNSimilarityEngine.Query], + enableSimClustersANN2: Boolean, + simClustersANN2Query: EngineQuery[SimClustersANNSimilarityEngine.Query], + enableSimClustersANN4: Boolean, + simClustersANN4Query: EngineQuery[SimClustersANNSimilarityEngine.Query], + enableSimClustersANN3: Boolean, + simClustersANN3Query: EngineQuery[SimClustersANNSimilarityEngine.Query], + enableSimClustersANN5: Boolean, + simClustersANN5Query: EngineQuery[SimClustersANNSimilarityEngine.Query], + simClustersMinScore: Double, + // UTG + enableUtg: Boolean, + utgCombinationMethod: UnifiedSETweetCombinationMethod.Value, + utgQuery: EngineQuery[ProducerBasedUserTweetGraphSimilarityEngine.Query]) + + def fromParams( + sourceInfo: SourceInfo, + params: configapi.Params, + ): EngineQuery[Query] = { + val maxCandidateNumPerSourceKey = params(GlobalParams.MaxCandidateNumPerSourceKeyParam) + val maxTweetAgeHours = params(GlobalParams.MaxTweetAgeHoursParam) + // SimClusters + val enableSimClustersANN = params( + ProducerBasedCandidateGenerationParams.EnableSimClustersANNParam) + val simClustersModelVersion = + ModelVersions.Enum.enumToSimClustersModelVersionMap(params(GlobalParams.ModelVersionParam)) + val simClustersANNConfigId = params(SimClustersANNParams.SimClustersANNConfigId) + // SimClusters - Experimental SANN Similarity Engine + val enableExperimentalSimClustersANN = params( + ProducerBasedCandidateGenerationParams.EnableExperimentalSimClustersANNParam) + val experimentalSimClustersANNConfigId = params( + SimClustersANNParams.ExperimentalSimClustersANNConfigId) + // SimClusters - SANN cluster 1 Similarity Engine + val enableSimClustersANN1 = params( + ProducerBasedCandidateGenerationParams.EnableSimClustersANN1Param) + val simClustersANN1ConfigId = params(SimClustersANNParams.SimClustersANN1ConfigId) + // SimClusters - SANN cluster 2 Similarity Engine + val enableSimClustersANN2 = params( + ProducerBasedCandidateGenerationParams.EnableSimClustersANN2Param) + val simClustersANN2ConfigId = params(SimClustersANNParams.SimClustersANN2ConfigId) + // SimClusters - SANN cluster 3 Similarity Engine + val enableSimClustersANN3 = params( + ProducerBasedCandidateGenerationParams.EnableSimClustersANN3Param) + val simClustersANN3ConfigId = params(SimClustersANNParams.SimClustersANN3ConfigId) + // SimClusters - SANN cluster 5 Similarity Engine + val enableSimClustersANN5 = params( + ProducerBasedCandidateGenerationParams.EnableSimClustersANN5Param) + val simClustersANN5ConfigId = params(SimClustersANNParams.SimClustersANN5ConfigId) + val enableSimClustersANN4 = params( + ProducerBasedCandidateGenerationParams.EnableSimClustersANN4Param) + val simClustersANN4ConfigId = params(SimClustersANNParams.SimClustersANN4ConfigId) + + val simClustersMinScore = params( + ProducerBasedCandidateGenerationParams.SimClustersMinScoreParam) + + // SimClusters ANN Query + val simClustersANNQuery = SimClustersANNSimilarityEngine.fromParams( + sourceInfo.internalId, + EmbeddingType.FavBasedProducer, + simClustersModelVersion, + simClustersANNConfigId, + params + ) + val experimentalSimClustersANNQuery = SimClustersANNSimilarityEngine.fromParams( + sourceInfo.internalId, + EmbeddingType.FavBasedProducer, + simClustersModelVersion, + experimentalSimClustersANNConfigId, + params + ) + val simClustersANN1Query = SimClustersANNSimilarityEngine.fromParams( + sourceInfo.internalId, + EmbeddingType.FavBasedProducer, + simClustersModelVersion, + simClustersANN1ConfigId, + params + ) + val simClustersANN2Query = SimClustersANNSimilarityEngine.fromParams( + sourceInfo.internalId, + EmbeddingType.FavBasedProducer, + simClustersModelVersion, + simClustersANN2ConfigId, + params + ) + val simClustersANN3Query = SimClustersANNSimilarityEngine.fromParams( + sourceInfo.internalId, + EmbeddingType.FavBasedProducer, + simClustersModelVersion, + simClustersANN3ConfigId, + params + ) + val simClustersANN5Query = SimClustersANNSimilarityEngine.fromParams( + sourceInfo.internalId, + EmbeddingType.FavBasedProducer, + simClustersModelVersion, + simClustersANN5ConfigId, + params + ) + val simClustersANN4Query = SimClustersANNSimilarityEngine.fromParams( + sourceInfo.internalId, + EmbeddingType.FavBasedProducer, + simClustersModelVersion, + simClustersANN4ConfigId, + params + ) + // UTG + val enableUtg = params(ProducerBasedCandidateGenerationParams.EnableUTGParam) + val utgCombinationMethod = params( + ProducerBasedCandidateGenerationParams.UtgCombinationMethodParam) + + EngineQuery( + Query( + sourceInfo = sourceInfo, + maxCandidateNumPerSourceKey = maxCandidateNumPerSourceKey, + maxTweetAgeHours = maxTweetAgeHours, + enableSimClustersANN = enableSimClustersANN, + simClustersANNQuery = simClustersANNQuery, + enableExperimentalSimClustersANN = enableExperimentalSimClustersANN, + experimentalSimClustersANNQuery = experimentalSimClustersANNQuery, + enableSimClustersANN1 = enableSimClustersANN1, + simClustersANN1Query = simClustersANN1Query, + enableSimClustersANN2 = enableSimClustersANN2, + simClustersANN2Query = simClustersANN2Query, + enableSimClustersANN3 = enableSimClustersANN3, + simClustersANN3Query = simClustersANN3Query, + enableSimClustersANN5 = enableSimClustersANN5, + simClustersANN5Query = simClustersANN5Query, + enableSimClustersANN4 = enableSimClustersANN4, + simClustersANN4Query = simClustersANN4Query, + simClustersMinScore = simClustersMinScore, + enableUtg = enableUtg, + utgCombinationMethod = utgCombinationMethod, + utgQuery = ProducerBasedUserTweetGraphSimilarityEngine + .fromParams(sourceInfo.internalId, params) + ), + params + ) + } + + def fromParamsForRelatedTweet( + internalId: InternalId, + params: configapi.Params + ): EngineQuery[Query] = { + val maxCandidateNumPerSourceKey = params(GlobalParams.MaxCandidateNumPerSourceKeyParam) + val maxTweetAgeHours = params(GlobalParams.MaxTweetAgeHoursParam) + // SimClusters + val enableSimClustersANN = params(RelatedTweetProducerBasedParams.EnableSimClustersANNParam) + val simClustersModelVersion = + ModelVersions.Enum.enumToSimClustersModelVersionMap(params(GlobalParams.ModelVersionParam)) + val simClustersANNConfigId = params(SimClustersANNParams.SimClustersANNConfigId) + val simClustersMinScore = + params(RelatedTweetProducerBasedParams.SimClustersMinScoreParam) + // SimClusters - Experimental SANN Similarity Engine + val enableExperimentalSimClustersANN = params( + RelatedTweetProducerBasedParams.EnableExperimentalSimClustersANNParam) + val experimentalSimClustersANNConfigId = params( + SimClustersANNParams.ExperimentalSimClustersANNConfigId) + // SimClusters - SANN cluster 1 Similarity Engine + val enableSimClustersANN1 = params(RelatedTweetProducerBasedParams.EnableSimClustersANN1Param) + val simClustersANN1ConfigId = params(SimClustersANNParams.SimClustersANN1ConfigId) + // SimClusters - SANN cluster 2 Similarity Engine + val enableSimClustersANN2 = params(RelatedTweetProducerBasedParams.EnableSimClustersANN2Param) + val simClustersANN2ConfigId = params(SimClustersANNParams.SimClustersANN2ConfigId) + // SimClusters - SANN cluster 3 Similarity Engine + val enableSimClustersANN3 = params(RelatedTweetProducerBasedParams.EnableSimClustersANN3Param) + val simClustersANN3ConfigId = params(SimClustersANNParams.SimClustersANN3ConfigId) + // SimClusters - SANN cluster 5 Similarity Engine + val enableSimClustersANN5 = params(RelatedTweetProducerBasedParams.EnableSimClustersANN5Param) + val simClustersANN5ConfigId = params(SimClustersANNParams.SimClustersANN5ConfigId) + + val enableSimClustersANN4 = params(RelatedTweetProducerBasedParams.EnableSimClustersANN4Param) + val simClustersANN4ConfigId = params(SimClustersANNParams.SimClustersANN4ConfigId) + // Build SANN Query + val simClustersANNQuery = SimClustersANNSimilarityEngine.fromParams( + internalId, + EmbeddingType.FavBasedProducer, + simClustersModelVersion, + simClustersANNConfigId, + params + ) + val experimentalSimClustersANNQuery = SimClustersANNSimilarityEngine.fromParams( + internalId, + EmbeddingType.FavBasedProducer, + simClustersModelVersion, + experimentalSimClustersANNConfigId, + params + ) + val simClustersANN1Query = SimClustersANNSimilarityEngine.fromParams( + internalId, + EmbeddingType.FavBasedProducer, + simClustersModelVersion, + simClustersANN1ConfigId, + params + ) + val simClustersANN2Query = SimClustersANNSimilarityEngine.fromParams( + internalId, + EmbeddingType.FavBasedProducer, + simClustersModelVersion, + simClustersANN2ConfigId, + params + ) + val simClustersANN3Query = SimClustersANNSimilarityEngine.fromParams( + internalId, + EmbeddingType.FavBasedProducer, + simClustersModelVersion, + simClustersANN3ConfigId, + params + ) + val simClustersANN5Query = SimClustersANNSimilarityEngine.fromParams( + internalId, + EmbeddingType.FavBasedProducer, + simClustersModelVersion, + simClustersANN5ConfigId, + params + ) + val simClustersANN4Query = SimClustersANNSimilarityEngine.fromParams( + internalId, + EmbeddingType.FavBasedProducer, + simClustersModelVersion, + simClustersANN4ConfigId, + params + ) + // UTG + val enableUtg = params(RelatedTweetProducerBasedParams.EnableUTGParam) + val utgCombinationMethod = params( + ProducerBasedCandidateGenerationParams.UtgCombinationMethodParam) + + // SourceType.RequestUserId is a placeholder. + val sourceInfo = SourceInfo(SourceType.RequestUserId, internalId, None) + + EngineQuery( + Query( + sourceInfo = sourceInfo, + maxCandidateNumPerSourceKey = maxCandidateNumPerSourceKey, + maxTweetAgeHours = maxTweetAgeHours, + enableSimClustersANN = enableSimClustersANN, + simClustersANNQuery = simClustersANNQuery, + enableExperimentalSimClustersANN = enableExperimentalSimClustersANN, + experimentalSimClustersANNQuery = experimentalSimClustersANNQuery, + enableSimClustersANN1 = enableSimClustersANN1, + simClustersANN1Query = simClustersANN1Query, + enableSimClustersANN2 = enableSimClustersANN2, + simClustersANN2Query = simClustersANN2Query, + enableSimClustersANN3 = enableSimClustersANN3, + simClustersANN3Query = simClustersANN3Query, + enableSimClustersANN5 = enableSimClustersANN5, + simClustersANN5Query = simClustersANN5Query, + enableSimClustersANN4 = enableSimClustersANN4, + simClustersANN4Query = simClustersANN4Query, + simClustersMinScore = simClustersMinScore, + enableUtg = enableUtg, + utgQuery = ProducerBasedUserTweetGraphSimilarityEngine.fromParams(internalId, params), + utgCombinationMethod = utgCombinationMethod + ), + params + ) + } + +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/ProducerBasedUserAdGraphSimilarityEngine.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/ProducerBasedUserAdGraphSimilarityEngine.scala new file mode 100644 index 0000000000..c9ebc91e70 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/ProducerBasedUserAdGraphSimilarityEngine.scala @@ -0,0 +1,96 @@ +package com.twitter.cr_mixer.similarity_engine + +import com.twitter.cr_mixer.model.SimilarityEngineInfo +import com.twitter.cr_mixer.model.TweetWithScore +import com.twitter.cr_mixer.param.GlobalParams +import com.twitter.cr_mixer.param.ProducerBasedUserAdGraphParams +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.recos.user_ad_graph.thriftscala.ProducerBasedRelatedAdRequest +import com.twitter.recos.user_ad_graph.thriftscala.UserAdGraph +import com.twitter.simclusters_v2.thriftscala.InternalId +import com.twitter.storehaus.ReadableStore +import com.twitter.util.Future +import javax.inject.Singleton +import com.twitter.cr_mixer.param.GlobalParams +import com.twitter.cr_mixer.thriftscala.SimilarityEngineType +import com.twitter.frigate.common.util.StatsUtil +import com.twitter.timelines.configapi + +/** + * This store looks for similar tweets from UserAdGraph for a Source ProducerId + * For a query producerId,User Tweet Graph (UAG), + * lets us find out which ad tweets the query producer's followers co-engaged + */ +@Singleton +case class ProducerBasedUserAdGraphSimilarityEngine( + userAdGraphService: UserAdGraph.MethodPerEndpoint, + statsReceiver: StatsReceiver) + extends ReadableStore[ProducerBasedUserAdGraphSimilarityEngine.Query, Seq[ + TweetWithScore + ]] { + + private val stats = statsReceiver.scope(this.getClass.getSimpleName) + private val fetchCandidatesStat = stats.scope("fetchCandidates") + + override def get( + query: ProducerBasedUserAdGraphSimilarityEngine.Query + ): Future[Option[Seq[TweetWithScore]]] = { + query.sourceId match { + case InternalId.UserId(producerId) => + StatsUtil.trackOptionItemsStats(fetchCandidatesStat) { + val relatedAdRequest = + ProducerBasedRelatedAdRequest( + producerId, + maxResults = Some(query.maxResults), + minCooccurrence = Some(query.minCooccurrence), + minScore = Some(query.minScore), + maxNumFollowers = Some(query.maxNumFollowers), + maxTweetAgeInHours = Some(query.maxTweetAgeInHours), + ) + + userAdGraphService.producerBasedRelatedAds(relatedAdRequest).map { relatedAdResponse => + val candidates = + relatedAdResponse.adTweets.map(tweet => TweetWithScore(tweet.adTweetId, tweet.score)) + Some(candidates) + } + } + case _ => + Future.value(None) + } + } +} + +object ProducerBasedUserAdGraphSimilarityEngine { + + def toSimilarityEngineInfo(score: Double): SimilarityEngineInfo = { + SimilarityEngineInfo( + similarityEngineType = SimilarityEngineType.ProducerBasedUserAdGraph, + modelId = None, + score = Some(score)) + } + + case class Query( + sourceId: InternalId, + maxResults: Int, + minCooccurrence: Int, // require at least {minCooccurrence} lhs user engaged with returned tweet + minScore: Double, + maxNumFollowers: Int, // max number of lhs users + maxTweetAgeInHours: Int) + + def fromParams( + sourceId: InternalId, + params: configapi.Params, + ): EngineQuery[Query] = { + EngineQuery( + Query( + sourceId = sourceId, + maxResults = params(GlobalParams.MaxCandidateNumPerSourceKeyParam), + minCooccurrence = params(ProducerBasedUserAdGraphParams.MinCoOccurrenceParam), + maxNumFollowers = params(ProducerBasedUserAdGraphParams.MaxNumFollowersParam), + maxTweetAgeInHours = params(GlobalParams.MaxTweetAgeHoursParam).inHours, + minScore = params(ProducerBasedUserAdGraphParams.MinScoreParam) + ), + params + ) + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/ProducerBasedUserTweetGraphSimilarityEngine.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/ProducerBasedUserTweetGraphSimilarityEngine.scala new file mode 100644 index 0000000000..6e7ca95bd0 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/ProducerBasedUserTweetGraphSimilarityEngine.scala @@ -0,0 +1,96 @@ +package com.twitter.cr_mixer.similarity_engine + +import com.twitter.cr_mixer.model.SimilarityEngineInfo +import com.twitter.cr_mixer.model.TweetWithScore +import com.twitter.cr_mixer.param.ProducerBasedUserTweetGraphParams +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.recos.user_tweet_graph.thriftscala.ProducerBasedRelatedTweetRequest +import com.twitter.simclusters_v2.thriftscala.InternalId +import com.twitter.storehaus.ReadableStore +import com.twitter.util.Future +import javax.inject.Singleton +import com.twitter.cr_mixer.param.GlobalParams +import com.twitter.cr_mixer.thriftscala.SimilarityEngineType +import com.twitter.frigate.common.util.StatsUtil +import com.twitter.timelines.configapi +import com.twitter.recos.user_tweet_graph.thriftscala.UserTweetGraph + +/** + * This store looks for similar tweets from UserTweetGraph for a Source ProducerId + * For a query producerId,User Tweet Graph (UTG), + * lets us find out which tweets the query producer's followers co-engaged + */ +@Singleton +case class ProducerBasedUserTweetGraphSimilarityEngine( + userTweetGraphService: UserTweetGraph.MethodPerEndpoint, + statsReceiver: StatsReceiver) + extends ReadableStore[ProducerBasedUserTweetGraphSimilarityEngine.Query, Seq[ + TweetWithScore + ]] { + + private val stats = statsReceiver.scope(this.getClass.getSimpleName) + private val fetchCandidatesStat = stats.scope("fetchCandidates") + + override def get( + query: ProducerBasedUserTweetGraphSimilarityEngine.Query + ): Future[Option[Seq[TweetWithScore]]] = { + query.sourceId match { + case InternalId.UserId(producerId) => + StatsUtil.trackOptionItemsStats(fetchCandidatesStat) { + val relatedTweetRequest = + ProducerBasedRelatedTweetRequest( + producerId, + maxResults = Some(query.maxResults), + minCooccurrence = Some(query.minCooccurrence), + minScore = Some(query.minScore), + maxNumFollowers = Some(query.maxNumFollowers), + maxTweetAgeInHours = Some(query.maxTweetAgeInHours), + ) + + userTweetGraphService.producerBasedRelatedTweets(relatedTweetRequest).map { + relatedTweetResponse => + val candidates = + relatedTweetResponse.tweets.map(tweet => TweetWithScore(tweet.tweetId, tweet.score)) + Some(candidates) + } + } + case _ => + Future.value(None) + } + } +} + +object ProducerBasedUserTweetGraphSimilarityEngine { + + def toSimilarityEngineInfo(score: Double): SimilarityEngineInfo = { + SimilarityEngineInfo( + similarityEngineType = SimilarityEngineType.ProducerBasedUserTweetGraph, + modelId = None, + score = Some(score)) + } + + case class Query( + sourceId: InternalId, + maxResults: Int, + minCooccurrence: Int, // require at least {minCooccurrence} lhs user engaged with returned tweet + minScore: Double, + maxNumFollowers: Int, // max number of lhs users + maxTweetAgeInHours: Int) + + def fromParams( + sourceId: InternalId, + params: configapi.Params, + ): EngineQuery[Query] = { + EngineQuery( + Query( + sourceId = sourceId, + maxResults = params(GlobalParams.MaxCandidateNumPerSourceKeyParam), + minCooccurrence = params(ProducerBasedUserTweetGraphParams.MinCoOccurrenceParam), + maxNumFollowers = params(ProducerBasedUserTweetGraphParams.MaxNumFollowersParam), + maxTweetAgeInHours = params(GlobalParams.MaxTweetAgeHoursParam).inHours, + minScore = params(ProducerBasedUserTweetGraphParams.MinScoreParam) + ), + params + ) + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/SimClustersANNSimilarityEngine.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/SimClustersANNSimilarityEngine.scala new file mode 100644 index 0000000000..228627c870 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/SimClustersANNSimilarityEngine.scala @@ -0,0 +1,113 @@ +package com.twitter.cr_mixer.similarity_engine + +import com.twitter.cr_mixer.config.SimClustersANNConfig +import com.twitter.cr_mixer.model.SimilarityEngineInfo +import com.twitter.cr_mixer.model.TweetWithScore +import com.twitter.cr_mixer.thriftscala.SimilarityEngineType +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.frigate.common.util.StatsUtil +import com.twitter.simclusters_v2.thriftscala.EmbeddingType +import com.twitter.simclusters_v2.thriftscala.InternalId +import com.twitter.simclusters_v2.thriftscala.ModelVersion +import com.twitter.simclusters_v2.thriftscala.SimClustersEmbeddingId +import com.twitter.simclustersann.thriftscala.SimClustersANNService +import com.twitter.simclustersann.thriftscala.{Query => SimClustersANNQuery} +import com.twitter.storehaus.ReadableStore +import com.twitter.timelines.configapi +import com.twitter.util.Future +import javax.inject.Singleton +import com.twitter.cr_mixer.exception.InvalidSANNConfigException +import com.twitter.relevance_platform.simclustersann.multicluster.ServiceNameMapper + +@Singleton +case class SimClustersANNSimilarityEngine( + simClustersANNServiceNameToClientMapper: Map[String, SimClustersANNService.MethodPerEndpoint], + statsReceiver: StatsReceiver) + extends ReadableStore[ + SimClustersANNSimilarityEngine.Query, + Seq[TweetWithScore] + ] { + + private val name: String = this.getClass.getSimpleName + private val stats = statsReceiver.scope(name) + private val fetchCandidatesStat = stats.scope("fetchCandidates") + + private def getSimClustersANNService( + query: SimClustersANNQuery + ): Option[SimClustersANNService.MethodPerEndpoint] = { + ServiceNameMapper + .getServiceName( + query.sourceEmbeddingId.modelVersion, + query.config.candidateEmbeddingType).flatMap(serviceName => + simClustersANNServiceNameToClientMapper.get(serviceName)) + } + + override def get( + query: SimClustersANNSimilarityEngine.Query + ): Future[Option[Seq[TweetWithScore]]] = { + StatsUtil.trackOptionItemsStats(fetchCandidatesStat) { + + getSimClustersANNService(query.simClustersANNQuery) match { + case Some(simClustersANNService) => + simClustersANNService.getTweetCandidates(query.simClustersANNQuery).map { + simClustersANNTweetCandidates => + val tweetWithScores = simClustersANNTweetCandidates.map { candidate => + TweetWithScore(candidate.tweetId, candidate.score) + } + Some(tweetWithScores) + } + case None => + throw InvalidSANNConfigException( + "No SANN Cluster configured to serve this query, check CandidateEmbeddingType and ModelVersion") + } + } + } +} + +object SimClustersANNSimilarityEngine { + case class Query( + simClustersANNQuery: SimClustersANNQuery, + simClustersANNConfigId: String) + + def toSimilarityEngineInfo( + query: EngineQuery[Query], + score: Double + ): SimilarityEngineInfo = { + SimilarityEngineInfo( + similarityEngineType = SimilarityEngineType.SimClustersANN, + modelId = Some( + s"SimClustersANN_${query.storeQuery.simClustersANNQuery.sourceEmbeddingId.embeddingType.toString}_" + + s"${query.storeQuery.simClustersANNQuery.sourceEmbeddingId.modelVersion.toString}_" + + s"${query.storeQuery.simClustersANNConfigId}"), + score = Some(score) + ) + } + + def fromParams( + internalId: InternalId, + embeddingType: EmbeddingType, + modelVersion: ModelVersion, + simClustersANNConfigId: String, + params: configapi.Params, + ): EngineQuery[Query] = { + + // SimClusters EmbeddingId and ANNConfig + val simClustersEmbeddingId = + SimClustersEmbeddingId(embeddingType, modelVersion, internalId) + val simClustersANNConfig = + SimClustersANNConfig + .getConfig(embeddingType.toString, modelVersion.toString, simClustersANNConfigId) + + EngineQuery( + Query( + SimClustersANNQuery( + sourceEmbeddingId = simClustersEmbeddingId, + config = simClustersANNConfig.toSANNConfigThrift + ), + simClustersANNConfigId + ), + params + ) + } + +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/SimilarityEngine.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/SimilarityEngine.scala new file mode 100644 index 0000000000..6bc332f75e --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/SimilarityEngine.scala @@ -0,0 +1,169 @@ +package com.twitter.cr_mixer.similarity_engine + +import com.twitter.cr_mixer.param.decider.CrMixerDecider +import com.twitter.cr_mixer.thriftscala.SimilarityEngineType +import com.twitter.finagle.GlobalRequestTimeoutException +import com.twitter.finagle.mux.ClientDiscardedRequestException +import com.twitter.finagle.memcached.Client +import com.twitter.finagle.mux.ServerApplicationError +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.frigate.common.util.StatsUtil +import com.twitter.hashing.KeyHasher +import com.twitter.hermit.store.common.ObservedMemcachedReadableStore +import com.twitter.relevance_platform.common.injection.LZ4Injection +import com.twitter.relevance_platform.common.injection.SeqObjectInjection +import com.twitter.storehaus.ReadableStore +import com.twitter.timelines.configapi.FSParam +import com.twitter.timelines.configapi.Params +import com.twitter.util.Duration +import com.twitter.util.Future +import com.twitter.util.TimeoutException +import com.twitter.util.logging.Logging +import org.apache.thrift.TApplicationException + +/** + * A SimilarityEngine is a wrapper which, given a [[Query]], returns a list of [[Candidate]] + * The main purposes of a SimilarityEngine is to provide a consistent interface for candidate + * generation logic, and provides default functions, including: + * - Identification + * - Observability + * - Timeout settings + * - Exception Handling + * - Gating by Deciders & FeatureSwitch settings + * - (coming soon): Dark traffic + * + * Note: + * A SimilarityEngine by itself is NOT meant to be cacheable. + * Caching should be implemented in the underlying ReadableStore that provides the [[Candidate]]s + * + * Please keep extension of this class local this directory only + * + */ +trait SimilarityEngine[Query, Candidate] { + + /** + * Uniquely identifies a similarity engine. + * Avoid using the same engine type for more than one engine, it will cause stats to double count + */ + private[similarity_engine] def identifier: SimilarityEngineType + + def getCandidates(query: Query): Future[Option[Seq[Candidate]]] + +} + +object SimilarityEngine extends Logging { + case class SimilarityEngineConfig( + timeout: Duration, + gatingConfig: GatingConfig) + + /** + * Controls for whether or not this Engine is enabled. + * In our previous design, we were expecting a Sim Engine will only take one set of Params, + * and that’s why we decided to have GatingConfig and the EnableFeatureSwitch in the trait. + * However, we now have two candidate generation pipelines: Tweet Rec, Related Tweets + * and they are now having their own set of Params, but EnableFeatureSwitch can only put in 1 fixed value. + * We need some further refactor work to make it more flexible. + * + * @param deciderConfig Gate the Engine by a decider. If specified, + * @param enableFeatureSwitch. DO NOT USE IT FOR NOW. It needs some refactorting. Please set it to None (SD-20268) + */ + case class GatingConfig( + deciderConfig: Option[DeciderConfig], + enableFeatureSwitch: Option[ + FSParam[Boolean] + ]) // Do NOT use the enableFeatureSwitch. It needs some refactoring. + + case class DeciderConfig( + decider: CrMixerDecider, + deciderString: String) + + case class MemCacheConfig[K]( + cacheClient: Client, + ttl: Duration, + asyncUpdate: Boolean = false, + keyToString: K => String) + + private[similarity_engine] def isEnabled( + params: Params, + gatingConfig: GatingConfig + ): Boolean = { + val enabledByDecider = + gatingConfig.deciderConfig.forall { config => + config.decider.isAvailable(config.deciderString) + } + + val enabledByFS = gatingConfig.enableFeatureSwitch.forall(params.apply) + + enabledByDecider && enabledByFS + } + + // Default key hasher for memcache keys + val keyHasher: KeyHasher = KeyHasher.FNV1A_64 + + /** + * Add a MemCache wrapper to a ReadableStore with a preset key and value injection functions + * Note: The [[Query]] object needs to be cacheable, + * i.e. it cannot be a runtime objects or complex objects, for example, configapi.Params + * + * @param underlyingStore un-cached store implementation + * @param keyPrefix a prefix differentiates 2 stores if they share the same key space. + * e.x. 2 implementations of ReadableStore[UserId, Seq[Candidiate] ] + * can use prefix "store_v1", "store_v2" + * @return A ReadableStore with a MemCache wrapper + */ + private[similarity_engine] def addMemCache[Query, Candidate <: Serializable]( + underlyingStore: ReadableStore[Query, Seq[Candidate]], + memCacheConfig: MemCacheConfig[Query], + keyPrefix: Option[String] = None, + statsReceiver: StatsReceiver + ): ReadableStore[Query, Seq[Candidate]] = { + val prefix = keyPrefix.getOrElse("") + + ObservedMemcachedReadableStore.fromCacheClient[Query, Seq[Candidate]]( + backingStore = underlyingStore, + cacheClient = memCacheConfig.cacheClient, + ttl = memCacheConfig.ttl, + asyncUpdate = memCacheConfig.asyncUpdate, + )( + valueInjection = LZ4Injection.compose(SeqObjectInjection[Candidate]()), + keyToString = { k: Query => s"CRMixer:$prefix${memCacheConfig.keyToString(k)}" }, + statsReceiver = statsReceiver + ) + } + + private val timer = com.twitter.finagle.util.DefaultTimer + + /** + * Applies runtime configs, like stats, timeouts, exception handling, onto fn + */ + private[similarity_engine] def getFromFn[Query, Candidate]( + fn: Query => Future[Option[Seq[Candidate]]], + storeQuery: Query, + engineConfig: SimilarityEngineConfig, + params: Params, + scopedStats: StatsReceiver + ): Future[Option[Seq[Candidate]]] = { + if (isEnabled(params, engineConfig.gatingConfig)) { + scopedStats.counter("gate_enabled").incr() + + StatsUtil + .trackOptionItemsStats(scopedStats) { + fn.apply(storeQuery).raiseWithin(engineConfig.timeout)(timer) + } + .rescue { + case _: TimeoutException | _: GlobalRequestTimeoutException | _: TApplicationException | + _: ClientDiscardedRequestException | + _: ServerApplicationError // TApplicationException inside + => + debug("Failed to fetch. request aborted or timed out") + Future.None + case e => + error("Failed to fetch. request aborted or timed out", e) + Future.None + } + } else { + scopedStats.counter("gate_disabled").incr() + Future.None + } + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/SimilaritySourceOrderingUtil.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/SimilaritySourceOrderingUtil.scala new file mode 100644 index 0000000000..b3da2b631b --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/SimilaritySourceOrderingUtil.scala @@ -0,0 +1,32 @@ +package com.twitter.cr_mixer.similarity_engine + +import com.twitter.cr_mixer.model.TweetWithCandidateGenerationInfo +import com.twitter.simclusters_v2.common.TweetId +import scala.collection.mutable +import scala.collection.mutable.ArrayBuffer + +object SimilaritySourceOrderingUtil { + /** + * This function flatten and dedup input candidates according to the order in the input Seq + * [[candidate10, candidate11], [candidate20, candidate21]] => [candidate10, candidate11, candidate20, candidate21] + */ + def keepGivenOrder( + candidates: Seq[Seq[TweetWithCandidateGenerationInfo]], + ): Seq[TweetWithCandidateGenerationInfo] = { + + val seen = mutable.Set[TweetId]() + val combinedCandidates = candidates.flatten + val result = ArrayBuffer[TweetWithCandidateGenerationInfo]() + + combinedCandidates.foreach { candidate => + val candidateTweetId = candidate.tweetId + val seenCandidate = seen.contains(candidateTweetId) // de-dup + if (!seenCandidate) { + result += candidate + seen.add(candidate.tweetId) + } + } + //convert result to immutable seq + result.toList + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/SkitHighPrecisionTopicTweetSimilarityEngine.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/SkitHighPrecisionTopicTweetSimilarityEngine.scala new file mode 100644 index 0000000000..37701e79e3 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/SkitHighPrecisionTopicTweetSimilarityEngine.scala @@ -0,0 +1,123 @@ +package com.twitter.cr_mixer.similarity_engine + +import com.google.inject.Inject +import com.google.inject.Singleton +import com.google.inject.name.Named +import com.twitter.contentrecommender.thriftscala.AlgorithmType +import com.twitter.cr_mixer.model.ModuleNames +import com.twitter.cr_mixer.model.TopicTweetWithScore +import com.twitter.cr_mixer.param.TopicTweetParams +import com.twitter.cr_mixer.similarity_engine.SkitTopicTweetSimilarityEngine._ +import com.twitter.cr_mixer.thriftscala.SimilarityEngineType +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.frigate.common.util.StatsUtil +import com.twitter.simclusters_v2.thriftscala.EmbeddingType +import com.twitter.simclusters_v2.thriftscala.TopicId +import com.twitter.storehaus.ReadableStore +import com.twitter.timelines.configapi +import com.twitter.topic_recos.thriftscala.TopicTweet +import com.twitter.topic_recos.thriftscala.TopicTweetPartitionFlatKey +import com.twitter.util.Future + +@Singleton +case class SkitHighPrecisionTopicTweetSimilarityEngine @Inject() ( + @Named(ModuleNames.SkitStratoStoreName) skitStratoStore: ReadableStore[ + TopicTweetPartitionFlatKey, + Seq[TopicTweet] + ], + statsReceiver: StatsReceiver) + extends ReadableStore[EngineQuery[Query], Seq[TopicTweetWithScore]] { + + private val name: String = this.getClass.getSimpleName + private val stats = statsReceiver.scope(name) + + override def get(query: EngineQuery[Query]): Future[Option[Seq[TopicTweetWithScore]]] = { + StatsUtil.trackOptionItemsStats(stats) { + fetch(query).map { tweets => + val topTweets = + tweets + .sortBy(-_.favCount) + .take(query.storeQuery.maxCandidates) + .map { tweet => + TopicTweetWithScore( + tweetId = tweet.tweetId, + score = tweet.favCount, + similarityEngineType = SimilarityEngineType.SkitHighPrecisionTopicTweet + ) + } + Some(topTweets) + } + } + } + + private def fetch(query: EngineQuery[Query]): Future[Seq[SkitTopicTweet]] = { + val latestTweetTimeInHour = System.currentTimeMillis() / 1000 / 60 / 60 + + val earliestTweetTimeInHour = latestTweetTimeInHour - + math.min(MaxTweetAgeInHours, query.storeQuery.maxTweetAge.inHours) + val timedKeys = for (timePartition <- earliestTweetTimeInHour to latestTweetTimeInHour) yield { + + TopicTweetPartitionFlatKey( + entityId = query.storeQuery.topicId.entityId, + timePartition = timePartition, + algorithmType = Some(AlgorithmType.SemanticCoreTweet), + tweetEmbeddingType = Some(EmbeddingType.LogFavBasedTweet), + language = query.storeQuery.topicId.language.getOrElse("").toLowerCase, + country = None, // Disable country. It is not used. + semanticCoreAnnotationVersionId = Some(query.storeQuery.semanticCoreVersionId) + ) + } + + getTweetsForKeys( + timedKeys, + query.storeQuery.topicId + ) + } + + /** + * Given a set of keys, multiget the underlying Strato store, combine and flatten the results. + */ + private def getTweetsForKeys( + keys: Seq[TopicTweetPartitionFlatKey], + sourceTopic: TopicId + ): Future[Seq[SkitTopicTweet]] = { + Future + .collect { skitStratoStore.multiGet(keys.toSet).values.toSeq } + .map { combinedResults => + val topTweets = combinedResults.flatten.flatten + topTweets.map { tweet => + SkitTopicTweet( + tweetId = tweet.tweetId, + favCount = tweet.scores.favCount.getOrElse(0L), + cosineSimilarityScore = tweet.scores.cosineSimilarity.getOrElse(0.0), + sourceTopic = sourceTopic + ) + } + } + } +} + +object SkitHighPrecisionTopicTweetSimilarityEngine { + + def fromParams( + topicId: TopicId, + isVideoOnly: Boolean, + params: configapi.Params, + ): EngineQuery[Query] = { + val maxCandidates = if (isVideoOnly) { + params(TopicTweetParams.MaxSkitHighPrecisionCandidatesParam) * 2 + } else { + params(TopicTweetParams.MaxSkitHighPrecisionCandidatesParam) + } + + EngineQuery( + Query( + topicId = topicId, + maxCandidates = maxCandidates, + maxTweetAge = params(TopicTweetParams.MaxTweetAge), + semanticCoreVersionId = params(TopicTweetParams.SemanticCoreVersionIdParam) + ), + params + ) + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/SkitTopicTweetSimilarityEngine.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/SkitTopicTweetSimilarityEngine.scala new file mode 100644 index 0000000000..44bb4b3199 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/SkitTopicTweetSimilarityEngine.scala @@ -0,0 +1,143 @@ +package com.twitter.cr_mixer.similarity_engine + +import com.google.inject.Inject +import com.google.inject.Singleton +import com.google.inject.name.Named +import com.twitter.contentrecommender.thriftscala.AlgorithmType +import com.twitter.conversions.DurationOps._ +import com.twitter.cr_mixer.model.ModuleNames +import com.twitter.cr_mixer.model.TopicTweetWithScore +import com.twitter.cr_mixer.param.TopicTweetParams +import com.twitter.cr_mixer.similarity_engine.SkitTopicTweetSimilarityEngine._ +import com.twitter.cr_mixer.thriftscala.SimilarityEngineType +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.frigate.common.util.StatsUtil +import com.twitter.simclusters_v2.common.TweetId +import com.twitter.simclusters_v2.thriftscala.EmbeddingType +import com.twitter.simclusters_v2.thriftscala.ModelVersion +import com.twitter.simclusters_v2.thriftscala.TopicId +import com.twitter.storehaus.ReadableStore +import com.twitter.timelines.configapi +import com.twitter.topic_recos.thriftscala.TopicTweet +import com.twitter.topic_recos.thriftscala.TopicTweetPartitionFlatKey +import com.twitter.util.Duration +import com.twitter.util.Future + +@Singleton +case class SkitTopicTweetSimilarityEngine @Inject() ( + @Named(ModuleNames.SkitStratoStoreName) skitStratoStore: ReadableStore[ + TopicTweetPartitionFlatKey, + Seq[TopicTweet] + ], + statsReceiver: StatsReceiver) + extends ReadableStore[EngineQuery[Query], Seq[TopicTweetWithScore]] { + + private val name: String = this.getClass.getSimpleName + private val stats = statsReceiver.scope(name) + + override def get(query: EngineQuery[Query]): Future[Option[Seq[TopicTweetWithScore]]] = { + StatsUtil.trackOptionItemsStats(stats) { + fetch(query).map { tweets => + val topTweets = + tweets + .sortBy(-_.cosineSimilarityScore) + .take(query.storeQuery.maxCandidates) + .map { tweet => + TopicTweetWithScore( + tweetId = tweet.tweetId, + score = tweet.cosineSimilarityScore, + similarityEngineType = SimilarityEngineType.SkitTfgTopicTweet + ) + } + Some(topTweets) + } + } + } + + private def fetch(query: EngineQuery[Query]): Future[Seq[SkitTopicTweet]] = { + val latestTweetTimeInHour = System.currentTimeMillis() / 1000 / 60 / 60 + + val earliestTweetTimeInHour = latestTweetTimeInHour - + math.min(MaxTweetAgeInHours, query.storeQuery.maxTweetAge.inHours) + val timedKeys = for (timePartition <- earliestTweetTimeInHour to latestTweetTimeInHour) yield { + + TopicTweetPartitionFlatKey( + entityId = query.storeQuery.topicId.entityId, + timePartition = timePartition, + algorithmType = Some(AlgorithmType.TfgTweet), + tweetEmbeddingType = Some(EmbeddingType.LogFavBasedTweet), + language = query.storeQuery.topicId.language.getOrElse("").toLowerCase, + country = None, // Disable country. It is not used. + semanticCoreAnnotationVersionId = Some(query.storeQuery.semanticCoreVersionId), + simclustersModelVersion = Some(ModelVersion.Model20m145k2020) + ) + } + + getTweetsForKeys( + timedKeys, + query.storeQuery.topicId + ) + } + + /** + * Given a set of keys, multiget the underlying Strato store, combine and flatten the results. + */ + private def getTweetsForKeys( + keys: Seq[TopicTweetPartitionFlatKey], + sourceTopic: TopicId + ): Future[Seq[SkitTopicTweet]] = { + Future + .collect { skitStratoStore.multiGet(keys.toSet).values.toSeq } + .map { combinedResults => + val topTweets = combinedResults.flatten.flatten + topTweets.map { tweet => + SkitTopicTweet( + tweetId = tweet.tweetId, + favCount = tweet.scores.favCount.getOrElse(0L), + cosineSimilarityScore = tweet.scores.cosineSimilarity.getOrElse(0.0), + sourceTopic = sourceTopic + ) + } + } + } +} + +object SkitTopicTweetSimilarityEngine { + + val MaxTweetAgeInHours: Int = 7.days.inHours // Simple guard to prevent overloading + + // Query is used as a cache key. Do not add any user level information in this. + case class Query( + topicId: TopicId, + maxCandidates: Int, + maxTweetAge: Duration, + semanticCoreVersionId: Long) + + case class SkitTopicTweet( + sourceTopic: TopicId, + tweetId: TweetId, + favCount: Long, + cosineSimilarityScore: Double) + + def fromParams( + topicId: TopicId, + isVideoOnly: Boolean, + params: configapi.Params, + ): EngineQuery[Query] = { + val maxCandidates = if (isVideoOnly) { + params(TopicTweetParams.MaxSkitTfgCandidatesParam) * 2 + } else { + params(TopicTweetParams.MaxSkitTfgCandidatesParam) + } + + EngineQuery( + Query( + topicId = topicId, + maxCandidates = maxCandidates, + maxTweetAge = params(TopicTweetParams.MaxTweetAge), + semanticCoreVersionId = params(TopicTweetParams.SemanticCoreVersionIdParam) + ), + params + ) + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/StandardSimilarityEngine.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/StandardSimilarityEngine.scala new file mode 100644 index 0000000000..ae71c37360 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/StandardSimilarityEngine.scala @@ -0,0 +1,65 @@ +package com.twitter.cr_mixer.similarity_engine + +import com.twitter.cr_mixer.similarity_engine.SimilarityEngine.MemCacheConfig +import com.twitter.cr_mixer.similarity_engine.SimilarityEngine.SimilarityEngineConfig +import com.twitter.cr_mixer.thriftscala.SimilarityEngineType +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.storehaus.ReadableStore +import com.twitter.timelines.configapi.Params +import com.twitter.util.Future + +/** + * @tparam Query ReadableStore's input type. + */ +case class EngineQuery[Query]( + storeQuery: Query, + params: Params, +) + +/** + * A straight forward SimilarityEngine implementation that wraps a ReadableStore + * + * @param implementingStore Provides the candidate retrieval's implementations + * @param memCacheConfig If specified, it will wrap the underlying store with a MemCache layer + * You should only enable this for cacheable queries, e.x. TweetIds. + * consumer based UserIds are generally not possible to cache. + * @tparam Query ReadableStore's input type + * @tparam Candidate ReadableStore's return type is Seq[[[Candidate]]] + */ +class StandardSimilarityEngine[Query, Candidate <: Serializable]( + implementingStore: ReadableStore[Query, Seq[Candidate]], + override val identifier: SimilarityEngineType, + globalStats: StatsReceiver, + engineConfig: SimilarityEngineConfig, + memCacheConfig: Option[MemCacheConfig[Query]] = None) + extends SimilarityEngine[EngineQuery[Query], Candidate] { + + private val scopedStats = globalStats.scope("similarityEngine", identifier.toString) + + def getScopedStats: StatsReceiver = scopedStats + + // Add memcache wrapper, if specified + private val store = { + memCacheConfig match { + case Some(config) => + SimilarityEngine.addMemCache( + underlyingStore = implementingStore, + memCacheConfig = config, + statsReceiver = scopedStats + ) + case _ => implementingStore + } + } + + override def getCandidates( + engineQuery: EngineQuery[Query] + ): Future[Option[Seq[Candidate]]] = { + SimilarityEngine.getFromFn( + store.get, + engineQuery.storeQuery, + engineConfig, + engineQuery.params, + scopedStats + ) + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/TweetBasedQigSimilarityEngine.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/TweetBasedQigSimilarityEngine.scala new file mode 100644 index 0000000000..317f097279 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/TweetBasedQigSimilarityEngine.scala @@ -0,0 +1,114 @@ +package com.twitter.cr_mixer.similarity_engine + +import com.twitter.cr_mixer.model.SimilarityEngineInfo +import com.twitter.cr_mixer.model.TweetWithScore +import com.twitter.cr_mixer.thriftscala.SimilarityEngineType +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.frigate.common.base.Stats +import com.twitter.product_mixer.core.thriftscala.ClientContext +import com.twitter.qig_ranker.thriftscala.Product +import com.twitter.qig_ranker.thriftscala.ProductContext +import com.twitter.qig_ranker.thriftscala.QigRanker +import com.twitter.qig_ranker.thriftscala.QigRankerProductResponse +import com.twitter.qig_ranker.thriftscala.QigRankerRequest +import com.twitter.qig_ranker.thriftscala.QigRankerResponse +import com.twitter.qig_ranker.thriftscala.TwistlySimilarTweetsProductContext +import com.twitter.simclusters_v2.thriftscala.InternalId +import com.twitter.storehaus.ReadableStore +import com.twitter.timelines.configapi +import com.twitter.util.Future +import javax.inject.Singleton + +/** + * This store looks for similar tweets from QueryInteractionGraph (QIG) for a source tweet id. + * For a given query tweet, QIG returns us the similar tweets that have an overlap of engagements + * (with the query tweet) on different search queries + */ +@Singleton +case class TweetBasedQigSimilarityEngine( + qigRanker: QigRanker.MethodPerEndpoint, + statsReceiver: StatsReceiver) + extends ReadableStore[ + TweetBasedQigSimilarityEngine.Query, + Seq[TweetWithScore] + ] { + + private val stats = statsReceiver.scope(this.getClass.getSimpleName) + private val fetchCandidatesStat = stats.scope("fetchCandidates") + + override def get( + query: TweetBasedQigSimilarityEngine.Query + ): Future[Option[Seq[TweetWithScore]]] = { + query.sourceId match { + case InternalId.TweetId(tweetId) => + val qigSimilarTweetsRequest = getQigSimilarTweetsRequest(tweetId) + + Stats.trackOption(fetchCandidatesStat) { + qigRanker + .getSimilarCandidates(qigSimilarTweetsRequest) + .map { qigSimilarTweetsResponse => + getCandidatesFromQigResponse(qigSimilarTweetsResponse) + } + } + case _ => + Future.value(None) + } + } + + private def getQigSimilarTweetsRequest( + tweetId: Long + ): QigRankerRequest = { + // Note: QigRanker needs a non-empty userId to be passed to return results. + // We are passing in a dummy userId until we fix this on QigRanker side + val clientContext = ClientContext(userId = Some(0L)) + val productContext = ProductContext.TwistlySimilarTweetsProductContext( + TwistlySimilarTweetsProductContext(tweetId = tweetId)) + + QigRankerRequest( + clientContext = clientContext, + product = Product.TwistlySimilarTweets, + productContext = Some(productContext), + ) + } + + private def getCandidatesFromQigResponse( + qigSimilarTweetsResponse: QigRankerResponse + ): Option[Seq[TweetWithScore]] = { + qigSimilarTweetsResponse.productResponse match { + case QigRankerProductResponse + .TwistlySimilarTweetCandidatesResponse(response) => + val tweetsWithScore = response.similarTweets + .map { similarTweetResult => + TweetWithScore( + similarTweetResult.tweetResult.tweetId, + similarTweetResult.tweetResult.score.getOrElse(0L)) + } + Some(tweetsWithScore) + + case _ => None + } + } +} + +object TweetBasedQigSimilarityEngine { + + def toSimilarityEngineInfo(score: Double): SimilarityEngineInfo = { + SimilarityEngineInfo( + similarityEngineType = SimilarityEngineType.Qig, + modelId = None, + score = Some(score)) + } + + case class Query(sourceId: InternalId) + + def fromParams( + sourceId: InternalId, + params: configapi.Params, + ): EngineQuery[Query] = { + EngineQuery( + Query(sourceId = sourceId), + params + ) + } + +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/TweetBasedUnifiedSimilarityEngine.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/TweetBasedUnifiedSimilarityEngine.scala new file mode 100644 index 0000000000..6b84e2f67d --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/TweetBasedUnifiedSimilarityEngine.scala @@ -0,0 +1,962 @@ +package com.twitter.cr_mixer.similarity_engine + +import com.twitter.cr_mixer.model.CandidateGenerationInfo +import com.twitter.cr_mixer.model.ModuleNames +import com.twitter.cr_mixer.model.SimilarityEngineInfo +import com.twitter.cr_mixer.model.SourceInfo +import com.twitter.cr_mixer.model.TweetWithCandidateGenerationInfo +import com.twitter.cr_mixer.model.TweetWithScore +import com.twitter.cr_mixer.param.GlobalParams +import com.twitter.cr_mixer.param.RelatedTweetTweetBasedParams +import com.twitter.cr_mixer.param.RelatedVideoTweetTweetBasedParams +import com.twitter.cr_mixer.param.SimClustersANNParams +import com.twitter.cr_mixer.param.TweetBasedCandidateGenerationParams +import com.twitter.cr_mixer.param.TweetBasedTwHINParams +import com.twitter.cr_mixer.thriftscala.SimilarityEngineType +import com.twitter.cr_mixer.thriftscala.SourceType +import com.twitter.cr_mixer.util.InterleaveUtil +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.frigate.common.util.StatsUtil +import com.twitter.simclusters_v2.common.ModelVersions +import com.twitter.simclusters_v2.thriftscala.EmbeddingType +import com.twitter.simclusters_v2.thriftscala.InternalId +import com.twitter.snowflake.id.SnowflakeId +import com.twitter.storehaus.ReadableStore +import com.twitter.timelines.configapi +import com.twitter.util.Duration +import com.twitter.util.Future +import com.twitter.util.Time +import javax.inject.Named +import javax.inject.Singleton +import scala.collection.mutable.ArrayBuffer + +/** + * This store fetches similar tweets from multiple tweet based candidate sources + * and combines them using different methods obtained from query params + */ +@Singleton +case class TweetBasedUnifiedSimilarityEngine( + @Named(ModuleNames.TweetBasedUserTweetGraphSimilarityEngine) + tweetBasedUserTweetGraphSimilarityEngine: StandardSimilarityEngine[ + TweetBasedUserTweetGraphSimilarityEngine.Query, + TweetWithScore + ], + @Named(ModuleNames.TweetBasedUserVideoGraphSimilarityEngine) + tweetBasedUserVideoGraphSimilarityEngine: StandardSimilarityEngine[ + TweetBasedUserVideoGraphSimilarityEngine.Query, + TweetWithScore + ], + simClustersANNSimilarityEngine: StandardSimilarityEngine[ + SimClustersANNSimilarityEngine.Query, + TweetWithScore + ], + @Named(ModuleNames.TweetBasedQigSimilarityEngine) + tweetBasedQigSimilarTweetsSimilarityEngine: StandardSimilarityEngine[ + TweetBasedQigSimilarityEngine.Query, + TweetWithScore + ], + @Named(ModuleNames.TweetBasedTwHINANNSimilarityEngine) + tweetBasedTwHINANNSimilarityEngine: HnswANNSimilarityEngine, + statsReceiver: StatsReceiver) + extends ReadableStore[ + TweetBasedUnifiedSimilarityEngine.Query, + Seq[TweetWithCandidateGenerationInfo] + ] { + + import TweetBasedUnifiedSimilarityEngine._ + private val stats = statsReceiver.scope(this.getClass.getSimpleName) + private val fetchCandidatesStat = stats.scope("fetchCandidates") + + override def get( + query: Query + ): Future[Option[Seq[TweetWithCandidateGenerationInfo]]] = { + + query.sourceInfo.internalId match { + case _: InternalId.TweetId => + StatsUtil.trackOptionItemsStats(fetchCandidatesStat) { + val twhinQuery = + HnswANNEngineQuery( + sourceId = query.sourceInfo.internalId, + modelId = query.twhinModelId, + params = query.params) + val utgCandidatesFut = + if (query.enableUtg) + tweetBasedUserTweetGraphSimilarityEngine.getCandidates(query.utgQuery) + else Future.None + + val uvgCandidatesFut = + if (query.enableUvg) + tweetBasedUserVideoGraphSimilarityEngine.getCandidates(query.uvgQuery) + else Future.None + + val sannCandidatesFut = if (query.enableSimClustersANN) { + simClustersANNSimilarityEngine.getCandidates(query.simClustersANNQuery) + } else Future.None + + val sann1CandidatesFut = + if (query.enableSimClustersANN1) { + simClustersANNSimilarityEngine.getCandidates(query.simClustersANN1Query) + } else Future.None + + val sann2CandidatesFut = + if (query.enableSimClustersANN2) { + simClustersANNSimilarityEngine.getCandidates(query.simClustersANN2Query) + } else Future.None + + val sann3CandidatesFut = + if (query.enableSimClustersANN3) { + simClustersANNSimilarityEngine.getCandidates(query.simClustersANN3Query) + } else Future.None + + val sann5CandidatesFut = + if (query.enableSimClustersANN5) { + simClustersANNSimilarityEngine.getCandidates(query.simClustersANN5Query) + } else Future.None + + val sann4CandidatesFut = + if (query.enableSimClustersANN4) { + simClustersANNSimilarityEngine.getCandidates(query.simClustersANN4Query) + } else Future.None + + val experimentalSANNCandidatesFut = + if (query.enableExperimentalSimClustersANN) { + simClustersANNSimilarityEngine.getCandidates(query.experimentalSimClustersANNQuery) + } else Future.None + + val qigCandidatesFut = + if (query.enableQig) + tweetBasedQigSimilarTweetsSimilarityEngine.getCandidates(query.qigQuery) + else Future.None + + val twHINCandidateFut = if (query.enableTwHIN) { + tweetBasedTwHINANNSimilarityEngine.getCandidates(twhinQuery) + } else Future.None + + Future + .join( + utgCandidatesFut, + sannCandidatesFut, + sann1CandidatesFut, + sann2CandidatesFut, + sann3CandidatesFut, + sann5CandidatesFut, + sann4CandidatesFut, + experimentalSANNCandidatesFut, + qigCandidatesFut, + twHINCandidateFut, + uvgCandidatesFut + ).map { + case ( + userTweetGraphCandidates, + simClustersANNCandidates, + simClustersANN1Candidates, + simClustersANN2Candidates, + simClustersANN3Candidates, + simClustersANN5Candidates, + simClustersANN4Candidates, + experimentalSANNCandidates, + qigSimilarTweetsCandidates, + twhinCandidates, + userVideoGraphCandidates) => + val filteredUTGTweets = + userTweetGraphFilter(userTweetGraphCandidates.toSeq.flatten) + val filteredUVGTweets = + userVideoGraphFilter(userVideoGraphCandidates.toSeq.flatten) + val filteredSANNTweets = simClustersCandidateMinScoreFilter( + simClustersANNCandidates.toSeq.flatten, + query.simClustersMinScore, + query.simClustersANNQuery.storeQuery.simClustersANNConfigId) + + val filteredSANN1Tweets = simClustersCandidateMinScoreFilter( + simClustersANN1Candidates.toSeq.flatten, + query.simClustersMinScore, + query.simClustersANN1Query.storeQuery.simClustersANNConfigId) + + val filteredSANN2Tweets = simClustersCandidateMinScoreFilter( + simClustersANN2Candidates.toSeq.flatten, + query.simClustersMinScore, + query.simClustersANN2Query.storeQuery.simClustersANNConfigId) + + val filteredSANN3Tweets = simClustersCandidateMinScoreFilter( + simClustersANN3Candidates.toSeq.flatten, + query.simClustersMinScore, + query.simClustersANN3Query.storeQuery.simClustersANNConfigId) + + val filteredSANN4Tweets = simClustersCandidateMinScoreFilter( + simClustersANN4Candidates.toSeq.flatten, + query.simClustersMinScore, + query.simClustersANN4Query.storeQuery.simClustersANNConfigId) + + val filteredSANN5Tweets = simClustersCandidateMinScoreFilter( + simClustersANN5Candidates.toSeq.flatten, + query.simClustersMinScore, + query.simClustersANN5Query.storeQuery.simClustersANNConfigId) + + val filteredExperimentalSANNTweets = simClustersCandidateMinScoreFilter( + experimentalSANNCandidates.toSeq.flatten, + query.simClustersVideoBasedMinScore, + query.experimentalSimClustersANNQuery.storeQuery.simClustersANNConfigId) + + val filteredQigTweets = qigSimilarTweetsFilter( + qigSimilarTweetsCandidates.toSeq.flatten, + query.qigMaxTweetAgeHours, + query.qigMaxNumSimilarTweets + ) + + val filteredTwHINTweets = twhinFilter( + twhinCandidates.toSeq.flatten.sortBy(-_.score), + query.twhinMaxTweetAgeHours, + tweetBasedTwHINANNSimilarityEngine.getScopedStats + ) + val utgTweetsWithCGInfo = filteredUTGTweets.map { tweetWithScore => + val similarityEngineInfo = TweetBasedUserTweetGraphSimilarityEngine + .toSimilarityEngineInfo(tweetWithScore.score) + TweetWithCandidateGenerationInfo( + tweetWithScore.tweetId, + CandidateGenerationInfo( + Some(query.sourceInfo), + similarityEngineInfo, + Seq(similarityEngineInfo) + )) + } + + val uvgTweetsWithCGInfo = filteredUVGTweets.map { tweetWithScore => + val similarityEngineInfo = TweetBasedUserVideoGraphSimilarityEngine + .toSimilarityEngineInfo(tweetWithScore.score) + TweetWithCandidateGenerationInfo( + tweetWithScore.tweetId, + CandidateGenerationInfo( + Some(query.sourceInfo), + similarityEngineInfo, + Seq(similarityEngineInfo) + )) + } + val sannTweetsWithCGInfo = filteredSANNTweets.map { tweetWithScore => + val similarityEngineInfo = SimClustersANNSimilarityEngine + .toSimilarityEngineInfo(query.simClustersANNQuery, tweetWithScore.score) + TweetWithCandidateGenerationInfo( + tweetWithScore.tweetId, + CandidateGenerationInfo( + Some(query.sourceInfo), + similarityEngineInfo, + Seq(similarityEngineInfo) + )) + } + val sann1TweetsWithCGInfo = filteredSANN1Tweets.map { tweetWithScore => + val similarityEngineInfo = SimClustersANNSimilarityEngine + .toSimilarityEngineInfo(query.simClustersANN1Query, tweetWithScore.score) + TweetWithCandidateGenerationInfo( + tweetWithScore.tweetId, + CandidateGenerationInfo( + Some(query.sourceInfo), + similarityEngineInfo, + Seq(similarityEngineInfo) + )) + } + val sann2TweetsWithCGInfo = filteredSANN2Tweets.map { tweetWithScore => + val similarityEngineInfo = SimClustersANNSimilarityEngine + .toSimilarityEngineInfo(query.simClustersANN2Query, tweetWithScore.score) + TweetWithCandidateGenerationInfo( + tweetWithScore.tweetId, + CandidateGenerationInfo( + Some(query.sourceInfo), + similarityEngineInfo, + Seq(similarityEngineInfo) + )) + } + val sann3TweetsWithCGInfo = filteredSANN3Tweets.map { tweetWithScore => + val similarityEngineInfo = SimClustersANNSimilarityEngine + .toSimilarityEngineInfo(query.simClustersANN3Query, tweetWithScore.score) + TweetWithCandidateGenerationInfo( + tweetWithScore.tweetId, + CandidateGenerationInfo( + Some(query.sourceInfo), + similarityEngineInfo, + Seq(similarityEngineInfo) + )) + } + val sann4TweetsWithCGInfo = filteredSANN4Tweets.map { tweetWithScore => + val similarityEngineInfo = SimClustersANNSimilarityEngine + .toSimilarityEngineInfo(query.simClustersANN4Query, tweetWithScore.score) + TweetWithCandidateGenerationInfo( + tweetWithScore.tweetId, + CandidateGenerationInfo( + Some(query.sourceInfo), + similarityEngineInfo, + Seq(similarityEngineInfo) + )) + } + val sann5TweetsWithCGInfo = filteredSANN5Tweets.map { tweetWithScore => + val similarityEngineInfo = SimClustersANNSimilarityEngine + .toSimilarityEngineInfo(query.simClustersANN5Query, tweetWithScore.score) + TweetWithCandidateGenerationInfo( + tweetWithScore.tweetId, + CandidateGenerationInfo( + Some(query.sourceInfo), + similarityEngineInfo, + Seq(similarityEngineInfo) + )) + } + + val experimentalSANNTweetsWithCGInfo = filteredExperimentalSANNTweets.map { + tweetWithScore => + val similarityEngineInfo = SimClustersANNSimilarityEngine + .toSimilarityEngineInfo( + query.experimentalSimClustersANNQuery, + tweetWithScore.score) + TweetWithCandidateGenerationInfo( + tweetWithScore.tweetId, + CandidateGenerationInfo( + Some(query.sourceInfo), + similarityEngineInfo, + Seq(similarityEngineInfo) + )) + } + val qigTweetsWithCGInfo = filteredQigTweets.map { tweetWithScore => + val similarityEngineInfo = TweetBasedQigSimilarityEngine + .toSimilarityEngineInfo(tweetWithScore.score) + TweetWithCandidateGenerationInfo( + tweetWithScore.tweetId, + CandidateGenerationInfo( + Some(query.sourceInfo), + similarityEngineInfo, + Seq(similarityEngineInfo) + )) + } + + val twHINTweetsWithCGInfo = filteredTwHINTweets.map { tweetWithScore => + val similarityEngineInfo = tweetBasedTwHINANNSimilarityEngine + .toSimilarityEngineInfo(twhinQuery, tweetWithScore.score) + TweetWithCandidateGenerationInfo( + tweetWithScore.tweetId, + CandidateGenerationInfo( + Some(query.sourceInfo), + similarityEngineInfo, + Seq(similarityEngineInfo) + )) + } + + val candidateSourcesToBeInterleaved = + ArrayBuffer[Seq[TweetWithCandidateGenerationInfo]]( + sannTweetsWithCGInfo, + experimentalSANNTweetsWithCGInfo, + sann1TweetsWithCGInfo, + sann2TweetsWithCGInfo, + sann3TweetsWithCGInfo, + sann5TweetsWithCGInfo, + sann4TweetsWithCGInfo, + qigTweetsWithCGInfo, + uvgTweetsWithCGInfo, + utgTweetsWithCGInfo, + twHINTweetsWithCGInfo + ) + + val interleavedCandidates = + InterleaveUtil.interleave(candidateSourcesToBeInterleaved) + + val unifiedCandidatesWithUnifiedCGInfo = + interleavedCandidates.map { candidate => + /*** + * when a candidate was made by interleave/keepGivenOrder, + * then we apply getTweetBasedUnifiedCGInfo() to override with the unified CGInfo + * + * we'll not have ALL SEs that generated the tweet + * in contributingSE list for interleave. We only have the chosen SE available. + */ + TweetWithCandidateGenerationInfo( + tweetId = candidate.tweetId, + candidateGenerationInfo = getTweetBasedUnifiedCGInfo( + candidate.candidateGenerationInfo.sourceInfoOpt, + candidate.getSimilarityScore, + candidate.candidateGenerationInfo.contributingSimilarityEngines + ) // getSimilarityScore comes from either unifiedScore or single score + ) + } + stats + .stat("unified_candidate_size").add(unifiedCandidatesWithUnifiedCGInfo.size) + + val truncatedCandidates = + unifiedCandidatesWithUnifiedCGInfo.take(query.maxCandidateNumPerSourceKey) + stats.stat("truncatedCandidates_size").add(truncatedCandidates.size) + + Some(truncatedCandidates) + } + } + + case _ => + stats.counter("sourceId_is_not_tweetId_cnt").incr() + Future.None + } + } + + private def simClustersCandidateMinScoreFilter( + simClustersAnnCandidates: Seq[TweetWithScore], + simClustersMinScore: Double, + simClustersANNConfigId: String + ): Seq[TweetWithScore] = { + val filteredCandidates = simClustersAnnCandidates + .filter { candidate => + candidate.score > simClustersMinScore + } + + stats.stat(simClustersANNConfigId, "simClustersAnnCandidates_size").add(filteredCandidates.size) + stats.counter(simClustersANNConfigId, "simClustersAnnRequests").incr() + if (filteredCandidates.isEmpty) + stats.counter(simClustersANNConfigId, "emptyFilteredSimClustersAnnCandidates").incr() + + filteredCandidates.map { candidate => + TweetWithScore(candidate.tweetId, candidate.score) + } + } + + /** Returns a list of tweets that are generated less than `maxTweetAgeHours` hours ago */ + private def tweetAgeFilter( + candidates: Seq[TweetWithScore], + maxTweetAgeHours: Duration + ): Seq[TweetWithScore] = { + // Tweet IDs are approximately chronological (see http://go/snowflake), + // so we are building the earliest tweet id once + // The per-candidate logic here then be candidate.tweetId > earliestPermittedTweetId, which is far cheaper. + val earliestTweetId = SnowflakeId.firstIdFor(Time.now - maxTweetAgeHours) + candidates.filter { candidate => candidate.tweetId >= earliestTweetId } + } + + private def twhinFilter( + twhinCandidates: Seq[TweetWithScore], + twhinMaxTweetAgeHours: Duration, + simEngineStats: StatsReceiver + ): Seq[TweetWithScore] = { + simEngineStats.stat("twhinCandidates_size").add(twhinCandidates.size) + val candidates = twhinCandidates.map { candidate => + TweetWithScore(candidate.tweetId, candidate.score) + } + + val filteredCandidates = tweetAgeFilter(candidates, twhinMaxTweetAgeHours) + simEngineStats.stat("filteredTwhinCandidates_size").add(filteredCandidates.size) + if (filteredCandidates.isEmpty) simEngineStats.counter("emptyFilteredTwhinCandidates").incr() + + filteredCandidates + } + + /** A no-op filter as UTG filtering already happens on UTG service side */ + private def userTweetGraphFilter( + userTweetGraphCandidates: Seq[TweetWithScore] + ): Seq[TweetWithScore] = { + val filteredCandidates = userTweetGraphCandidates + + stats.stat("userTweetGraphCandidates_size").add(userTweetGraphCandidates.size) + if (filteredCandidates.isEmpty) stats.counter("emptyFilteredUserTweetGraphCandidates").incr() + + filteredCandidates.map { candidate => + TweetWithScore(candidate.tweetId, candidate.score) + } + } + + /** A no-op filter as UVG filtering already happens on UVG service side */ + private def userVideoGraphFilter( + userVideoGraphCandidates: Seq[TweetWithScore] + ): Seq[TweetWithScore] = { + val filteredCandidates = userVideoGraphCandidates + + stats.stat("userVideoGraphCandidates_size").add(userVideoGraphCandidates.size) + if (filteredCandidates.isEmpty) stats.counter("emptyFilteredUserVideoGraphCandidates").incr() + + filteredCandidates.map { candidate => + TweetWithScore(candidate.tweetId, candidate.score) + } + } + private def qigSimilarTweetsFilter( + qigSimilarTweetsCandidates: Seq[TweetWithScore], + qigMaxTweetAgeHours: Duration, + qigMaxNumSimilarTweets: Int + ): Seq[TweetWithScore] = { + val ageFilteredCandidates = tweetAgeFilter(qigSimilarTweetsCandidates, qigMaxTweetAgeHours) + stats.stat("ageFilteredQigSimilarTweetsCandidates_size").add(ageFilteredCandidates.size) + + val filteredCandidates = ageFilteredCandidates.take(qigMaxNumSimilarTweets) + if (filteredCandidates.isEmpty) stats.counter("emptyFilteredQigSimilarTweetsCandidates").incr() + + filteredCandidates + } + + /*** + * Every candidate will have the CG Info with TweetBasedUnifiedSimilarityEngine + * as they are generated by a composite of Similarity Engines. + * Additionally, we store the contributing SEs (eg., SANN, UTG). + */ + private def getTweetBasedUnifiedCGInfo( + sourceInfoOpt: Option[SourceInfo], + unifiedScore: Double, + contributingSimilarityEngines: Seq[SimilarityEngineInfo] + ): CandidateGenerationInfo = { + CandidateGenerationInfo( + sourceInfoOpt, + SimilarityEngineInfo( + similarityEngineType = SimilarityEngineType.TweetBasedUnifiedSimilarityEngine, + modelId = None, // We do not assign modelId for a unified similarity engine + score = Some(unifiedScore) + ), + contributingSimilarityEngines + ) + } +} + +object TweetBasedUnifiedSimilarityEngine { + + case class Query( + sourceInfo: SourceInfo, + maxCandidateNumPerSourceKey: Int, + enableSimClustersANN: Boolean, + simClustersANNQuery: EngineQuery[SimClustersANNSimilarityEngine.Query], + enableExperimentalSimClustersANN: Boolean, + experimentalSimClustersANNQuery: EngineQuery[SimClustersANNSimilarityEngine.Query], + enableSimClustersANN1: Boolean, + simClustersANN1Query: EngineQuery[SimClustersANNSimilarityEngine.Query], + enableSimClustersANN2: Boolean, + simClustersANN2Query: EngineQuery[SimClustersANNSimilarityEngine.Query], + enableSimClustersANN3: Boolean, + simClustersANN3Query: EngineQuery[SimClustersANNSimilarityEngine.Query], + enableSimClustersANN5: Boolean, + simClustersANN5Query: EngineQuery[SimClustersANNSimilarityEngine.Query], + enableSimClustersANN4: Boolean, + simClustersANN4Query: EngineQuery[SimClustersANNSimilarityEngine.Query], + simClustersMinScore: Double, + simClustersVideoBasedMinScore: Double, + twhinModelId: String, + enableTwHIN: Boolean, + twhinMaxTweetAgeHours: Duration, + qigMaxTweetAgeHours: Duration, + qigMaxNumSimilarTweets: Int, + enableUtg: Boolean, + utgQuery: EngineQuery[TweetBasedUserTweetGraphSimilarityEngine.Query], + enableUvg: Boolean, + uvgQuery: EngineQuery[TweetBasedUserVideoGraphSimilarityEngine.Query], + enableQig: Boolean, + qigQuery: EngineQuery[TweetBasedQigSimilarityEngine.Query], + params: configapi.Params) + + def fromParams( + sourceInfo: SourceInfo, + params: configapi.Params, + ): EngineQuery[Query] = { + // SimClusters + val enableSimClustersANN = + params(TweetBasedCandidateGenerationParams.EnableSimClustersANNParam) + + val simClustersModelVersion = + ModelVersions.Enum.enumToSimClustersModelVersionMap(params(GlobalParams.ModelVersionParam)) + val simClustersMinScore = params(TweetBasedCandidateGenerationParams.SimClustersMinScoreParam) + val simClustersVideoBasedMinScore = params( + TweetBasedCandidateGenerationParams.SimClustersVideoBasedMinScoreParam) + val simClustersANNConfigId = params(SimClustersANNParams.SimClustersANNConfigId) + // SimClusters - Experimental SANN Similarity Engine (Video based SE) + val enableExperimentalSimClustersANN = + params(TweetBasedCandidateGenerationParams.EnableExperimentalSimClustersANNParam) + + val experimentalSimClustersANNConfigId = params( + SimClustersANNParams.ExperimentalSimClustersANNConfigId) + // SimClusters - SANN cluster 1 Similarity Engine + val enableSimClustersANN1 = + params(TweetBasedCandidateGenerationParams.EnableSimClustersANN1Param) + + val simClustersANN1ConfigId = params(SimClustersANNParams.SimClustersANN1ConfigId) + // SimClusters - SANN cluster 2 Similarity Engine + val enableSimClustersANN2 = + params(TweetBasedCandidateGenerationParams.EnableSimClustersANN2Param) + val simClustersANN2ConfigId = params(SimClustersANNParams.SimClustersANN2ConfigId) + // SimClusters - SANN cluster 3 Similarity Engine + val enableSimClustersANN3 = + params(TweetBasedCandidateGenerationParams.EnableSimClustersANN3Param) + val simClustersANN3ConfigId = params(SimClustersANNParams.SimClustersANN3ConfigId) + // SimClusters - SANN cluster 5 Similarity Engine + val enableSimClustersANN5 = + params(TweetBasedCandidateGenerationParams.EnableSimClustersANN5Param) + val simClustersANN5ConfigId = params(SimClustersANNParams.SimClustersANN5ConfigId) + // SimClusters - SANN cluster 4 Similarity Engine + val enableSimClustersANN4 = + params(TweetBasedCandidateGenerationParams.EnableSimClustersANN4Param) + val simClustersANN4ConfigId = params(SimClustersANNParams.SimClustersANN4ConfigId) + // SimClusters ANN Queries for different SANN clusters + val simClustersANNQuery = SimClustersANNSimilarityEngine.fromParams( + sourceInfo.internalId, + EmbeddingType.LogFavLongestL2EmbeddingTweet, + simClustersModelVersion, + simClustersANNConfigId, + params + ) + val experimentalSimClustersANNQuery = SimClustersANNSimilarityEngine.fromParams( + sourceInfo.internalId, + EmbeddingType.LogFavLongestL2EmbeddingTweet, + simClustersModelVersion, + experimentalSimClustersANNConfigId, + params + ) + val simClustersANN1Query = SimClustersANNSimilarityEngine.fromParams( + sourceInfo.internalId, + EmbeddingType.LogFavLongestL2EmbeddingTweet, + simClustersModelVersion, + simClustersANN1ConfigId, + params + ) + val simClustersANN2Query = SimClustersANNSimilarityEngine.fromParams( + sourceInfo.internalId, + EmbeddingType.LogFavLongestL2EmbeddingTweet, + simClustersModelVersion, + simClustersANN2ConfigId, + params + ) + val simClustersANN3Query = SimClustersANNSimilarityEngine.fromParams( + sourceInfo.internalId, + EmbeddingType.LogFavLongestL2EmbeddingTweet, + simClustersModelVersion, + simClustersANN3ConfigId, + params + ) + val simClustersANN5Query = SimClustersANNSimilarityEngine.fromParams( + sourceInfo.internalId, + EmbeddingType.LogFavLongestL2EmbeddingTweet, + simClustersModelVersion, + simClustersANN5ConfigId, + params + ) + val simClustersANN4Query = SimClustersANNSimilarityEngine.fromParams( + sourceInfo.internalId, + EmbeddingType.LogFavLongestL2EmbeddingTweet, + simClustersModelVersion, + simClustersANN4ConfigId, + params + ) + // TweetBasedCandidateGeneration + val maxCandidateNumPerSourceKey = params(GlobalParams.MaxCandidateNumPerSourceKeyParam) + // TwHIN + val twhinModelId = params(TweetBasedTwHINParams.ModelIdParam) + val enableTwHIN = + params(TweetBasedCandidateGenerationParams.EnableTwHINParam) + + val twhinMaxTweetAgeHours = params(GlobalParams.MaxTweetAgeHoursParam) + + // QIG + val enableQig = + params(TweetBasedCandidateGenerationParams.EnableQigSimilarTweetsParam) + val qigMaxTweetAgeHours = params(GlobalParams.MaxTweetAgeHoursParam) + val qigMaxNumSimilarTweets = params( + TweetBasedCandidateGenerationParams.QigMaxNumSimilarTweetsParam) + + // UTG + val enableUtg = + params(TweetBasedCandidateGenerationParams.EnableUTGParam) + // UVG + val enableUvg = + params(TweetBasedCandidateGenerationParams.EnableUVGParam) + EngineQuery( + Query( + sourceInfo = sourceInfo, + maxCandidateNumPerSourceKey = maxCandidateNumPerSourceKey, + enableSimClustersANN = enableSimClustersANN, + simClustersANNQuery = simClustersANNQuery, + enableExperimentalSimClustersANN = enableExperimentalSimClustersANN, + experimentalSimClustersANNQuery = experimentalSimClustersANNQuery, + enableSimClustersANN1 = enableSimClustersANN1, + simClustersANN1Query = simClustersANN1Query, + enableSimClustersANN2 = enableSimClustersANN2, + simClustersANN2Query = simClustersANN2Query, + enableSimClustersANN3 = enableSimClustersANN3, + simClustersANN3Query = simClustersANN3Query, + enableSimClustersANN5 = enableSimClustersANN5, + simClustersANN5Query = simClustersANN5Query, + enableSimClustersANN4 = enableSimClustersANN4, + simClustersANN4Query = simClustersANN4Query, + simClustersMinScore = simClustersMinScore, + simClustersVideoBasedMinScore = simClustersVideoBasedMinScore, + twhinModelId = twhinModelId, + enableTwHIN = enableTwHIN, + twhinMaxTweetAgeHours = twhinMaxTweetAgeHours, + qigMaxTweetAgeHours = qigMaxTweetAgeHours, + qigMaxNumSimilarTweets = qigMaxNumSimilarTweets, + enableUtg = enableUtg, + utgQuery = TweetBasedUserTweetGraphSimilarityEngine + .fromParams(sourceInfo.internalId, params), + enableQig = enableQig, + qigQuery = TweetBasedQigSimilarityEngine.fromParams(sourceInfo.internalId, params), + enableUvg = enableUvg, + uvgQuery = + TweetBasedUserVideoGraphSimilarityEngine.fromParams(sourceInfo.internalId, params), + params = params + ), + params + ) + } + + def fromParamsForRelatedTweet( + internalId: InternalId, + params: configapi.Params, + ): EngineQuery[Query] = { + // SimClusters + val enableSimClustersANN = params(RelatedTweetTweetBasedParams.EnableSimClustersANNParam) + val simClustersModelVersion = + ModelVersions.Enum.enumToSimClustersModelVersionMap(params(GlobalParams.ModelVersionParam)) + val simClustersMinScore = params(RelatedTweetTweetBasedParams.SimClustersMinScoreParam) + val simClustersANNConfigId = params(SimClustersANNParams.SimClustersANNConfigId) + val enableExperimentalSimClustersANN = + params(RelatedTweetTweetBasedParams.EnableExperimentalSimClustersANNParam) + val experimentalSimClustersANNConfigId = params( + SimClustersANNParams.ExperimentalSimClustersANNConfigId) + // SimClusters - SANN cluster 1 Similarity Engine + val enableSimClustersANN1 = params(RelatedTweetTweetBasedParams.EnableSimClustersANN1Param) + val simClustersANN1ConfigId = params(SimClustersANNParams.SimClustersANN1ConfigId) + // SimClusters - SANN cluster 2 Similarity Engine + val enableSimClustersANN2 = params(RelatedTweetTweetBasedParams.EnableSimClustersANN2Param) + val simClustersANN2ConfigId = params(SimClustersANNParams.SimClustersANN2ConfigId) + // SimClusters - SANN cluster 3 Similarity Engine + val enableSimClustersANN3 = params(RelatedTweetTweetBasedParams.EnableSimClustersANN3Param) + val simClustersANN3ConfigId = params(SimClustersANNParams.SimClustersANN3ConfigId) + // SimClusters - SANN cluster 5 Similarity Engine + val enableSimClustersANN5 = params(RelatedTweetTweetBasedParams.EnableSimClustersANN5Param) + val simClustersANN5ConfigId = params(SimClustersANNParams.SimClustersANN5ConfigId) + // SimClusters - SANN cluster 4 Similarity Engine + val enableSimClustersANN4 = params(RelatedTweetTweetBasedParams.EnableSimClustersANN4Param) + val simClustersANN4ConfigId = params(SimClustersANNParams.SimClustersANN4ConfigId) + // SimClusters ANN Queries for different SANN clusters + val simClustersANNQuery = SimClustersANNSimilarityEngine.fromParams( + internalId, + EmbeddingType.LogFavLongestL2EmbeddingTweet, + simClustersModelVersion, + simClustersANNConfigId, + params + ) + val experimentalSimClustersANNQuery = SimClustersANNSimilarityEngine.fromParams( + internalId, + EmbeddingType.LogFavLongestL2EmbeddingTweet, + simClustersModelVersion, + experimentalSimClustersANNConfigId, + params + ) + val simClustersANN1Query = SimClustersANNSimilarityEngine.fromParams( + internalId, + EmbeddingType.LogFavLongestL2EmbeddingTweet, + simClustersModelVersion, + simClustersANN1ConfigId, + params + ) + val simClustersANN2Query = SimClustersANNSimilarityEngine.fromParams( + internalId, + EmbeddingType.LogFavLongestL2EmbeddingTweet, + simClustersModelVersion, + simClustersANN2ConfigId, + params + ) + val simClustersANN3Query = SimClustersANNSimilarityEngine.fromParams( + internalId, + EmbeddingType.LogFavLongestL2EmbeddingTweet, + simClustersModelVersion, + simClustersANN3ConfigId, + params + ) + val simClustersANN5Query = SimClustersANNSimilarityEngine.fromParams( + internalId, + EmbeddingType.LogFavLongestL2EmbeddingTweet, + simClustersModelVersion, + simClustersANN5ConfigId, + params + ) + val simClustersANN4Query = SimClustersANNSimilarityEngine.fromParams( + internalId, + EmbeddingType.LogFavLongestL2EmbeddingTweet, + simClustersModelVersion, + simClustersANN4ConfigId, + params + ) + // TweetBasedCandidateGeneration + val maxCandidateNumPerSourceKey = params(GlobalParams.MaxCandidateNumPerSourceKeyParam) + // TwHIN + val twhinModelId = params(TweetBasedTwHINParams.ModelIdParam) + val enableTwHIN = params(RelatedTweetTweetBasedParams.EnableTwHINParam) + val twhinMaxTweetAgeHours = params(GlobalParams.MaxTweetAgeHoursParam) + // QIG + val enableQig = params(RelatedTweetTweetBasedParams.EnableQigSimilarTweetsParam) + val qigMaxTweetAgeHours = params(GlobalParams.MaxTweetAgeHoursParam) + val qigMaxNumSimilarTweets = params( + TweetBasedCandidateGenerationParams.QigMaxNumSimilarTweetsParam) + // UTG + val enableUtg = params(RelatedTweetTweetBasedParams.EnableUTGParam) + // UVG + val enableUvg = params(RelatedTweetTweetBasedParams.EnableUVGParam) + // SourceType.RequestTweetId is a placeholder. + val sourceInfo = SourceInfo(SourceType.RequestTweetId, internalId, None) + + EngineQuery( + Query( + sourceInfo = sourceInfo, + maxCandidateNumPerSourceKey = maxCandidateNumPerSourceKey, + enableSimClustersANN = enableSimClustersANN, + simClustersMinScore = simClustersMinScore, + simClustersVideoBasedMinScore = simClustersMinScore, + simClustersANNQuery = simClustersANNQuery, + enableExperimentalSimClustersANN = enableExperimentalSimClustersANN, + experimentalSimClustersANNQuery = experimentalSimClustersANNQuery, + enableSimClustersANN1 = enableSimClustersANN1, + simClustersANN1Query = simClustersANN1Query, + enableSimClustersANN2 = enableSimClustersANN2, + simClustersANN2Query = simClustersANN2Query, + enableSimClustersANN3 = enableSimClustersANN3, + simClustersANN3Query = simClustersANN3Query, + enableSimClustersANN5 = enableSimClustersANN5, + simClustersANN5Query = simClustersANN5Query, + enableSimClustersANN4 = enableSimClustersANN4, + simClustersANN4Query = simClustersANN4Query, + twhinModelId = twhinModelId, + enableTwHIN = enableTwHIN, + twhinMaxTweetAgeHours = twhinMaxTweetAgeHours, + qigMaxTweetAgeHours = qigMaxTweetAgeHours, + qigMaxNumSimilarTweets = qigMaxNumSimilarTweets, + enableUtg = enableUtg, + utgQuery = TweetBasedUserTweetGraphSimilarityEngine + .fromParams(sourceInfo.internalId, params), + enableQig = enableQig, + qigQuery = TweetBasedQigSimilarityEngine.fromParams(sourceInfo.internalId, params), + enableUvg = enableUvg, + uvgQuery = + TweetBasedUserVideoGraphSimilarityEngine.fromParams(sourceInfo.internalId, params), + params = params, + ), + params + ) + } + def fromParamsForRelatedVideoTweet( + internalId: InternalId, + params: configapi.Params, + ): EngineQuery[Query] = { + // SimClusters + val enableSimClustersANN = params(RelatedVideoTweetTweetBasedParams.EnableSimClustersANNParam) + val simClustersModelVersion = + ModelVersions.Enum.enumToSimClustersModelVersionMap(params(GlobalParams.ModelVersionParam)) + val simClustersMinScore = params(RelatedVideoTweetTweetBasedParams.SimClustersMinScoreParam) + val simClustersANNConfigId = params(SimClustersANNParams.SimClustersANNConfigId) + val enableExperimentalSimClustersANN = params( + RelatedVideoTweetTweetBasedParams.EnableExperimentalSimClustersANNParam) + val experimentalSimClustersANNConfigId = params( + SimClustersANNParams.ExperimentalSimClustersANNConfigId) + // SimClusters - SANN cluster 1 Similarity Engine + val enableSimClustersANN1 = params(RelatedVideoTweetTweetBasedParams.EnableSimClustersANN1Param) + val simClustersANN1ConfigId = params(SimClustersANNParams.SimClustersANN1ConfigId) + // SimClusters - SANN cluster 2 Similarity Engine + val enableSimClustersANN2 = params(RelatedVideoTweetTweetBasedParams.EnableSimClustersANN2Param) + val simClustersANN2ConfigId = params(SimClustersANNParams.SimClustersANN2ConfigId) + // SimClusters - SANN cluster 3 Similarity Engine + val enableSimClustersANN3 = params(RelatedVideoTweetTweetBasedParams.EnableSimClustersANN3Param) + val simClustersANN3ConfigId = params(SimClustersANNParams.SimClustersANN3ConfigId) + // SimClusters - SANN cluster 5 Similarity Engine + val enableSimClustersANN5 = params(RelatedVideoTweetTweetBasedParams.EnableSimClustersANN5Param) + val simClustersANN5ConfigId = params(SimClustersANNParams.SimClustersANN5ConfigId) + + // SimClusters - SANN cluster 4 Similarity Engine + val enableSimClustersANN4 = params(RelatedVideoTweetTweetBasedParams.EnableSimClustersANN4Param) + val simClustersANN4ConfigId = params(SimClustersANNParams.SimClustersANN4ConfigId) + // SimClusters ANN Queries for different SANN clusters + val simClustersANNQuery = SimClustersANNSimilarityEngine.fromParams( + internalId, + EmbeddingType.LogFavLongestL2EmbeddingTweet, + simClustersModelVersion, + simClustersANNConfigId, + params + ) + val experimentalSimClustersANNQuery = SimClustersANNSimilarityEngine.fromParams( + internalId, + EmbeddingType.LogFavLongestL2EmbeddingTweet, + simClustersModelVersion, + experimentalSimClustersANNConfigId, + params + ) + val simClustersANN1Query = SimClustersANNSimilarityEngine.fromParams( + internalId, + EmbeddingType.LogFavLongestL2EmbeddingTweet, + simClustersModelVersion, + simClustersANN1ConfigId, + params + ) + val simClustersANN2Query = SimClustersANNSimilarityEngine.fromParams( + internalId, + EmbeddingType.LogFavLongestL2EmbeddingTweet, + simClustersModelVersion, + simClustersANN2ConfigId, + params + ) + val simClustersANN3Query = SimClustersANNSimilarityEngine.fromParams( + internalId, + EmbeddingType.LogFavLongestL2EmbeddingTweet, + simClustersModelVersion, + simClustersANN3ConfigId, + params + ) + val simClustersANN5Query = SimClustersANNSimilarityEngine.fromParams( + internalId, + EmbeddingType.LogFavLongestL2EmbeddingTweet, + simClustersModelVersion, + simClustersANN5ConfigId, + params + ) + + val simClustersANN4Query = SimClustersANNSimilarityEngine.fromParams( + internalId, + EmbeddingType.LogFavLongestL2EmbeddingTweet, + simClustersModelVersion, + simClustersANN4ConfigId, + params + ) + // TweetBasedCandidateGeneration + val maxCandidateNumPerSourceKey = params(GlobalParams.MaxCandidateNumPerSourceKeyParam) + // TwHIN + val twhinModelId = params(TweetBasedTwHINParams.ModelIdParam) + val enableTwHIN = params(RelatedVideoTweetTweetBasedParams.EnableTwHINParam) + val twhinMaxTweetAgeHours = params(GlobalParams.MaxTweetAgeHoursParam) + // QIG + val enableQig = params(RelatedVideoTweetTweetBasedParams.EnableQigSimilarTweetsParam) + val qigMaxTweetAgeHours = params(GlobalParams.MaxTweetAgeHoursParam) + val qigMaxNumSimilarTweets = params( + TweetBasedCandidateGenerationParams.QigMaxNumSimilarTweetsParam) + // UTG + val enableUtg = params(RelatedVideoTweetTweetBasedParams.EnableUTGParam) + + // SourceType.RequestTweetId is a placeholder. + val sourceInfo = SourceInfo(SourceType.RequestTweetId, internalId, None) + + val enableUvg = params(RelatedVideoTweetTweetBasedParams.EnableUVGParam) + EngineQuery( + Query( + sourceInfo = sourceInfo, + maxCandidateNumPerSourceKey = maxCandidateNumPerSourceKey, + enableSimClustersANN = enableSimClustersANN, + simClustersMinScore = simClustersMinScore, + simClustersVideoBasedMinScore = simClustersMinScore, + simClustersANNQuery = simClustersANNQuery, + enableExperimentalSimClustersANN = enableExperimentalSimClustersANN, + experimentalSimClustersANNQuery = experimentalSimClustersANNQuery, + enableSimClustersANN1 = enableSimClustersANN1, + simClustersANN1Query = simClustersANN1Query, + enableSimClustersANN2 = enableSimClustersANN2, + simClustersANN2Query = simClustersANN2Query, + enableSimClustersANN3 = enableSimClustersANN3, + simClustersANN3Query = simClustersANN3Query, + enableSimClustersANN5 = enableSimClustersANN5, + simClustersANN5Query = simClustersANN5Query, + enableSimClustersANN4 = enableSimClustersANN4, + simClustersANN4Query = simClustersANN4Query, + twhinModelId = twhinModelId, + enableTwHIN = enableTwHIN, + twhinMaxTweetAgeHours = twhinMaxTweetAgeHours, + qigMaxTweetAgeHours = qigMaxTweetAgeHours, + qigMaxNumSimilarTweets = qigMaxNumSimilarTweets, + enableUtg = enableUtg, + utgQuery = TweetBasedUserTweetGraphSimilarityEngine + .fromParams(sourceInfo.internalId, params), + enableUvg = enableUvg, + uvgQuery = + TweetBasedUserVideoGraphSimilarityEngine.fromParams(sourceInfo.internalId, params), + enableQig = enableQig, + qigQuery = TweetBasedQigSimilarityEngine.fromParams(sourceInfo.internalId, params), + params = params + ), + params + ) + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/TweetBasedUserAdGraphSimilarityEngine.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/TweetBasedUserAdGraphSimilarityEngine.scala new file mode 100644 index 0000000000..365bead351 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/TweetBasedUserAdGraphSimilarityEngine.scala @@ -0,0 +1,129 @@ +package com.twitter.cr_mixer.similarity_engine + +import com.twitter.cr_mixer.model.SimilarityEngineInfo +import com.twitter.cr_mixer.model.TweetWithScore +import com.twitter.cr_mixer.param.GlobalParams +import com.twitter.cr_mixer.param.TweetBasedUserAdGraphParams +import com.twitter.cr_mixer.thriftscala.SimilarityEngineType +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.frigate.common.util.StatsUtil +import com.twitter.recos.user_ad_graph.thriftscala.ConsumersBasedRelatedAdRequest +import com.twitter.recos.user_ad_graph.thriftscala.RelatedAdResponse +import com.twitter.recos.user_ad_graph.thriftscala.UserAdGraph +import com.twitter.simclusters_v2.common.TweetId +import com.twitter.simclusters_v2.thriftscala.InternalId +import com.twitter.storehaus.ReadableStore +import com.twitter.timelines.configapi +import com.twitter.twistly.thriftscala.TweetRecentEngagedUsers +import com.twitter.util.Future +import javax.inject.Singleton + +/** + * This store looks for similar tweets from UserAdGraph for a Source TweetId + * For a query tweet,User Ad Graph (UAG) + * lets us find out which other tweets share a lot of the same engagers with the query tweet + */ +@Singleton +case class TweetBasedUserAdGraphSimilarityEngine( + userAdGraphService: UserAdGraph.MethodPerEndpoint, + tweetEngagedUsersStore: ReadableStore[TweetId, TweetRecentEngagedUsers], + statsReceiver: StatsReceiver) + extends ReadableStore[ + TweetBasedUserAdGraphSimilarityEngine.Query, + Seq[TweetWithScore] + ] { + + import TweetBasedUserAdGraphSimilarityEngine._ + + private val stats = statsReceiver.scope(this.getClass.getSimpleName) + private val fetchCoverageExpansionCandidatesStat = stats.scope("fetchCoverageExpansionCandidates") + override def get( + query: TweetBasedUserAdGraphSimilarityEngine.Query + ): Future[Option[Seq[TweetWithScore]]] = { + query.sourceId match { + case InternalId.TweetId(tweetId) => getCandidates(tweetId, query) + case _ => + Future.value(None) + } + } + + // We first fetch tweet's recent engaged users as consumeSeedSet from MH store, + // then query consumersBasedUTG using the consumerSeedSet + private def getCandidates( + tweetId: TweetId, + query: TweetBasedUserAdGraphSimilarityEngine.Query + ): Future[Option[Seq[TweetWithScore]]] = { + StatsUtil + .trackOptionItemsStats(fetchCoverageExpansionCandidatesStat) { + tweetEngagedUsersStore + .get(tweetId).flatMap { + _.map { tweetRecentEngagedUsers => + val consumerSeedSet = + tweetRecentEngagedUsers.recentEngagedUsers + .map { _.userId }.take(query.maxConsumerSeedsNum) + val consumersBasedRelatedAdRequest = + ConsumersBasedRelatedAdRequest( + consumerSeedSet = consumerSeedSet, + maxResults = Some(query.maxResults), + minCooccurrence = Some(query.minCooccurrence), + excludeTweetIds = Some(Seq(tweetId)), + minScore = Some(query.consumersBasedMinScore), + maxTweetAgeInHours = Some(query.maxTweetAgeInHours) + ) + toTweetWithScore(userAdGraphService + .consumersBasedRelatedAds(consumersBasedRelatedAdRequest).map { Some(_) }) + }.getOrElse(Future.value(None)) + } + } + } + +} + +object TweetBasedUserAdGraphSimilarityEngine { + + def toSimilarityEngineInfo(score: Double): SimilarityEngineInfo = { + SimilarityEngineInfo( + similarityEngineType = SimilarityEngineType.TweetBasedUserAdGraph, + modelId = None, + score = Some(score)) + } + private def toTweetWithScore( + relatedAdResponseFut: Future[Option[RelatedAdResponse]] + ): Future[Option[Seq[TweetWithScore]]] = { + relatedAdResponseFut.map { relatedAdResponseOpt => + relatedAdResponseOpt.map { relatedAdResponse => + val candidates = + relatedAdResponse.adTweets.map(tweet => TweetWithScore(tweet.adTweetId, tweet.score)) + + candidates + } + } + } + + case class Query( + sourceId: InternalId, + maxResults: Int, + minCooccurrence: Int, + consumersBasedMinScore: Double, + maxTweetAgeInHours: Int, + maxConsumerSeedsNum: Int, + ) + + def fromParams( + sourceId: InternalId, + params: configapi.Params, + ): EngineQuery[Query] = { + EngineQuery( + Query( + sourceId = sourceId, + maxResults = params(GlobalParams.MaxCandidateNumPerSourceKeyParam), + minCooccurrence = params(TweetBasedUserAdGraphParams.MinCoOccurrenceParam), + consumersBasedMinScore = params(TweetBasedUserAdGraphParams.ConsumersBasedMinScoreParam), + maxTweetAgeInHours = params(GlobalParams.MaxTweetAgeHoursParam).inHours, + maxConsumerSeedsNum = params(TweetBasedUserAdGraphParams.MaxConsumerSeedsNumParam), + ), + params + ) + } + +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/TweetBasedUserTweetGraphSimilarityEngine.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/TweetBasedUserTweetGraphSimilarityEngine.scala new file mode 100644 index 0000000000..316f980a72 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/TweetBasedUserTweetGraphSimilarityEngine.scala @@ -0,0 +1,184 @@ +package com.twitter.cr_mixer.similarity_engine + +import com.twitter.cr_mixer.model.SimilarityEngineInfo +import com.twitter.cr_mixer.model.TweetWithScore +import com.twitter.cr_mixer.param.GlobalParams +import com.twitter.cr_mixer.param.TweetBasedUserTweetGraphParams +import com.twitter.cr_mixer.thriftscala.SimilarityEngineType +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.frigate.common.util.StatsUtil +import com.twitter.recos.user_tweet_graph.thriftscala.RelatedTweetResponse +import com.twitter.recos.user_tweet_graph.thriftscala.TweetBasedRelatedTweetRequest +import com.twitter.recos.user_tweet_graph.thriftscala.ConsumersBasedRelatedTweetRequest +import com.twitter.recos.user_tweet_graph.thriftscala.UserTweetGraph +import com.twitter.simclusters_v2.common.TweetId +import com.twitter.simclusters_v2.thriftscala.InternalId +import com.twitter.storehaus.ReadableStore +import com.twitter.twistly.thriftscala.TweetRecentEngagedUsers +import com.twitter.util.Future +import javax.inject.Singleton +import com.twitter.snowflake.id.SnowflakeId +import com.twitter.timelines.configapi +import com.twitter.util.Duration +import com.twitter.util.Time +import scala.concurrent.duration.HOURS + +/** + * This store looks for similar tweets from UserTweetGraph for a Source TweetId + * For a query tweet,User Tweet Graph (UTG), + * lets us find out which other tweets share a lot of the same engagers with the query tweet + * one-pager: go/UTG + */ +@Singleton +case class TweetBasedUserTweetGraphSimilarityEngine( + userTweetGraphService: UserTweetGraph.MethodPerEndpoint, + tweetEngagedUsersStore: ReadableStore[TweetId, TweetRecentEngagedUsers], + statsReceiver: StatsReceiver) + extends ReadableStore[ + TweetBasedUserTweetGraphSimilarityEngine.Query, + Seq[TweetWithScore] + ] { + + import TweetBasedUserTweetGraphSimilarityEngine._ + + private val stats = statsReceiver.scope(this.getClass.getSimpleName) + private val fetchCandidatesStat = stats.scope("fetchCandidates") + private val fetchCoverageExpansionCandidatesStat = stats.scope("fetchCoverageExpansionCandidates") + + override def get( + query: TweetBasedUserTweetGraphSimilarityEngine.Query + ): Future[Option[Seq[TweetWithScore]]] = { + query.sourceId match { + case InternalId.TweetId(tweetId) if query.enableCoverageExpansionAllTweet => + getCoverageExpansionCandidates(tweetId, query) + + case InternalId.TweetId(tweetId) if query.enableCoverageExpansionOldTweet => // For Home + if (isOldTweet(tweetId)) getCoverageExpansionCandidates(tweetId, query) + else getCandidates(tweetId, query) + + case InternalId.TweetId(tweetId) => getCandidates(tweetId, query) + case _ => + Future.value(None) + } + } + + // This is the main candidate source + private def getCandidates( + tweetId: TweetId, + query: TweetBasedUserTweetGraphSimilarityEngine.Query + ): Future[Option[Seq[TweetWithScore]]] = { + StatsUtil.trackOptionItemsStats(fetchCandidatesStat) { + val tweetBasedRelatedTweetRequest = { + TweetBasedRelatedTweetRequest( + tweetId, + maxResults = Some(query.maxResults), + minCooccurrence = Some(query.minCooccurrence), + excludeTweetIds = Some(Seq(tweetId)), + minScore = Some(query.tweetBasedMinScore), + maxTweetAgeInHours = Some(query.maxTweetAgeInHours) + ) + } + toTweetWithScore( + userTweetGraphService.tweetBasedRelatedTweets(tweetBasedRelatedTweetRequest).map { + Some(_) + }) + } + } + + // function for DDGs, for coverage expansion algo, we first fetch tweet's recent engaged users as consumeSeedSet from MH store, + // and query consumersBasedUTG using the consumeSeedSet + private def getCoverageExpansionCandidates( + tweetId: TweetId, + query: TweetBasedUserTweetGraphSimilarityEngine.Query + ): Future[Option[Seq[TweetWithScore]]] = { + StatsUtil + .trackOptionItemsStats(fetchCoverageExpansionCandidatesStat) { + tweetEngagedUsersStore + .get(tweetId).flatMap { + _.map { tweetRecentEngagedUsers => + val consumerSeedSet = + tweetRecentEngagedUsers.recentEngagedUsers + .map { _.userId }.take(query.maxConsumerSeedsNum) + val consumersBasedRelatedTweetRequest = + ConsumersBasedRelatedTweetRequest( + consumerSeedSet = consumerSeedSet, + maxResults = Some(query.maxResults), + minCooccurrence = Some(query.minCooccurrence), + excludeTweetIds = Some(Seq(tweetId)), + minScore = Some(query.consumersBasedMinScore), + maxTweetAgeInHours = Some(query.maxTweetAgeInHours) + ) + + toTweetWithScore(userTweetGraphService + .consumersBasedRelatedTweets(consumersBasedRelatedTweetRequest).map { Some(_) }) + }.getOrElse(Future.value(None)) + } + } + } + +} + +object TweetBasedUserTweetGraphSimilarityEngine { + + def toSimilarityEngineInfo(score: Double): SimilarityEngineInfo = { + SimilarityEngineInfo( + similarityEngineType = SimilarityEngineType.TweetBasedUserTweetGraph, + modelId = None, + score = Some(score)) + } + + private val oldTweetCap: Duration = Duration(48, HOURS) + + private def toTweetWithScore( + relatedTweetResponseFut: Future[Option[RelatedTweetResponse]] + ): Future[Option[Seq[TweetWithScore]]] = { + relatedTweetResponseFut.map { relatedTweetResponseOpt => + relatedTweetResponseOpt.map { relatedTweetResponse => + val candidates = + relatedTweetResponse.tweets.map(tweet => TweetWithScore(tweet.tweetId, tweet.score)) + candidates + } + } + } + + private def isOldTweet(tweetId: TweetId): Boolean = { + SnowflakeId + .timeFromIdOpt(tweetId).forall { tweetTime => tweetTime < Time.now - oldTweetCap } + // If there's no snowflake timestamp, we have no idea when this tweet happened. + } + + case class Query( + sourceId: InternalId, + maxResults: Int, + minCooccurrence: Int, + tweetBasedMinScore: Double, + consumersBasedMinScore: Double, + maxTweetAgeInHours: Int, + maxConsumerSeedsNum: Int, + enableCoverageExpansionOldTweet: Boolean, + enableCoverageExpansionAllTweet: Boolean, + ) + + def fromParams( + sourceId: InternalId, + params: configapi.Params, + ): EngineQuery[Query] = { + EngineQuery( + Query( + sourceId = sourceId, + maxResults = params(GlobalParams.MaxCandidateNumPerSourceKeyParam), + minCooccurrence = params(TweetBasedUserTweetGraphParams.MinCoOccurrenceParam), + tweetBasedMinScore = params(TweetBasedUserTweetGraphParams.TweetBasedMinScoreParam), + consumersBasedMinScore = params(TweetBasedUserTweetGraphParams.ConsumersBasedMinScoreParam), + maxTweetAgeInHours = params(GlobalParams.MaxTweetAgeHoursParam).inHours, + maxConsumerSeedsNum = params(TweetBasedUserTweetGraphParams.MaxConsumerSeedsNumParam), + enableCoverageExpansionOldTweet = + params(TweetBasedUserTweetGraphParams.EnableCoverageExpansionOldTweetParam), + enableCoverageExpansionAllTweet = + params(TweetBasedUserTweetGraphParams.EnableCoverageExpansionAllTweetParam), + ), + params + ) + } + +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/TweetBasedUserVideoGraphSimilarityEngine.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/TweetBasedUserVideoGraphSimilarityEngine.scala new file mode 100644 index 0000000000..6190cd4fb3 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/TweetBasedUserVideoGraphSimilarityEngine.scala @@ -0,0 +1,184 @@ +package com.twitter.cr_mixer.similarity_engine + +import com.twitter.cr_mixer.model.SimilarityEngineInfo +import com.twitter.cr_mixer.model.TweetWithScore +import com.twitter.cr_mixer.param.GlobalParams +import com.twitter.cr_mixer.param.TweetBasedUserVideoGraphParams +import com.twitter.cr_mixer.thriftscala.SimilarityEngineType +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.frigate.common.util.StatsUtil +import com.twitter.recos.user_video_graph.thriftscala.RelatedTweetResponse +import com.twitter.recos.user_video_graph.thriftscala.ConsumersBasedRelatedTweetRequest +import com.twitter.recos.user_video_graph.thriftscala.TweetBasedRelatedTweetRequest +import com.twitter.recos.user_video_graph.thriftscala.UserVideoGraph +import com.twitter.simclusters_v2.common.TweetId +import com.twitter.simclusters_v2.thriftscala.InternalId +import com.twitter.storehaus.ReadableStore +import com.twitter.snowflake.id.SnowflakeId +import com.twitter.timelines.configapi +import com.twitter.twistly.thriftscala.TweetRecentEngagedUsers +import com.twitter.util.Duration +import javax.inject.Singleton +import com.twitter.util.Future +import com.twitter.util.Time +import scala.concurrent.duration.HOURS + +/** + * This store looks for similar tweets from UserVideoGraph for a Source TweetId + * For a query tweet,User Video Graph (UVG), + * lets us find out which other video tweets share a lot of the same engagers with the query tweet + */ +@Singleton +case class TweetBasedUserVideoGraphSimilarityEngine( + userVideoGraphService: UserVideoGraph.MethodPerEndpoint, + tweetEngagedUsersStore: ReadableStore[TweetId, TweetRecentEngagedUsers], + statsReceiver: StatsReceiver) + extends ReadableStore[ + TweetBasedUserVideoGraphSimilarityEngine.Query, + Seq[TweetWithScore] + ] { + + import TweetBasedUserVideoGraphSimilarityEngine._ + + private val stats = statsReceiver.scope(this.getClass.getSimpleName) + private val fetchCandidatesStat = stats.scope("fetchCandidates") + private val fetchCoverageExpansionCandidatesStat = stats.scope("fetchCoverageExpansionCandidates") + + override def get( + query: TweetBasedUserVideoGraphSimilarityEngine.Query + ): Future[Option[Seq[TweetWithScore]]] = { + + query.sourceId match { + case InternalId.TweetId(tweetId) if query.enableCoverageExpansionAllTweet => + getCoverageExpansionCandidates(tweetId, query) + + case InternalId.TweetId(tweetId) if query.enableCoverageExpansionOldTweet => // For Home + if (isOldTweet(tweetId)) getCoverageExpansionCandidates(tweetId, query) + else getCandidates(tweetId, query) + + case InternalId.TweetId(tweetId) => getCandidates(tweetId, query) + case _ => + Future.value(None) + } + } + + private def getCandidates( + tweetId: TweetId, + query: TweetBasedUserVideoGraphSimilarityEngine.Query + ): Future[Option[Seq[TweetWithScore]]] = { + StatsUtil.trackOptionItemsStats(fetchCandidatesStat) { + val tweetBasedRelatedTweetRequest = { + TweetBasedRelatedTweetRequest( + tweetId, + maxResults = Some(query.maxResults), + minCooccurrence = Some(query.minCooccurrence), + excludeTweetIds = Some(Seq(tweetId)), + minScore = Some(query.tweetBasedMinScore), + maxTweetAgeInHours = Some(query.maxTweetAgeInHours) + ) + } + toTweetWithScore( + userVideoGraphService.tweetBasedRelatedTweets(tweetBasedRelatedTweetRequest).map { + Some(_) + }) + } + } + + private def getCoverageExpansionCandidates( + tweetId: TweetId, + query: TweetBasedUserVideoGraphSimilarityEngine.Query + ): Future[Option[Seq[TweetWithScore]]] = { + StatsUtil + .trackOptionItemsStats(fetchCoverageExpansionCandidatesStat) { + tweetEngagedUsersStore + .get(tweetId).flatMap { + _.map { tweetRecentEngagedUsers => + val consumerSeedSet = + tweetRecentEngagedUsers.recentEngagedUsers + .map { + _.userId + }.take(query.maxConsumerSeedsNum) + val consumersBasedRelatedTweetRequest = + ConsumersBasedRelatedTweetRequest( + consumerSeedSet = consumerSeedSet, + maxResults = Some(query.maxResults), + minCooccurrence = Some(query.minCooccurrence), + excludeTweetIds = Some(Seq(tweetId)), + minScore = Some(query.consumersBasedMinScore), + maxTweetAgeInHours = Some(query.maxTweetAgeInHours) + ) + + toTweetWithScore(userVideoGraphService + .consumersBasedRelatedTweets(consumersBasedRelatedTweetRequest).map { + Some(_) + }) + }.getOrElse(Future.value(None)) + } + } + } + +} + +object TweetBasedUserVideoGraphSimilarityEngine { + + private val oldTweetCap: Duration = Duration(24, HOURS) + + def toSimilarityEngineInfo(score: Double): SimilarityEngineInfo = { + SimilarityEngineInfo( + similarityEngineType = SimilarityEngineType.TweetBasedUserVideoGraph, + modelId = None, + score = Some(score)) + } + + private def toTweetWithScore( + relatedTweetResponseFut: Future[Option[RelatedTweetResponse]] + ): Future[Option[Seq[TweetWithScore]]] = { + relatedTweetResponseFut.map { relatedTweetResponseOpt => + relatedTweetResponseOpt.map { relatedTweetResponse => + val candidates = + relatedTweetResponse.tweets.map(tweet => TweetWithScore(tweet.tweetId, tweet.score)) + candidates + } + } + } + + private def isOldTweet(tweetId: TweetId): Boolean = { + SnowflakeId + .timeFromIdOpt(tweetId).forall { tweetTime => tweetTime < Time.now - oldTweetCap } + // If there's no snowflake timestamp, we have no idea when this tweet happened. + } + + case class Query( + sourceId: InternalId, + maxResults: Int, + minCooccurrence: Int, + tweetBasedMinScore: Double, + consumersBasedMinScore: Double, + maxTweetAgeInHours: Int, + maxConsumerSeedsNum: Int, + enableCoverageExpansionOldTweet: Boolean, + enableCoverageExpansionAllTweet: Boolean) + + def fromParams( + sourceId: InternalId, + params: configapi.Params, + ): EngineQuery[Query] = { + EngineQuery( + Query( + sourceId = sourceId, + maxResults = params(GlobalParams.MaxCandidateNumPerSourceKeyParam), + minCooccurrence = params(TweetBasedUserVideoGraphParams.MinCoOccurrenceParam), + tweetBasedMinScore = params(TweetBasedUserVideoGraphParams.TweetBasedMinScoreParam), + consumersBasedMinScore = params(TweetBasedUserVideoGraphParams.ConsumersBasedMinScoreParam), + maxTweetAgeInHours = params(GlobalParams.MaxTweetAgeHoursParam).inHours, + maxConsumerSeedsNum = params(TweetBasedUserVideoGraphParams.MaxConsumerSeedsNumParam), + enableCoverageExpansionOldTweet = + params(TweetBasedUserVideoGraphParams.EnableCoverageExpansionOldTweetParam), + enableCoverageExpansionAllTweet = + params(TweetBasedUserVideoGraphParams.EnableCoverageExpansionAllTweetParam) + ), + params + ) + } + +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/TwhinCollabFilterSimilarityEngine.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/TwhinCollabFilterSimilarityEngine.scala new file mode 100644 index 0000000000..eccab6aa31 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/TwhinCollabFilterSimilarityEngine.scala @@ -0,0 +1,72 @@ +package com.twitter.cr_mixer.similarity_engine + +import com.twitter.cr_mixer.model.SimilarityEngineInfo +import com.twitter.simclusters_v2.common.TweetId +import com.twitter.cr_mixer.model.TweetWithScore +import com.twitter.cr_mixer.thriftscala.SimilarityEngineType +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.simclusters_v2.thriftscala.InternalId +import com.twitter.storehaus.ReadableStore +import com.twitter.timelines.configapi +import com.twitter.util.Future +import javax.inject.Singleton + +@Singleton +case class TwhinCollabFilterSimilarityEngine( + twhinCandidatesStratoStore: ReadableStore[Long, Seq[TweetId]], + statsReceiver: StatsReceiver) + extends ReadableStore[ + TwhinCollabFilterSimilarityEngine.Query, + Seq[TweetWithScore] + ] { + + import TwhinCollabFilterSimilarityEngine._ + override def get( + query: TwhinCollabFilterSimilarityEngine.Query + ): Future[Option[Seq[TweetWithScore]]] = { + + query.sourceId match { + case InternalId.UserId(userId) => + twhinCandidatesStratoStore.get(userId).map { + _.map { + _.map { tweetId => TweetWithScore(tweetId, defaultScore) } + } + } + case _ => + Future.None + } + } +} + +object TwhinCollabFilterSimilarityEngine { + + val defaultScore: Double = 1.0 + + case class TwhinCollabFilterView(clusterVersion: String) + + case class Query( + sourceId: InternalId, + ) + + def toSimilarityEngineInfo( + query: LookupEngineQuery[Query], + score: Double + ): SimilarityEngineInfo = { + SimilarityEngineInfo( + similarityEngineType = SimilarityEngineType.TwhinCollabFilter, + modelId = Some(query.lookupKey), + score = Some(score)) + } + + def fromParams( + sourceId: InternalId, + modelId: String, + params: configapi.Params, + ): LookupEngineQuery[Query] = { + LookupEngineQuery( + Query(sourceId = sourceId), + modelId, + params + ) + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/UserTweetEntityGraphSimilarityEngine.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/UserTweetEntityGraphSimilarityEngine.scala new file mode 100644 index 0000000000..9c61b3d1c3 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/similarity_engine/UserTweetEntityGraphSimilarityEngine.scala @@ -0,0 +1,110 @@ +package com.twitter.cr_mixer.similarity_engine + +import com.twitter.recos.recos_common.thriftscala.SocialProofType +import com.twitter.cr_mixer.model.SimilarityEngineInfo +import com.twitter.cr_mixer.model.TweetWithScoreAndSocialProof +import com.twitter.cr_mixer.param.UtegTweetGlobalParams +import com.twitter.cr_mixer.thriftscala.SimilarityEngineType +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.recos.user_tweet_entity_graph.thriftscala.TweetEntityDisplayLocation +import com.twitter.recos.user_tweet_entity_graph.thriftscala.UserTweetEntityGraph +import com.twitter.recos.user_tweet_entity_graph.thriftscala.RecommendTweetEntityRequest +import com.twitter.recos.user_tweet_entity_graph.thriftscala.RecommendationType +import com.twitter.recos.user_tweet_entity_graph.thriftscala.UserTweetEntityRecommendationUnion.TweetRec +import com.twitter.simclusters_v2.common.UserId +import com.twitter.simclusters_v2.common.TweetId +import com.twitter.storehaus.ReadableStore +import com.twitter.timelines.configapi +import com.twitter.util.Duration +import com.twitter.util.Future +import javax.inject.Singleton + +@Singleton +case class UserTweetEntityGraphSimilarityEngine( + userTweetEntityGraph: UserTweetEntityGraph.MethodPerEndpoint, + statsReceiver: StatsReceiver) + extends ReadableStore[ + UserTweetEntityGraphSimilarityEngine.Query, + Seq[TweetWithScoreAndSocialProof] + ] { + + override def get( + query: UserTweetEntityGraphSimilarityEngine.Query + ): Future[Option[Seq[TweetWithScoreAndSocialProof]]] = { + val recommendTweetEntityRequest = + RecommendTweetEntityRequest( + requesterId = query.userId, + displayLocation = TweetEntityDisplayLocation.HomeTimeline, + recommendationTypes = Seq(RecommendationType.Tweet), + seedsWithWeights = query.seedsWithWeights, + maxResultsByType = Some(Map(RecommendationType.Tweet -> query.maxUtegCandidates)), + maxTweetAgeInMillis = Some(query.maxTweetAge.inMilliseconds), + excludedTweetIds = query.excludedTweetIds, + maxUserSocialProofSize = Some(UserTweetEntityGraphSimilarityEngine.MaxUserSocialProofSize), + maxTweetSocialProofSize = + Some(UserTweetEntityGraphSimilarityEngine.MaxTweetSocialProofSize), + minUserSocialProofSizes = Some(Map(RecommendationType.Tweet -> 1)), + tweetTypes = None, + socialProofTypes = query.socialProofTypes, + socialProofTypeUnions = None, + tweetAuthors = None, + maxEngagementAgeInMillis = None, + excludedTweetAuthors = None, + ) + + userTweetEntityGraph + .recommendTweets(recommendTweetEntityRequest) + .map { recommendTweetsResponse => + val candidates = recommendTweetsResponse.recommendations.flatMap { + case TweetRec(recommendation) => + Some( + TweetWithScoreAndSocialProof( + recommendation.tweetId, + recommendation.score, + recommendation.socialProofByType.toMap)) + case _ => None + } + Some(candidates) + } + } +} + +object UserTweetEntityGraphSimilarityEngine { + + private val MaxUserSocialProofSize = 10 + private val MaxTweetSocialProofSize = 10 + + def toSimilarityEngineInfo(score: Double): SimilarityEngineInfo = { + SimilarityEngineInfo( + similarityEngineType = SimilarityEngineType.Uteg, + modelId = None, + score = Some(score)) + } + + case class Query( + userId: UserId, + seedsWithWeights: Map[UserId, Double], + excludedTweetIds: Option[Seq[Long]] = None, + maxUtegCandidates: Int, + maxTweetAge: Duration, + socialProofTypes: Option[Seq[SocialProofType]]) + + def fromParams( + userId: UserId, + seedsWithWeights: Map[UserId, Double], + excludedTweetIds: Option[Seq[TweetId]] = None, + params: configapi.Params, + ): EngineQuery[Query] = { + EngineQuery( + Query( + userId = userId, + seedsWithWeights = seedsWithWeights, + excludedTweetIds = excludedTweetIds, + maxUtegCandidates = params(UtegTweetGlobalParams.MaxUtegCandidatesToRequestParam), + maxTweetAge = params(UtegTweetGlobalParams.CandidateRefreshSinceTimeOffsetHoursParam), + socialProofTypes = Some(Seq(SocialProofType.Favorite)) + ), + params + ) + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/source_signal/BUILD b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/source_signal/BUILD new file mode 100644 index 0000000000..37b0a7585f --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/source_signal/BUILD @@ -0,0 +1,32 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + strict_deps = True, + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/com/twitter/storehaus:core", + "3rdparty/jvm/javax/inject:javax.inject", + "3rdparty/src/jvm/com/twitter/storehaus:core", + "configapi/configapi-core", + "cr-mixer/server/src/main/scala/com/twitter/cr_mixer/config", + "cr-mixer/server/src/main/scala/com/twitter/cr_mixer/model", + "cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param", + "cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param/decider", + "cr-mixer/thrift/src/main/thrift:thrift-scala", + "decider/src/main/scala", + "finatra-internal/mtls-thriftmux/src/main/scala", + "finatra/inject/inject-core/src/main/scala", + "follow-recommendations-service/thrift/src/main/thrift:thrift-scala", + "frigate/frigate-common:base", + "frigate/frigate-common:util", + "frigate/frigate-common/src/main/scala/com/twitter/frigate/common/base", + "frigate/frigate-common/src/main/scala/com/twitter/frigate/common/candidate", + "src/scala/com/twitter/simclusters_v2/common", + "src/thrift/com/twitter/core_workflows/user_model:user_model-scala", + "src/thrift/com/twitter/hermit/stp:hermit-stp-scala", + "src/thrift/com/twitter/onboarding/relevance/coldstart_lookalike:coldstartlookalike-thrift-scala", + "src/thrift/com/twitter/simclusters_v2:simclusters_v2-thrift-scala", + "src/thrift/com/twitter/wtf/candidate:wtf-candidate-scala", + "user-signal-service/thrift/src/main/thrift:thrift-scala", + ], +) diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/source_signal/FrsSourceGraphFetcher.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/source_signal/FrsSourceGraphFetcher.scala new file mode 100644 index 0000000000..16162c67dc --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/source_signal/FrsSourceGraphFetcher.scala @@ -0,0 +1,54 @@ +package com.twitter.cr_mixer.source_signal + +import com.twitter.cr_mixer.config.TimeoutConfig +import com.twitter.cr_mixer.model.GraphSourceInfo +import com.twitter.cr_mixer.model.ModuleNames +import com.twitter.cr_mixer.param.FrsParams +import com.twitter.cr_mixer.source_signal.FrsStore.FrsQueryResult +import com.twitter.cr_mixer.source_signal.SourceFetcher.FetcherQuery +import com.twitter.cr_mixer.thriftscala.SourceType +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.storehaus.ReadableStore +import com.twitter.util.Future +import javax.inject.Inject +import javax.inject.Named +import javax.inject.Singleton + +/*** + * This store fetches user recommendations from FRS (go/frs) for a given userId + */ +@Singleton +case class FrsSourceGraphFetcher @Inject() ( + @Named(ModuleNames.FrsStore) frsStore: ReadableStore[FrsStore.Query, Seq[FrsQueryResult]], + override val timeoutConfig: TimeoutConfig, + globalStats: StatsReceiver) + extends SourceGraphFetcher { + + override protected val stats: StatsReceiver = globalStats.scope(identifier) + override protected val graphSourceType: SourceType = SourceType.FollowRecommendation + + override def isEnabled(query: FetcherQuery): Boolean = { + query.params(FrsParams.EnableSourceGraphParam) + } + + override def fetchAndProcess( + query: FetcherQuery, + ): Future[Option[GraphSourceInfo]] = { + + val rawSignals = trackPerItemStats(query)( + frsStore + .get( + FrsStore + .Query(query.userId, query.params(FrsParams.MaxConsumerSeedsNumParam))).map { + _.map { + _.map { v => (v.userId, v.score) } + } + } + ) + rawSignals.map { + _.map { userWithScores => + convertGraphSourceInfo(userWithScores) + } + } + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/source_signal/FrsSourceSignalFetcher.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/source_signal/FrsSourceSignalFetcher.scala new file mode 100644 index 0000000000..4e90693760 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/source_signal/FrsSourceSignalFetcher.scala @@ -0,0 +1,65 @@ +package com.twitter.cr_mixer.source_signal + +import com.twitter.cr_mixer.config.TimeoutConfig +import com.twitter.cr_mixer.model.ModuleNames +import com.twitter.cr_mixer.model.SourceInfo +import com.twitter.cr_mixer.param.FrsParams +import com.twitter.cr_mixer.param.GlobalParams +import com.twitter.cr_mixer.source_signal.FrsStore.FrsQueryResult +import com.twitter.cr_mixer.source_signal.SourceFetcher.FetcherQuery +import com.twitter.cr_mixer.thriftscala.SourceType +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.simclusters_v2.common.UserId +import com.twitter.simclusters_v2.thriftscala.InternalId +import com.twitter.storehaus.ReadableStore +import com.twitter.util.Future +import javax.inject.Singleton +import javax.inject.Inject +import javax.inject.Named + +@Singleton +case class FrsSourceSignalFetcher @Inject() ( + @Named(ModuleNames.FrsStore) frsStore: ReadableStore[FrsStore.Query, Seq[FrsQueryResult]], + override val timeoutConfig: TimeoutConfig, + globalStats: StatsReceiver) + extends SourceSignalFetcher { + + override protected val stats: StatsReceiver = globalStats.scope(identifier) + override type SignalConvertType = UserId + + override def isEnabled(query: FetcherQuery): Boolean = { + query.params(FrsParams.EnableSourceParam) + } + + override def fetchAndProcess(query: FetcherQuery): Future[Option[Seq[SourceInfo]]] = { + // Fetch raw signals + val rawSignals = frsStore + .get(FrsStore.Query(query.userId, query.params(GlobalParams.UnifiedMaxSourceKeyNum))) + .map { + _.map { + _.map { + _.userId + } + } + } + // Process signals + rawSignals.map { + _.map { frsUsers => + convertSourceInfo(SourceType.FollowRecommendation, frsUsers) + } + } + } + + override def convertSourceInfo( + sourceType: SourceType, + signals: Seq[SignalConvertType] + ): Seq[SourceInfo] = { + signals.map { signal => + SourceInfo( + sourceType = sourceType, + internalId = InternalId.UserId(signal), + sourceEventTime = None + ) + } + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/source_signal/FrsStore.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/source_signal/FrsStore.scala new file mode 100644 index 0000000000..0221bc318a --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/source_signal/FrsStore.scala @@ -0,0 +1,81 @@ +package com.twitter.cr_mixer.source_signal + +import com.twitter.cr_mixer.param.decider.CrMixerDecider +import com.twitter.cr_mixer.param.decider.DeciderConstants +import com.twitter.cr_mixer.source_signal.FrsStore.Query +import com.twitter.cr_mixer.source_signal.FrsStore.FrsQueryResult +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.follow_recommendations.thriftscala.ClientContext +import com.twitter.follow_recommendations.thriftscala.DisplayLocation +import com.twitter.follow_recommendations.thriftscala.FollowRecommendationsThriftService +import com.twitter.follow_recommendations.thriftscala.Recommendation +import com.twitter.follow_recommendations.thriftscala.RecommendationRequest +import com.twitter.storehaus.ReadableStore +import javax.inject.Singleton +import com.twitter.simclusters_v2.common.UserId +import com.twitter.util.Future + +@Singleton +case class FrsStore( + frsClient: FollowRecommendationsThriftService.MethodPerEndpoint, + statsReceiver: StatsReceiver, + decider: CrMixerDecider) + extends ReadableStore[Query, Seq[FrsQueryResult]] { + + override def get( + query: Query + ): Future[Option[Seq[FrsQueryResult]]] = { + if (decider.isAvailable(DeciderConstants.enableFRSTrafficDeciderKey)) { + val recommendationRequest = + buildFollowRecommendationRequest(query) + + frsClient + .getRecommendations(recommendationRequest).map { recommendationResponse => + Some(recommendationResponse.recommendations.collect { + case recommendation: Recommendation.User => + FrsQueryResult( + recommendation.user.userId, + recommendation.user.scoringDetails + .flatMap(_.score).getOrElse(0.0), + recommendation.user.scoringDetails + .flatMap(_.candidateSourceDetails.flatMap(_.primarySource)), + recommendation.user.scoringDetails + .flatMap(_.candidateSourceDetails.flatMap(_.candidateSourceScores)).map(_.toMap) + ) + }) + } + } else { + Future.None + } + } + + private def buildFollowRecommendationRequest( + query: Query + ): RecommendationRequest = { + RecommendationRequest( + clientContext = ClientContext( + userId = Some(query.userId), + countryCode = query.countryCodeOpt, + languageCode = query.languageCodeOpt), + displayLocation = query.displayLocation, + maxResults = Some(query.maxConsumerSeedsNum), + excludedIds = Some(query.excludedUserIds) + ) + } +} + +object FrsStore { + case class Query( + userId: UserId, + maxConsumerSeedsNum: Int, + displayLocation: DisplayLocation = DisplayLocation.ContentRecommender, + excludedUserIds: Seq[UserId] = Seq.empty, + languageCodeOpt: Option[String] = None, + countryCodeOpt: Option[String] = None) + + case class FrsQueryResult( + userId: UserId, + score: Double, + primarySource: Option[Int], + sourceWithScores: Option[Map[String, Double]]) +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/source_signal/RealGraphInSourceGraphFetcher.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/source_signal/RealGraphInSourceGraphFetcher.scala new file mode 100644 index 0000000000..ac708d0bbe --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/source_signal/RealGraphInSourceGraphFetcher.scala @@ -0,0 +1,55 @@ +package com.twitter.cr_mixer.source_signal + +import com.twitter.cr_mixer.config.TimeoutConfig +import com.twitter.cr_mixer.model.GraphSourceInfo +import com.twitter.cr_mixer.model.ModuleNames +import com.twitter.cr_mixer.param.RealGraphInParams +import com.twitter.cr_mixer.source_signal.SourceFetcher.FetcherQuery +import com.twitter.cr_mixer.thriftscala.SourceType +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.simclusters_v2.common.UserId +import com.twitter.storehaus.ReadableStore +import com.twitter.util.Future +import com.twitter.wtf.candidate.thriftscala.CandidateSeq +import javax.inject.Inject +import javax.inject.Named +import javax.inject.Singleton + +/** + * This store fetch user recommendations from In-Network RealGraph (go/realgraph) for a given userId + */ +@Singleton +case class RealGraphInSourceGraphFetcher @Inject() ( + @Named(ModuleNames.RealGraphInStore) realGraphStoreMh: ReadableStore[UserId, CandidateSeq], + override val timeoutConfig: TimeoutConfig, + globalStats: StatsReceiver) + extends SourceGraphFetcher { + + override protected val stats: StatsReceiver = globalStats.scope(identifier) + override protected val graphSourceType: SourceType = SourceType.RealGraphIn + + override def isEnabled(query: FetcherQuery): Boolean = { + query.params(RealGraphInParams.EnableSourceGraphParam) + } + + override def fetchAndProcess( + query: FetcherQuery, + ): Future[Option[GraphSourceInfo]] = { + val rawSignals = trackPerItemStats(query)( + realGraphStoreMh.get(query.userId).map { + _.map { candidateSeq => + candidateSeq.candidates + .map { candidate => + // Bundle the userId with its score + (candidate.userId, candidate.score) + } + } + } + ) + rawSignals.map { + _.map { userWithScores => + convertGraphSourceInfo(userWithScores) + } + } + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/source_signal/RealGraphOonSourceGraphFetcher.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/source_signal/RealGraphOonSourceGraphFetcher.scala new file mode 100644 index 0000000000..e03d140a4a --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/source_signal/RealGraphOonSourceGraphFetcher.scala @@ -0,0 +1,55 @@ +package com.twitter.cr_mixer.source_signal + +import com.twitter.cr_mixer.config.TimeoutConfig +import com.twitter.cr_mixer.model.GraphSourceInfo +import com.twitter.cr_mixer.model.ModuleNames +import com.twitter.cr_mixer.param.RealGraphOonParams +import com.twitter.cr_mixer.source_signal.SourceFetcher.FetcherQuery +import com.twitter.cr_mixer.thriftscala.SourceType +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.simclusters_v2.common.UserId +import com.twitter.storehaus.ReadableStore +import com.twitter.util.Future +import com.twitter.wtf.candidate.thriftscala.CandidateSeq +import javax.inject.Inject +import javax.inject.Named +import javax.inject.Singleton + +/** + * This store fetch user recommendations from RealGraphOON (go/realgraph) for a given userId + */ +@Singleton +case class RealGraphOonSourceGraphFetcher @Inject() ( + @Named(ModuleNames.RealGraphOonStore) realGraphOonStore: ReadableStore[UserId, CandidateSeq], + override val timeoutConfig: TimeoutConfig, + globalStats: StatsReceiver) + extends SourceGraphFetcher { + + override protected val stats: StatsReceiver = globalStats.scope(identifier) + override protected val graphSourceType: SourceType = SourceType.RealGraphOon + + override def isEnabled(query: FetcherQuery): Boolean = { + query.params(RealGraphOonParams.EnableSourceGraphParam) + } + + override def fetchAndProcess( + query: FetcherQuery, + ): Future[Option[GraphSourceInfo]] = { + val rawSignals = trackPerItemStats(query)( + realGraphOonStore.get(query.userId).map { + _.map { candidateSeq => + candidateSeq.candidates + .map { candidate => + // Bundle the userId with its score + (candidate.userId, candidate.score) + }.take(query.params(RealGraphOonParams.MaxConsumerSeedsNumParam)) + } + } + ) + rawSignals.map { + _.map { userWithScores => + convertGraphSourceInfo(userWithScores) + } + } + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/source_signal/SourceFetcher.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/source_signal/SourceFetcher.scala new file mode 100644 index 0000000000..4fa4dfb937 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/source_signal/SourceFetcher.scala @@ -0,0 +1,101 @@ +package com.twitter.cr_mixer.source_signal + +import com.twitter.core_workflows.user_model.thriftscala.UserState +import com.twitter.cr_mixer.config.TimeoutConfig +import com.twitter.cr_mixer.source_signal.SourceFetcher.FetcherQuery +import com.twitter.simclusters_v2.common.UserId +import com.twitter.timelines.configapi.Params +import com.twitter.cr_mixer.thriftscala.{Product => TProduct} +import com.twitter.finagle.GlobalRequestTimeoutException +import com.twitter.finagle.mux.ClientDiscardedRequestException +import com.twitter.finagle.mux.ServerApplicationError +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.storehaus.ReadableStore +import com.twitter.util.Future +import com.twitter.util.TimeoutException +import org.apache.thrift.TApplicationException +import com.twitter.util.logging.Logging + +/** + * A SourceFetcher is a trait which, given a [[FetcherQuery]], returns [[ResultType]] + * The main purposes of a SourceFetcher is to provide a consistent interface for source fetch + * logic, and provides default functions, including: + * - Identification + * - Observability + * - Timeout settings + * - Exception Handling + */ +trait SourceFetcher[ResultType] extends ReadableStore[FetcherQuery, ResultType] with Logging { + + protected final val timer = com.twitter.finagle.util.DefaultTimer + protected final def identifier: String = this.getClass.getSimpleName + protected def stats: StatsReceiver + protected def timeoutConfig: TimeoutConfig + + /*** + * Use FeatureSwitch to decide if a specific source is enabled. + */ + def isEnabled(query: FetcherQuery): Boolean + + /*** + * This function fetches the raw sources and process them. + * Custom stats tracking can be added depending on the type of ResultType + */ + def fetchAndProcess( + query: FetcherQuery, + ): Future[Option[ResultType]] + + /*** + * Side-effect function to track stats for signal fetching and processing. + */ + def trackStats( + query: FetcherQuery + )( + func: => Future[Option[ResultType]] + ): Future[Option[ResultType]] + + /*** + * This function is called by the top level class to fetch sources. It executes the pipeline to + * fetch raw data, process and transform the sources. Exceptions, Stats, and timeout control are + * handled here. + */ + override def get( + query: FetcherQuery + ): Future[Option[ResultType]] = { + val scopedStats = stats.scope(query.product.originalName) + if (isEnabled(query)) { + scopedStats.counter("gate_enabled").incr() + trackStats(query)(fetchAndProcess(query)) + .raiseWithin(timeoutConfig.signalFetchTimeout)(timer) + .onFailure { e => + scopedStats.scope("exceptions").counter(e.getClass.getSimpleName).incr() + } + .rescue { + case _: TimeoutException | _: GlobalRequestTimeoutException | _: TApplicationException | + _: ClientDiscardedRequestException | + _: ServerApplicationError // TApplicationException inside + => + Future.None + case e => + logger.info(e) + Future.None + } + } else { + scopedStats.counter("gate_disabled").incr() + Future.None + } + } +} + +object SourceFetcher { + + /*** + * Every SourceFetcher all share the same input: FetcherQuery + */ + case class FetcherQuery( + userId: UserId, + product: TProduct, + userState: UserState, + params: Params) + +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/source_signal/SourceGraphFetcher.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/source_signal/SourceGraphFetcher.scala new file mode 100644 index 0000000000..ac33d91e9e --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/source_signal/SourceGraphFetcher.scala @@ -0,0 +1,70 @@ +package com.twitter.cr_mixer.source_signal + +import com.twitter.cr_mixer.model.GraphSourceInfo +import com.twitter.cr_mixer.source_signal.SourceFetcher.FetcherQuery +import com.twitter.cr_mixer.thriftscala.SourceType +import com.twitter.frigate.common.util.StatsUtil +import com.twitter.simclusters_v2.common.UserId +import com.twitter.util.Future + +/*** + * A SourceGraphFetcher is a trait that extends from `SourceFetcher` + * and is specialized in tackling User Graph (eg., RealGraphOon, FRS) fetch. + * + * The [[ResultType]] of a SourceGraphFetcher is a `GraphSourceInfo` which contains a userSeedSet. + * When we pass in userId, the underlying store returns one GraphSourceInfo. + */ +trait SourceGraphFetcher extends SourceFetcher[GraphSourceInfo] { + protected final val DefaultSeedScore = 1.0 + protected def graphSourceType: SourceType + + /*** + * RawDataType contains a consumers seed UserId and a score (weight) + */ + protected type RawDataType = (UserId, Double) + + def trackStats( + query: FetcherQuery + )( + func: => Future[Option[GraphSourceInfo]] + ): Future[Option[GraphSourceInfo]] = { + val productScopedStats = stats.scope(query.product.originalName) + val productUserStateScopedStats = productScopedStats.scope(query.userState.toString) + StatsUtil + .trackOptionStats(productScopedStats) { + StatsUtil + .trackOptionStats(productUserStateScopedStats) { + func + } + } + } + + // Track per item stats on the fetched graph results + def trackPerItemStats( + query: FetcherQuery + )( + func: => Future[Option[Seq[RawDataType]]] + ): Future[Option[Seq[RawDataType]]] = { + val productScopedStats = stats.scope(query.product.originalName) + val productUserStateScopedStats = productScopedStats.scope(query.userState.toString) + StatsUtil.trackOptionItemsStats(productScopedStats) { + StatsUtil.trackOptionItemsStats(productUserStateScopedStats) { + func + } + } + } + + /*** + * Convert Seq[RawDataType] into GraphSourceInfo + */ + protected final def convertGraphSourceInfo( + userWithScores: Seq[RawDataType] + ): GraphSourceInfo = { + GraphSourceInfo( + sourceType = graphSourceType, + seedWithScores = userWithScores.map { userWithScore => + userWithScore._1 -> userWithScore._2 + }.toMap + ) + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/source_signal/SourceInfoRouter.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/source_signal/SourceInfoRouter.scala new file mode 100644 index 0000000000..5942fb8e4b --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/source_signal/SourceInfoRouter.scala @@ -0,0 +1,68 @@ +package com.twitter.cr_mixer.source_signal + +import com.twitter.core_workflows.user_model.thriftscala.UserState +import com.twitter.cr_mixer.model.GraphSourceInfo +import com.twitter.cr_mixer.model.SourceInfo +import com.twitter.cr_mixer.source_signal.SourceFetcher.FetcherQuery +import com.twitter.cr_mixer.thriftscala.SourceType +import com.twitter.cr_mixer.thriftscala.{Product => TProduct} +import com.twitter.simclusters_v2.common.UserId +import com.twitter.timelines.configapi +import com.twitter.util.Future +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +case class SourceInfoRouter @Inject() ( + ussSourceSignalFetcher: UssSourceSignalFetcher, + frsSourceSignalFetcher: FrsSourceSignalFetcher, + frsSourceGraphFetcher: FrsSourceGraphFetcher, + realGraphOonSourceGraphFetcher: RealGraphOonSourceGraphFetcher, + realGraphInSourceGraphFetcher: RealGraphInSourceGraphFetcher, +) { + + def get( + userId: UserId, + product: TProduct, + userState: UserState, + params: configapi.Params + ): Future[(Set[SourceInfo], Map[String, Option[GraphSourceInfo]])] = { + + val fetcherQuery = FetcherQuery(userId, product, userState, params) + Future.join( + getSourceSignals(fetcherQuery), + getSourceGraphs(fetcherQuery) + ) + } + + private def getSourceSignals( + fetcherQuery: FetcherQuery + ): Future[Set[SourceInfo]] = { + Future + .join( + ussSourceSignalFetcher.get(fetcherQuery), + frsSourceSignalFetcher.get(fetcherQuery)).map { + case (ussSignalsOpt, frsSignalsOpt) => + (ussSignalsOpt.getOrElse(Seq.empty) ++ frsSignalsOpt.getOrElse(Seq.empty)).toSet + } + } + + private def getSourceGraphs( + fetcherQuery: FetcherQuery + ): Future[Map[String, Option[GraphSourceInfo]]] = { + + Future + .join( + frsSourceGraphFetcher.get(fetcherQuery), + realGraphOonSourceGraphFetcher.get(fetcherQuery), + realGraphInSourceGraphFetcher.get(fetcherQuery) + ).map { + case (frsGraphOpt, realGraphOonGraphOpt, realGraphInGraphOpt) => + Map( + SourceType.FollowRecommendation.name -> frsGraphOpt, + SourceType.RealGraphOon.name -> realGraphOonGraphOpt, + SourceType.RealGraphIn.name -> realGraphInGraphOpt, + ) + } + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/source_signal/SourceSignalFetcher.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/source_signal/SourceSignalFetcher.scala new file mode 100644 index 0000000000..01d302661e --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/source_signal/SourceSignalFetcher.scala @@ -0,0 +1,45 @@ +package com.twitter.cr_mixer.source_signal + +import com.twitter.cr_mixer.model.SourceInfo +import com.twitter.cr_mixer.source_signal.SourceFetcher.FetcherQuery +import com.twitter.cr_mixer.thriftscala.SourceType +import com.twitter.frigate.common.util.StatsUtil +import com.twitter.util.Future + +/*** + * A SourceSignalFetcher is a trait that extends from `SourceFetcher` + * and is specialized in tackling Signals (eg., USS, FRS) fetch. + * Currently, we define Signals as (but not limited to) a set of past engagements that + * the user makes, such as RecentFav, RecentFollow, etc. + * + * The [[ResultType]] of a SourceSignalFetcher is `Seq[SourceInfo]`. When we pass in userId, + * the underlying store returns a list of signals. + */ +trait SourceSignalFetcher extends SourceFetcher[Seq[SourceInfo]] { + + protected type SignalConvertType + + def trackStats( + query: FetcherQuery + )( + func: => Future[Option[Seq[SourceInfo]]] + ): Future[Option[Seq[SourceInfo]]] = { + val productScopedStats = stats.scope(query.product.originalName) + val productUserStateScopedStats = productScopedStats.scope(query.userState.toString) + StatsUtil + .trackOptionItemsStats(productScopedStats) { + StatsUtil + .trackOptionItemsStats(productUserStateScopedStats) { + func + } + } + } + + /*** + * Convert a list of Signals of type [[SignalConvertType]] into SourceInfo + */ + def convertSourceInfo( + sourceType: SourceType, + signals: Seq[SignalConvertType] + ): Seq[SourceInfo] +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/source_signal/UssSourceSignalFetcher.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/source_signal/UssSourceSignalFetcher.scala new file mode 100644 index 0000000000..dcce3e94e5 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/source_signal/UssSourceSignalFetcher.scala @@ -0,0 +1,160 @@ +package com.twitter.cr_mixer.source_signal + +import com.twitter.cr_mixer.config.TimeoutConfig +import com.twitter.cr_mixer.model.ModuleNames +import com.twitter.cr_mixer.model.SourceInfo +import com.twitter.cr_mixer.thriftscala.SourceType +import com.twitter.cr_mixer.source_signal.SourceFetcher.FetcherQuery +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.storehaus.ReadableStore +import com.twitter.usersignalservice.thriftscala.{Signal => UssSignal} +import com.twitter.usersignalservice.thriftscala.SignalType +import com.twitter.frigate.common.util.StatsUtil.Size +import com.twitter.frigate.common.util.StatsUtil.Success +import com.twitter.frigate.common.util.StatsUtil.Empty +import com.twitter.util.Future +import com.twitter.util.Time +import javax.inject.Singleton +import javax.inject.Inject +import javax.inject.Named + +@Singleton +case class UssSourceSignalFetcher @Inject() ( + @Named(ModuleNames.UssStore) ussStore: ReadableStore[UssStore.Query, Seq[ + (SignalType, Seq[UssSignal]) + ]], + override val timeoutConfig: TimeoutConfig, + globalStats: StatsReceiver) + extends SourceSignalFetcher { + + override protected val stats: StatsReceiver = globalStats.scope(identifier) + override type SignalConvertType = UssSignal + + // always enable USS call. We have fine-grained FS to decider which signal to fetch + override def isEnabled(query: FetcherQuery): Boolean = true + + override def fetchAndProcess( + query: FetcherQuery, + ): Future[Option[Seq[SourceInfo]]] = { + // Fetch raw signals + val rawSignals = ussStore.get(UssStore.Query(query.userId, query.params, query.product)).map { + _.map { + _.map { + case (signalType, signals) => + trackUssSignalStatsPerSignalType(query, signalType, signals) + (signalType, signals) + } + } + } + + /** + * Process signals: + * Transform a Seq of USS Signals with signalType specified to a Seq of SourceInfo + * We do case match to make sure the SignalType can correctly map to a SourceType defined in CrMixer + * and it should be simplified. + */ + rawSignals.map { + _.map { nestedSignal => + val sourceInfoList = nestedSignal.flatMap { + case (signalType, ussSignals) => + signalType match { + case SignalType.TweetFavorite => + convertSourceInfo(sourceType = SourceType.TweetFavorite, signals = ussSignals) + case SignalType.Retweet => + convertSourceInfo(sourceType = SourceType.Retweet, signals = ussSignals) + case SignalType.Reply => + convertSourceInfo(sourceType = SourceType.Reply, signals = ussSignals) + case SignalType.OriginalTweet => + convertSourceInfo(sourceType = SourceType.OriginalTweet, signals = ussSignals) + case SignalType.AccountFollow => + convertSourceInfo(sourceType = SourceType.UserFollow, signals = ussSignals) + case SignalType.RepeatedProfileVisit180dMinVisit6V1 | + SignalType.RepeatedProfileVisit90dMinVisit6V1 | + SignalType.RepeatedProfileVisit14dMinVisit2V1 => + convertSourceInfo( + sourceType = SourceType.UserRepeatedProfileVisit, + signals = ussSignals) + case SignalType.NotificationOpenAndClickV1 => + convertSourceInfo(sourceType = SourceType.NotificationClick, signals = ussSignals) + case SignalType.TweetShareV1 => + convertSourceInfo(sourceType = SourceType.TweetShare, signals = ussSignals) + case SignalType.RealGraphOon => + convertSourceInfo(sourceType = SourceType.RealGraphOon, signals = ussSignals) + case SignalType.GoodTweetClick | SignalType.GoodTweetClick5s | + SignalType.GoodTweetClick10s | SignalType.GoodTweetClick30s => + convertSourceInfo(sourceType = SourceType.GoodTweetClick, signals = ussSignals) + case SignalType.VideoView90dPlayback50V1 => + convertSourceInfo( + sourceType = SourceType.VideoTweetPlayback50, + signals = ussSignals) + case SignalType.VideoView90dQualityV1 => + convertSourceInfo( + sourceType = SourceType.VideoTweetQualityView, + signals = ussSignals) + case SignalType.GoodProfileClick | SignalType.GoodProfileClick20s | + SignalType.GoodProfileClick30s => + convertSourceInfo(sourceType = SourceType.GoodProfileClick, signals = ussSignals) + // negative signals + case SignalType.AccountBlock => + convertSourceInfo(sourceType = SourceType.AccountBlock, signals = ussSignals) + case SignalType.AccountMute => + convertSourceInfo(sourceType = SourceType.AccountMute, signals = ussSignals) + case SignalType.TweetReport => + convertSourceInfo(sourceType = SourceType.TweetReport, signals = ussSignals) + case SignalType.TweetDontLike => + convertSourceInfo(sourceType = SourceType.TweetDontLike, signals = ussSignals) + // Aggregated Signals + case SignalType.TweetBasedUnifiedEngagementWeightedSignal | + SignalType.TweetBasedUnifiedUniformSignal => + convertSourceInfo(sourceType = SourceType.TweetAggregation, signals = ussSignals) + case SignalType.ProducerBasedUnifiedEngagementWeightedSignal | + SignalType.ProducerBasedUnifiedUniformSignal => + convertSourceInfo(sourceType = SourceType.ProducerAggregation, signals = ussSignals) + + // Default + case _ => + Seq.empty[SourceInfo] + } + } + sourceInfoList + } + } + } + + override def convertSourceInfo( + sourceType: SourceType, + signals: Seq[SignalConvertType] + ): Seq[SourceInfo] = { + signals.map { signal => + SourceInfo( + sourceType = sourceType, + internalId = signal.targetInternalId.getOrElse( + throw new IllegalArgumentException( + s"${sourceType.toString} Signal does not have internalId")), + sourceEventTime = + if (signal.timestamp == 0L) None else Some(Time.fromMilliseconds(signal.timestamp)) + ) + } + } + + private def trackUssSignalStatsPerSignalType( + query: FetcherQuery, + signalType: SignalType, + ussSignals: Seq[UssSignal] + ): Unit = { + val productScopedStats = stats.scope(query.product.originalName) + val productUserStateScopedStats = productScopedStats.scope(query.userState.toString) + val productStats = productScopedStats.scope(signalType.toString) + val productUserStateStats = productUserStateScopedStats.scope(signalType.toString) + + productStats.counter(Success).incr() + productUserStateStats.counter(Success).incr() + val size = ussSignals.size + productStats.stat(Size).add(size) + productUserStateStats.stat(Size).add(size) + if (size == 0) { + productStats.counter(Empty).incr() + productUserStateStats.counter(Empty).incr() + } + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/source_signal/UssStore.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/source_signal/UssStore.scala new file mode 100644 index 0000000000..02f0287f5c --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/source_signal/UssStore.scala @@ -0,0 +1,209 @@ +package com.twitter.cr_mixer.source_signal + +import com.twitter.cr_mixer.param.GlobalParams +import com.twitter.cr_mixer.param.GoodProfileClickParams +import com.twitter.cr_mixer.param.GoodTweetClickParams +import com.twitter.cr_mixer.param.RealGraphOonParams +import com.twitter.cr_mixer.param.RecentFollowsParams +import com.twitter.cr_mixer.param.RecentNegativeSignalParams +import com.twitter.cr_mixer.param.RecentNotificationsParams +import com.twitter.cr_mixer.param.RecentOriginalTweetsParams +import com.twitter.cr_mixer.param.RecentReplyTweetsParams +import com.twitter.cr_mixer.param.RecentRetweetsParams +import com.twitter.cr_mixer.param.RecentTweetFavoritesParams +import com.twitter.cr_mixer.param.RepeatedProfileVisitsParams +import com.twitter.cr_mixer.param.TweetSharesParams +import com.twitter.cr_mixer.param.UnifiedUSSSignalParams +import com.twitter.cr_mixer.param.VideoViewTweetsParams +import com.twitter.cr_mixer.source_signal.UssStore.Query +import com.twitter.cr_mixer.thriftscala.SourceType +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.simclusters_v2.common.UserId +import com.twitter.storehaus.ReadableStore +import com.twitter.usersignalservice.thriftscala.{Signal => UssSignal} +import com.twitter.usersignalservice.thriftscala.SignalType +import javax.inject.Singleton +import com.twitter.timelines.configapi +import com.twitter.timelines.configapi.Params +import com.twitter.usersignalservice.thriftscala.BatchSignalRequest +import com.twitter.usersignalservice.thriftscala.BatchSignalResponse +import com.twitter.usersignalservice.thriftscala.SignalRequest +import com.twitter.util.Future +import com.twitter.cr_mixer.thriftscala.Product +import com.twitter.usersignalservice.thriftscala.ClientIdentifier + +@Singleton +case class UssStore( + stratoStore: ReadableStore[BatchSignalRequest, BatchSignalResponse], + statsReceiver: StatsReceiver) + extends ReadableStore[Query, Seq[(SignalType, Seq[UssSignal])]] { + + import com.twitter.cr_mixer.source_signal.UssStore._ + + override def get(query: Query): Future[Option[Seq[(SignalType, Seq[UssSignal])]]] = { + val ussClientIdentifier = query.product match { + case Product.Home => + ClientIdentifier.CrMixerHome + case Product.Notifications => + ClientIdentifier.CrMixerNotifications + case Product.Email => + ClientIdentifier.CrMixerEmail + case _ => + ClientIdentifier.Unknown + } + val batchSignalRequest = + BatchSignalRequest( + query.userId, + buildUserSignalServiceRequests(query.params), + Some(ussClientIdentifier)) + + stratoStore + .get(batchSignalRequest) + .map { + _.map { batchSignalResponse => + batchSignalResponse.signalResponse.toSeq.map { + case (signalType, ussSignals) => + (signalType, ussSignals) + } + } + } + } + + private def buildUserSignalServiceRequests( + param: Params, + ): Seq[SignalRequest] = { + val unifiedMaxSourceKeyNum = param(GlobalParams.UnifiedMaxSourceKeyNum) + val goodTweetClickMaxSignalNum = param(GoodTweetClickParams.MaxSignalNumParam) + val aggrTweetMaxSourceKeyNum = param(UnifiedUSSSignalParams.UnifiedTweetSourceNumberParam) + val aggrProducerMaxSourceKeyNum = param(UnifiedUSSSignalParams.UnifiedProducerSourceNumberParam) + + val maybeRecentTweetFavorite = + if (param(RecentTweetFavoritesParams.EnableSourceParam)) + Some(SignalRequest(Some(unifiedMaxSourceKeyNum), SignalType.TweetFavorite)) + else None + val maybeRecentRetweet = + if (param(RecentRetweetsParams.EnableSourceParam)) + Some(SignalRequest(Some(unifiedMaxSourceKeyNum), SignalType.Retweet)) + else None + val maybeRecentReply = + if (param(RecentReplyTweetsParams.EnableSourceParam)) + Some(SignalRequest(Some(unifiedMaxSourceKeyNum), SignalType.Reply)) + else None + val maybeRecentOriginalTweet = + if (param(RecentOriginalTweetsParams.EnableSourceParam)) + Some(SignalRequest(Some(unifiedMaxSourceKeyNum), SignalType.OriginalTweet)) + else None + val maybeRecentFollow = + if (param(RecentFollowsParams.EnableSourceParam)) + Some(SignalRequest(Some(unifiedMaxSourceKeyNum), SignalType.AccountFollow)) + else None + val maybeRepeatedProfileVisits = + if (param(RepeatedProfileVisitsParams.EnableSourceParam)) + Some( + SignalRequest( + Some(unifiedMaxSourceKeyNum), + param(RepeatedProfileVisitsParams.ProfileMinVisitType).signalType)) + else None + val maybeRecentNotifications = + if (param(RecentNotificationsParams.EnableSourceParam)) + Some(SignalRequest(Some(unifiedMaxSourceKeyNum), SignalType.NotificationOpenAndClickV1)) + else None + val maybeTweetShares = + if (param(TweetSharesParams.EnableSourceParam)) { + Some(SignalRequest(Some(unifiedMaxSourceKeyNum), SignalType.TweetShareV1)) + } else None + val maybeRealGraphOon = + if (param(RealGraphOonParams.EnableSourceParam)) { + Some(SignalRequest(Some(unifiedMaxSourceKeyNum), SignalType.RealGraphOon)) + } else None + + val maybeGoodTweetClick = + if (param(GoodTweetClickParams.EnableSourceParam)) + Some( + SignalRequest( + Some(goodTweetClickMaxSignalNum), + param(GoodTweetClickParams.ClickMinDwellTimeType).signalType)) + else None + val maybeVideoViewTweets = + if (param(VideoViewTweetsParams.EnableSourceParam)) { + Some( + SignalRequest( + Some(unifiedMaxSourceKeyNum), + param(VideoViewTweetsParams.VideoViewTweetTypeParam).signalType)) + } else None + val maybeGoodProfileClick = + if (param(GoodProfileClickParams.EnableSourceParam)) + Some( + SignalRequest( + Some(unifiedMaxSourceKeyNum), + param(GoodProfileClickParams.ClickMinDwellTimeType).signalType)) + else None + val maybeAggTweetSignal = + if (param(UnifiedUSSSignalParams.EnableTweetAggSourceParam)) + Some( + SignalRequest( + Some(aggrTweetMaxSourceKeyNum), + param(UnifiedUSSSignalParams.TweetAggTypeParam).signalType + ) + ) + else None + val maybeAggProducerSignal = + if (param(UnifiedUSSSignalParams.EnableProducerAggSourceParam)) + Some( + SignalRequest( + Some(aggrProducerMaxSourceKeyNum), + param(UnifiedUSSSignalParams.ProducerAggTypeParam).signalType + ) + ) + else None + + // negative signals + val maybeNegativeSignals = if (param(RecentNegativeSignalParams.EnableSourceParam)) { + EnabledNegativeSignalTypes + .map(negativeSignal => SignalRequest(Some(unifiedMaxSourceKeyNum), negativeSignal)).toSeq + } else Seq.empty + + val allPositiveSignals = + if (param(UnifiedUSSSignalParams.ReplaceIndividualUSSSourcesParam)) + Seq( + maybeRecentOriginalTweet, + maybeRecentNotifications, + maybeRealGraphOon, + maybeGoodTweetClick, + maybeGoodProfileClick, + maybeAggProducerSignal, + maybeAggTweetSignal, + ) + else + Seq( + maybeRecentTweetFavorite, + maybeRecentRetweet, + maybeRecentReply, + maybeRecentOriginalTweet, + maybeRecentFollow, + maybeRepeatedProfileVisits, + maybeRecentNotifications, + maybeTweetShares, + maybeRealGraphOon, + maybeGoodTweetClick, + maybeVideoViewTweets, + maybeGoodProfileClick, + maybeAggProducerSignal, + maybeAggTweetSignal, + ) + allPositiveSignals.flatten ++ maybeNegativeSignals + } + +} + +object UssStore { + case class Query( + userId: UserId, + params: configapi.Params, + product: Product) + + val EnabledNegativeSourceTypes: Set[SourceType] = + Set(SourceType.AccountBlock, SourceType.AccountMute) + private val EnabledNegativeSignalTypes: Set[SignalType] = + Set(SignalType.AccountBlock, SignalType.AccountMute) +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/util/BUILD b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/util/BUILD new file mode 100644 index 0000000000..46c71420a0 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/util/BUILD @@ -0,0 +1,29 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + strict_deps = True, + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/com/twitter/storehaus:core", + "3rdparty/jvm/org/lz4:lz4-java", + "3rdparty/src/jvm/com/twitter/storehaus:core", + "configapi/configapi-core", + "content-recommender/thrift/src/main/thrift:thrift-scala", + "cr-mixer/server/src/main/scala/com/twitter/cr_mixer/config", + "cr-mixer/server/src/main/scala/com/twitter/cr_mixer/model", + "cr-mixer/server/src/main/scala/com/twitter/cr_mixer/param", + "cr-mixer/thrift/src/main/thrift:thrift-scala", + "finatra/inject/inject-core/src/main/scala", + "frigate/frigate-common/src/main/scala/com/twitter/frigate/common/util:stats_util", + "relevance-platform/src/main/scala/com/twitter/relevance_platform/common/stats", + "src/java/com/twitter/search/common/schema/base", + "src/java/com/twitter/search/common/schema/earlybird", + "src/java/com/twitter/search/queryparser/query:core-query-nodes", + "src/java/com/twitter/search/queryparser/query/search:search-query-nodes", + "src/scala/com/twitter/simclusters_v2/common", + "src/thrift/com/twitter/core_workflows/user_model:user_model-scala", + "src/thrift/com/twitter/search:earlybird-scala", + "src/thrift/com/twitter/search/common:ranking-scala", + "src/thrift/com/twitter/simclusters_v2:simclusters_v2-thrift-scala", + ], +) diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/util/CandidateGenerationKeyUtil.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/util/CandidateGenerationKeyUtil.scala new file mode 100644 index 0000000000..fd698f6d9b --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/util/CandidateGenerationKeyUtil.scala @@ -0,0 +1,39 @@ +package com.twitter.cr_mixer.util + +import com.twitter.cr_mixer.model.CandidateGenerationInfo +import com.twitter.cr_mixer.model.SourceInfo +import com.twitter.cr_mixer.thriftscala.CandidateGenerationKey +import com.twitter.cr_mixer.thriftscala.SimilarityEngine +import com.twitter.cr_mixer.thriftscala.SourceType +import com.twitter.simclusters_v2.common.UserId +import com.twitter.simclusters_v2.thriftscala.InternalId +import com.twitter.util.Time + +object CandidateGenerationKeyUtil { + private val PlaceholderUserId = 0L // this default value will not be used + + private val DefaultSourceInfo: SourceInfo = SourceInfo( + sourceType = SourceType.RequestUserId, + sourceEventTime = None, + internalId = InternalId.UserId(PlaceholderUserId) + ) + + def toThrift( + candidateGenerationInfo: CandidateGenerationInfo, + requestUserId: UserId + ): CandidateGenerationKey = { + CandidateGenerationKey( + sourceType = candidateGenerationInfo.sourceInfoOpt.getOrElse(DefaultSourceInfo).sourceType, + sourceEventTime = candidateGenerationInfo.sourceInfoOpt + .getOrElse(DefaultSourceInfo).sourceEventTime.getOrElse(Time.fromMilliseconds(0L)).inMillis, + id = candidateGenerationInfo.sourceInfoOpt + .map(_.internalId).getOrElse(InternalId.UserId(requestUserId)), + modelId = candidateGenerationInfo.similarityEngineInfo.modelId.getOrElse(""), + similarityEngineType = + Some(candidateGenerationInfo.similarityEngineInfo.similarityEngineType), + contributingSimilarityEngine = + Some(candidateGenerationInfo.contributingSimilarityEngines.map(se => + SimilarityEngine(se.similarityEngineType, se.modelId, se.score))) + ) + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/util/CountWeightedInterleaveUtil.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/util/CountWeightedInterleaveUtil.scala new file mode 100644 index 0000000000..bfae900571 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/util/CountWeightedInterleaveUtil.scala @@ -0,0 +1,180 @@ +package com.twitter.cr_mixer.util + +import com.twitter.cr_mixer.model.Candidate +import com.twitter.cr_mixer.model.InitialCandidate +import com.twitter.cr_mixer.model.RankedCandidate +import com.twitter.cr_mixer.model.SourceInfo +import com.twitter.cr_mixer.param.BlenderParams.BlendGroupingMethodEnum +import com.twitter.cr_mixer.thriftscala.SimilarityEngineType +import com.twitter.simclusters_v2.thriftscala.InternalId + +object CountWeightedInterleaveUtil { + + /** + * Grouping key for interleaving candidates + * + * @param sourceInfoOpt optional SourceInfo, containing the source information + * @param similarityEngineTypeOpt optional SimilarityEngineType, containing similarity engine + * information + * @param modelIdOpt optional modelId, containing the model ID + * @param authorIdOpt optional authorId, containing the tweet author ID + * @param groupIdOpt optional groupId, containing the ID corresponding to the blending group + */ + case class GroupingKey( + sourceInfoOpt: Option[SourceInfo], + similarityEngineTypeOpt: Option[SimilarityEngineType], + modelIdOpt: Option[String], + authorIdOpt: Option[Long], + groupIdOpt: Option[Int]) + + /** + * Converts candidates to grouping key based upon the feature that we interleave with. + */ + def toGroupingKey[CandidateType <: Candidate]( + candidate: CandidateType, + interleaveFeature: Option[BlendGroupingMethodEnum.Value], + groupId: Option[Int], + ): GroupingKey = { + val grouping: GroupingKey = candidate match { + case c: RankedCandidate => + interleaveFeature.getOrElse(BlendGroupingMethodEnum.SourceKeyDefault) match { + case BlendGroupingMethodEnum.SourceKeyDefault => + GroupingKey( + sourceInfoOpt = c.reasonChosen.sourceInfoOpt, + similarityEngineTypeOpt = + Some(c.reasonChosen.similarityEngineInfo.similarityEngineType), + modelIdOpt = c.reasonChosen.similarityEngineInfo.modelId, + authorIdOpt = None, + groupIdOpt = groupId + ) + // Some candidate sources don't have a sourceType, so it defaults to similarityEngine + case BlendGroupingMethodEnum.SourceTypeSimilarityEngine => + val sourceInfoOpt = c.reasonChosen.sourceInfoOpt.map(_.sourceType).map { sourceType => + SourceInfo( + sourceType = sourceType, + internalId = InternalId.UserId(0), + sourceEventTime = None) + } + GroupingKey( + sourceInfoOpt = sourceInfoOpt, + similarityEngineTypeOpt = + Some(c.reasonChosen.similarityEngineInfo.similarityEngineType), + modelIdOpt = c.reasonChosen.similarityEngineInfo.modelId, + authorIdOpt = None, + groupIdOpt = groupId + ) + case BlendGroupingMethodEnum.AuthorId => + GroupingKey( + sourceInfoOpt = None, + similarityEngineTypeOpt = None, + modelIdOpt = None, + authorIdOpt = Some(c.tweetInfo.authorId), + groupIdOpt = groupId + ) + case _ => + throw new UnsupportedOperationException( + s"Unsupported interleave feature: $interleaveFeature") + } + case _ => + GroupingKey( + sourceInfoOpt = None, + similarityEngineTypeOpt = None, + modelIdOpt = None, + authorIdOpt = None, + groupIdOpt = groupId + ) + } + grouping + } + + /** + * Rather than manually calculating and maintaining the weights to rank with, we instead + * calculate the weights on the fly, based upon the frequencies of the candidates within each + * group. To ensure that diversity of the feature is maintained, we additionally employ a + * 'shrinkage' parameter which enforces more diversity by moving the weights closer to uniformity. + * More details are available at go/weighted-interleave. + * + * @param candidateSeqKeyByFeature candidate to key. + * @param rankerWeightShrinkage value between [0, 1] with 1 being complete uniformity. + * @return Interleaving weights keyed by feature. + */ + private def calculateWeightsKeyByFeature[CandidateType <: Candidate]( + candidateSeqKeyByFeature: Map[GroupingKey, Seq[CandidateType]], + rankerWeightShrinkage: Double + ): Map[GroupingKey, Double] = { + val maxNumberCandidates: Double = candidateSeqKeyByFeature.values + .map { candidates => + candidates.size + }.max.toDouble + candidateSeqKeyByFeature.map { + case (featureKey: GroupingKey, candidateSeq: Seq[CandidateType]) => + val observedWeight: Double = candidateSeq.size.toDouble / maxNumberCandidates + // How much to shrink empirical estimates to 1 (Default is to make all weights 1). + val finalWeight = + (1.0 - rankerWeightShrinkage) * observedWeight + rankerWeightShrinkage * 1.0 + featureKey -> finalWeight + } + } + + /** + * Builds out the groups and weights for weighted interleaving of the candidates. + * More details are available at go/weighted-interleave. + * + * @param rankedCandidateSeq candidates to interleave. + * @param rankerWeightShrinkage value between [0, 1] with 1 being complete uniformity. + * @return Candidates grouped by feature key and with calculated interleaving weights. + */ + def buildRankedCandidatesWithWeightKeyByFeature( + rankedCandidateSeq: Seq[RankedCandidate], + rankerWeightShrinkage: Double, + interleaveFeature: BlendGroupingMethodEnum.Value + ): Seq[(Seq[RankedCandidate], Double)] = { + // To accommodate the re-grouping in InterleaveRanker + // In InterleaveBlender, we have already abandoned the grouping keys, and use Seq[Seq[]] to do interleave + // Since that we build the candidateSeq with groupingKey, we can guarantee there is no empty candidateSeq + val candidateSeqKeyByFeature: Map[GroupingKey, Seq[RankedCandidate]] = + rankedCandidateSeq.groupBy { candidate: RankedCandidate => + toGroupingKey(candidate, Some(interleaveFeature), None) + } + + // These weights [0, 1] are used to do weighted interleaving + // The default value of 1.0 ensures the group is always sampled. + val candidateWeightsKeyByFeature: Map[GroupingKey, Double] = + calculateWeightsKeyByFeature(candidateSeqKeyByFeature, rankerWeightShrinkage) + + candidateSeqKeyByFeature.map { + case (groupingKey: GroupingKey, candidateSeq: Seq[RankedCandidate]) => + Tuple2( + candidateSeq.sortBy(-_.predictionScore), + candidateWeightsKeyByFeature.getOrElse(groupingKey, 1.0)) + }.toSeq + } + + /** + * Takes current grouping (as implied by the outer Seq) and computes blending weights. + * + * @param initialCandidatesSeqSeq grouped candidates to interleave. + * @param rankerWeightShrinkage value between [0, 1] with 1 being complete uniformity. + * @return Grouped candidates with calculated interleaving weights. + */ + def buildInitialCandidatesWithWeightKeyByFeature( + initialCandidatesSeqSeq: Seq[Seq[InitialCandidate]], + rankerWeightShrinkage: Double, + ): Seq[(Seq[InitialCandidate], Double)] = { + val candidateSeqKeyByFeature: Map[GroupingKey, Seq[InitialCandidate]] = + initialCandidatesSeqSeq.zipWithIndex.map(_.swap).toMap.map { + case (groupId: Int, initialCandidatesSeq: Seq[InitialCandidate]) => + toGroupingKey(initialCandidatesSeq.head, None, Some(groupId)) -> initialCandidatesSeq + } + + // These weights [0, 1] are used to do weighted interleaving + // The default value of 1.0 ensures the group is always sampled. + val candidateWeightsKeyByFeature = + calculateWeightsKeyByFeature(candidateSeqKeyByFeature, rankerWeightShrinkage) + + candidateSeqKeyByFeature.map { + case (groupingKey: GroupingKey, candidateSeq: Seq[InitialCandidate]) => + Tuple2(candidateSeq, candidateWeightsKeyByFeature.getOrElse(groupingKey, 1.0)) + }.toSeq + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/util/EarlybirdSearchUtil.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/util/EarlybirdSearchUtil.scala new file mode 100644 index 0000000000..6ddd358dca --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/util/EarlybirdSearchUtil.scala @@ -0,0 +1,130 @@ +package com.twitter.cr_mixer.util + +import com.twitter.search.common.schema.earlybird.EarlybirdFieldConstants.EarlybirdFieldConstant +import com.twitter.search.queryparser.query.search.SearchOperator +import com.twitter.search.queryparser.query.search.SearchOperatorConstants +import com.twitter.search.queryparser.query.{Query => EbQuery} +import com.twitter.search.queryparser.query.Conjunction +import scala.collection.JavaConverters._ +import com.twitter.search.earlybird.thriftscala.ThriftSearchResultMetadataOptions +import com.twitter.simclusters_v2.common.TweetId +import com.twitter.search.queryparser.query.Query +import com.twitter.util.Duration +import com.twitter.search.common.query.thriftjava.thriftscala.CollectorTerminationParams + +object EarlybirdSearchUtil { + val EarlybirdClientId: String = "cr-mixer.prod" + + val Mentions: String = EarlybirdFieldConstant.MENTIONS_FACET + val Hashtags: String = EarlybirdFieldConstant.HASHTAGS_FACET + val FacetsToFetch: Seq[String] = Seq(Mentions, Hashtags) + + val MetadataOptions: ThriftSearchResultMetadataOptions = ThriftSearchResultMetadataOptions( + getTweetUrls = true, + getResultLocation = false, + getLuceneScore = false, + getInReplyToStatusId = true, + getReferencedTweetAuthorId = true, + getMediaBits = true, + getAllFeatures = true, + getFromUserId = true, + returnSearchResultFeatures = true, + // Set getExclusiveConversationAuthorId in order to retrieve Exclusive / SuperFollow tweets. + getExclusiveConversationAuthorId = true + ) + + // Filter out retweets and replies + val TweetTypesToExclude: Seq[String] = + Seq( + SearchOperatorConstants.NATIVE_RETWEETS, + SearchOperatorConstants.REPLIES) + + def GetCollectorTerminationParams( + maxNumHitsPerShard: Int, + processingTimeout: Duration + ): Option[CollectorTerminationParams] = { + Some( + CollectorTerminationParams( + // maxHitsToProcess is used for early termination on each EB shard + maxHitsToProcess = Some(maxNumHitsPerShard), + timeoutMs = processingTimeout.inMilliseconds.toInt + )) + } + + /** + * Get EarlybirdQuery + * This function creates a EBQuery based on the search input + */ + def GetEarlybirdQuery( + beforeTweetIdExclusive: Option[TweetId], + afterTweetIdExclusive: Option[TweetId], + excludedTweetIds: Set[TweetId], + filterOutRetweetsAndReplies: Boolean + ): Option[EbQuery] = + CreateConjunction( + Seq( + CreateRangeQuery(beforeTweetIdExclusive, afterTweetIdExclusive), + CreateExcludedTweetIdsQuery(excludedTweetIds), + CreateTweetTypesFilters(filterOutRetweetsAndReplies) + ).flatten) + + def CreateRangeQuery( + beforeTweetIdExclusive: Option[TweetId], + afterTweetIdExclusive: Option[TweetId] + ): Option[EbQuery] = { + val beforeIdClause = beforeTweetIdExclusive.map { beforeId => + // MAX_ID is an inclusive value therefore we subtract 1 from beforeId. + new SearchOperator(SearchOperator.Type.MAX_ID, (beforeId - 1).toString) + } + val afterIdClause = afterTweetIdExclusive.map { afterId => + new SearchOperator(SearchOperator.Type.SINCE_ID, afterId.toString) + } + CreateConjunction(Seq(beforeIdClause, afterIdClause).flatten) + } + + def CreateTweetTypesFilters(filterOutRetweetsAndReplies: Boolean): Option[EbQuery] = { + if (filterOutRetweetsAndReplies) { + val tweetTypeFilters = TweetTypesToExclude.map { searchOperator => + new SearchOperator(SearchOperator.Type.EXCLUDE, searchOperator) + } + CreateConjunction(tweetTypeFilters) + } else None + } + + def CreateConjunction(clauses: Seq[EbQuery]): Option[EbQuery] = { + clauses.size match { + case 0 => None + case 1 => Some(clauses.head) + case _ => Some(new Conjunction(clauses.asJava)) + } + } + + def CreateExcludedTweetIdsQuery(tweetIds: Set[TweetId]): Option[EbQuery] = { + if (tweetIds.nonEmpty) { + Some( + new SearchOperator.Builder() + .setType(SearchOperator.Type.NAMED_MULTI_TERM_DISJUNCTION) + .addOperand(EarlybirdFieldConstant.ID_FIELD.getFieldName) + .addOperand(EXCLUDE_TWEET_IDS) + .setOccur(Query.Occur.MUST_NOT) + .build()) + } else None + } + + /** + * Get NamedDisjunctions with excludedTweetIds + */ + def GetNamedDisjunctions(excludedTweetIds: Set[TweetId]): Option[Map[String, Seq[Long]]] = + if (excludedTweetIds.nonEmpty) + createNamedDisjunctionsExcludedTweetIds(excludedTweetIds) + else None + + val EXCLUDE_TWEET_IDS = "exclude_tweet_ids" + private def createNamedDisjunctionsExcludedTweetIds( + tweetIds: Set[TweetId] + ): Option[Map[String, Seq[Long]]] = { + if (tweetIds.nonEmpty) { + Some(Map(EXCLUDE_TWEET_IDS -> tweetIds.toSeq)) + } else None + } +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/util/InterleaveUtil.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/util/InterleaveUtil.scala new file mode 100644 index 0000000000..c75abde2ee --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/util/InterleaveUtil.scala @@ -0,0 +1,160 @@ +package com.twitter.cr_mixer.util + +import com.twitter.cr_mixer.model.Candidate +import com.twitter.cr_mixer.model.CandidateGenerationInfo +import com.twitter.cr_mixer.model.RankedCandidate +import com.twitter.cr_mixer.model.SourceInfo +import com.twitter.cr_mixer.thriftscala.SimilarityEngineType +import com.twitter.simclusters_v2.common.TweetId +import scala.collection.mutable +import scala.collection.mutable.ArrayBuffer + +object InterleaveUtil { + + /** + * Interleaves candidates by iteratively taking one candidate from the 1st Seq and adding it to the result. + * Once we take a candidate from a Seq, we move this Seq to the end of the queue to process, + * and remove the candidate from that Seq. + * + * We keep a mutable.Set[TweetId] buffer to ensure there are no duplicates. + * + * @param candidates candidates assumed to be sorted by eventTime (latest event comes first) + * @return interleaved candidates + */ + def interleave[CandidateType <: Candidate]( + candidates: Seq[Seq[CandidateType]] + ): Seq[CandidateType] = { + + // copy candidates into a mutable map so this method is thread-safe + val candidatesPerSequence = candidates.map { tweetCandidates => + mutable.Queue() ++= tweetCandidates + } + + val seen = mutable.Set[TweetId]() + + val candidateSeqQueue = mutable.Queue() ++= candidatesPerSequence + + val result = ArrayBuffer[CandidateType]() + + while (candidateSeqQueue.nonEmpty) { + val candidatesQueue = candidateSeqQueue.head + + if (candidatesQueue.nonEmpty) { + val candidate = candidatesQueue.dequeue() + val candidateTweetId = candidate.tweetId + val seenCandidate = seen.contains(candidateTweetId) + if (!seenCandidate) { + result += candidate + seen.add(candidate.tweetId) + candidateSeqQueue.enqueue( + candidateSeqQueue.dequeue() + ) // move this Seq to end + } + } else { + candidateSeqQueue.dequeue() //finished processing this Seq + } + } + //convert result to immutable seq + result.toList + } + + /** + * Interleaves candidates by iteratively + * 1. Checking weight to see if enough accumulation has occurred to sample from + * 2. If yes, taking one candidate from the the Seq and adding it to the result. + * 3. Move this Seq to the end of the queue to process (and remove the candidate from that Seq if + * we sampled it from step 2). + * + * We keep count of the iterations to prevent infinite loops. + * We keep a mutable.Set[TweetId] buffer to ensure there are no duplicates. + * + * @param candidatesAndWeight candidates assumed to be sorted by eventTime (latest event comes first), + * along with sampling weights to help prioritize important groups. + * @param maxWeightAdjustments Maximum number of iterations to account for weighting before + * defaulting to uniform interleaving. + * @return interleaved candidates + */ + def weightedInterleave[CandidateType <: Candidate]( + candidatesAndWeight: Seq[(Seq[CandidateType], Double)], + maxWeightAdjustments: Int = 0 + ): Seq[CandidateType] = { + + // Set to avoid numerical issues around 1.0 + val min_weight = 1 - 1e-30 + + // copy candidates into a mutable map so this method is thread-safe + // adds a counter to use towards sampling + val candidatesAndWeightsPerSequence: Seq[ + (mutable.Queue[CandidateType], InterleaveWeights) + ] = + candidatesAndWeight.map { candidatesAndWeight => + (mutable.Queue() ++= candidatesAndWeight._1, InterleaveWeights(candidatesAndWeight._2, 0.0)) + } + + val seen: mutable.Set[TweetId] = mutable.Set[TweetId]() + + val candidateSeqQueue: mutable.Queue[(mutable.Queue[CandidateType], InterleaveWeights)] = + mutable.Queue() ++= candidatesAndWeightsPerSequence + + val result: ArrayBuffer[CandidateType] = ArrayBuffer[CandidateType]() + var number_iterations: Int = 0 + + while (candidateSeqQueue.nonEmpty) { + val (candidatesQueue, currentWeights) = candidateSeqQueue.head + if (candidatesQueue.nonEmpty) { + // Confirm weighting scheme + currentWeights.summed_weight += currentWeights.weight + number_iterations += 1 + if (currentWeights.summed_weight >= min_weight || number_iterations >= maxWeightAdjustments) { + // If we sample, then adjust the counter + currentWeights.summed_weight -= 1.0 + val candidate = candidatesQueue.dequeue() + val candidateTweetId = candidate.tweetId + val seenCandidate = seen.contains(candidateTweetId) + if (!seenCandidate) { + result += candidate + seen.add(candidate.tweetId) + candidateSeqQueue.enqueue(candidateSeqQueue.dequeue()) // move this Seq to end + } + } else { + candidateSeqQueue.enqueue(candidateSeqQueue.dequeue()) // move this Seq to end + } + } else { + candidateSeqQueue.dequeue() //finished processing this Seq + } + } + //convert result to immutable seq + result.toList + } + + def buildCandidatesKeyByCGInfo( + candidates: Seq[RankedCandidate], + ): Seq[Seq[RankedCandidate]] = { + // To accommodate the re-grouping in InterleaveRanker + // In InterleaveBlender, we have already abandoned the grouping keys, and use Seq[Seq[]] to do interleave + // Since that we build the candidateSeq with groupingKey, we can guarantee there is no empty candidateSeq + val candidateSeqKeyByCG = + candidates.groupBy(candidate => GroupingKey.toGroupingKey(candidate.reasonChosen)) + candidateSeqKeyByCG.map { + case (groupingKey, candidateSeq) => + candidateSeq.sortBy(-_.predictionScore) + }.toSeq + } +} + +case class GroupingKey( + sourceInfoOpt: Option[SourceInfo], + similarityEngineType: SimilarityEngineType, + modelId: Option[String]) {} + +object GroupingKey { + def toGroupingKey(candidateGenerationInfo: CandidateGenerationInfo): GroupingKey = { + GroupingKey( + candidateGenerationInfo.sourceInfoOpt, + candidateGenerationInfo.similarityEngineInfo.similarityEngineType, + candidateGenerationInfo.similarityEngineInfo.modelId + ) + } +} + +case class InterleaveWeights(weight: Double, var summed_weight: Double) diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/util/MetricTagUtil.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/util/MetricTagUtil.scala new file mode 100644 index 0000000000..caa6d9f074 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/util/MetricTagUtil.scala @@ -0,0 +1,135 @@ +package com.twitter.cr_mixer.util + +import com.twitter.cr_mixer.model.RankedCandidate +import com.twitter.cr_mixer.model.SimilarityEngineInfo +import com.twitter.cr_mixer.model.SourceInfo +import com.twitter.cr_mixer.thriftscala.MetricTag +import com.twitter.cr_mixer.thriftscala.SimilarityEngineType +import com.twitter.cr_mixer.thriftscala.SourceType + +object MetricTagUtil { + + def buildMetricTags(candidate: RankedCandidate): Seq[MetricTag] = { + val interestedInMetricTag = isFromInterestedIn(candidate) + + val cgInfoMetricTags = candidate.potentialReasons + .flatMap { cgInfo => + val sourceMetricTag = cgInfo.sourceInfoOpt.flatMap { sourceInfo => + toMetricTagFromSource(sourceInfo.sourceType) + } + val similarityEngineTags = toMetricTagFromSimilarityEngine( + cgInfo.similarityEngineInfo, + cgInfo.contributingSimilarityEngines) + + val combinedMetricTag = cgInfo.sourceInfoOpt.flatMap { sourceInfo => + toMetricTagFromSourceAndSimilarityEngine(sourceInfo, cgInfo.similarityEngineInfo) + } + + Seq(sourceMetricTag) ++ similarityEngineTags ++ Seq(combinedMetricTag) + }.flatten.toSet + (interestedInMetricTag ++ cgInfoMetricTags).toSeq + } + + /*** + * match a sourceType to a metricTag + */ + private def toMetricTagFromSource(sourceType: SourceType): Option[MetricTag] = { + sourceType match { + case SourceType.TweetFavorite => Some(MetricTag.TweetFavorite) // Personalized Topics in Home + case SourceType.Retweet => Some(MetricTag.Retweet) // Personalized Topics in Home + case SourceType.NotificationClick => + Some(MetricTag.PushOpenOrNtabClick) // Health Filter in MR + case SourceType.OriginalTweet => + Some(MetricTag.OriginalTweet) + case SourceType.Reply => + Some(MetricTag.Reply) + case SourceType.TweetShare => + Some(MetricTag.TweetShare) + case SourceType.UserFollow => + Some(MetricTag.UserFollow) + case SourceType.UserRepeatedProfileVisit => + Some(MetricTag.UserRepeatedProfileVisit) + case SourceType.TwiceUserId => + Some(MetricTag.TwiceUserId) + case _ => None + } + } + + /*** + * If the SEInfo is built by a unified sim engine, we un-wrap the contributing sim engines. + * If not, we log the sim engine as usual. + * @param seInfo (CandidateGenerationInfo.similarityEngineInfo): SimilarityEngineInfo, + * @param cseInfo (CandidateGenerationInfo.contributingSimilarityEngines): Seq[SimilarityEngineInfo] + */ + private def toMetricTagFromSimilarityEngine( + seInfo: SimilarityEngineInfo, + cseInfo: Seq[SimilarityEngineInfo] + ): Seq[Option[MetricTag]] = { + seInfo.similarityEngineType match { + case SimilarityEngineType.TweetBasedUnifiedSimilarityEngine => // un-wrap the unified sim engine + cseInfo.map { contributingSimEngine => + toMetricTagFromSimilarityEngine(contributingSimEngine, Seq.empty) + }.flatten + case SimilarityEngineType.ProducerBasedUnifiedSimilarityEngine => // un-wrap the unified sim engine + cseInfo.map { contributingSimEngine => + toMetricTagFromSimilarityEngine(contributingSimEngine, Seq.empty) + }.flatten + // SimClustersANN can either be called on its own, or be called under unified sim engine + case SimilarityEngineType.SimClustersANN => // the old "UserInterestedIn" will be replaced by this. Also, OfflineTwice + Seq(Some(MetricTag.SimClustersANN), seInfo.modelId.flatMap(toMetricTagFromModelId(_))) + case SimilarityEngineType.ConsumerEmbeddingBasedTwHINANN => + Seq(Some(MetricTag.ConsumerEmbeddingBasedTwHINANN)) + case SimilarityEngineType.TwhinCollabFilter => Seq(Some(MetricTag.TwhinCollabFilter)) + // In the current implementation, TweetBasedUserTweetGraph/TweetBasedTwHINANN has a tag when + // it's either a base SE or a contributing SE. But for now they only show up in contributing SE. + case SimilarityEngineType.TweetBasedUserTweetGraph => + Seq(Some(MetricTag.TweetBasedUserTweetGraph)) + case SimilarityEngineType.TweetBasedTwHINANN => + Seq(Some(MetricTag.TweetBasedTwHINANN)) + case _ => Seq.empty + } + } + + /*** + * pass in a model id, and match it with the metric tag type. + */ + private def toMetricTagFromModelId( + modelId: String + ): Option[MetricTag] = { + + val pushOpenBasedModelRegex = "(.*_Model20m145k2020_20220819)".r + + modelId match { + case pushOpenBasedModelRegex(_*) => + Some(MetricTag.RequestHealthFilterPushOpenBasedTweetEmbedding) + case _ => None + } + } + + private def toMetricTagFromSourceAndSimilarityEngine( + sourceInfo: SourceInfo, + seInfo: SimilarityEngineInfo + ): Option[MetricTag] = { + sourceInfo.sourceType match { + case SourceType.Lookalike + if seInfo.similarityEngineType == SimilarityEngineType.ConsumersBasedUserTweetGraph => + Some(MetricTag.LookalikeUTG) + case _ => None + } + } + + /** + * Special use case: used by Notifications team to generate the UserInterestedIn CRT push copy. + * + * if we have different types of InterestedIn (eg. UserInterestedIn, NextInterestedIn), + * this if statement will have to be refactored to contain the real UserInterestedIn. + * @return + */ + private def isFromInterestedIn(candidate: RankedCandidate): Set[MetricTag] = { + if (candidate.reasonChosen.sourceInfoOpt.isEmpty + && candidate.reasonChosen.similarityEngineInfo.similarityEngineType == SimilarityEngineType.SimClustersANN) { + Set(MetricTag.UserInterestedIn) + } else Set.empty + } + +} diff --git a/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/util/SignalTimestampStatsUtil.scala b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/util/SignalTimestampStatsUtil.scala new file mode 100644 index 0000000000..ae27894321 --- /dev/null +++ b/cr-mixer/server/src/main/scala/com/twitter/cr_mixer/util/SignalTimestampStatsUtil.scala @@ -0,0 +1,66 @@ +package com.twitter.cr_mixer.util + +import com.twitter.cr_mixer.model.CandidateGenerationInfo +import com.twitter.cr_mixer.model.RankedCandidate +import com.twitter.cr_mixer.model.SourceInfo +import com.twitter.cr_mixer.thriftscala.SourceType +import com.twitter.cr_mixer.thriftscala.TweetRecommendation +import javax.inject.Inject +import com.twitter.finagle.stats.StatsReceiver +import javax.inject.Singleton +import com.twitter.relevance_platform.common.stats.BucketTimestampStats + +@Singleton +class SignalTimestampStatsUtil @Inject() (statsReceiver: StatsReceiver) { + import SignalTimestampStatsUtil._ + + private val signalDelayAgePerDayStats = + new BucketTimestampStats[TweetRecommendation]( + BucketTimestampStats.MillisecondsPerDay, + _.latestSourceSignalTimestampInMillis.getOrElse(0), + Some(SignalTimestampMaxDays))( + statsReceiver.scope("signal_timestamp_per_day") + ) // only stats past 90 days + private val signalDelayAgePerHourStats = + new BucketTimestampStats[TweetRecommendation]( + BucketTimestampStats.MillisecondsPerHour, + _.latestSourceSignalTimestampInMillis.getOrElse(0), + Some(SignalTimestampMaxHours))( + statsReceiver.scope("signal_timestamp_per_hour") + ) // only stats past 24 hours + private val signalDelayAgePerMinStats = + new BucketTimestampStats[TweetRecommendation]( + BucketTimestampStats.MillisecondsPerMinute, + _.latestSourceSignalTimestampInMillis.getOrElse(0), + Some(SignalTimestampMaxMins))( + statsReceiver.scope("signal_timestamp_per_min") + ) // only stats past 60 minutes + + def statsSignalTimestamp( + tweets: Seq[TweetRecommendation], + ): Seq[TweetRecommendation] = { + signalDelayAgePerMinStats.count(tweets) + signalDelayAgePerHourStats.count(tweets) + signalDelayAgePerDayStats.count(tweets) + } +} + +object SignalTimestampStatsUtil { + val SignalTimestampMaxMins = 60 // stats at most 60 mins + val SignalTimestampMaxHours = 24 // stats at most 24 hours + val SignalTimestampMaxDays = 90 // stats at most 90 days + + def buildLatestSourceSignalTimestamp(candidate: RankedCandidate): Option[Long] = { + val timestampSeq = candidate.potentialReasons + .collect { + case CandidateGenerationInfo(Some(SourceInfo(sourceType, _, Some(sourceEventTime))), _, _) + if sourceType == SourceType.TweetFavorite => + sourceEventTime.inMilliseconds + } + if (timestampSeq.nonEmpty) { + Some(timestampSeq.max(Ordering.Long)) + } else { + None + } + } +} diff --git a/cr-mixer/thrift/src/main/thrift/BUILD b/cr-mixer/thrift/src/main/thrift/BUILD new file mode 100644 index 0000000000..3ccb856816 --- /dev/null +++ b/cr-mixer/thrift/src/main/thrift/BUILD @@ -0,0 +1,48 @@ +create_thrift_libraries( + base_name = "thrift", + sources = ["*.thrift"], + platform = "java8", + tags = ["bazel-compatible"], + dependency_roots = [ + "finatra-internal/thrift/src/main/thrift", + "product-mixer/core/src/main/thrift/com/twitter/product_mixer/core:thrift", + "src/thrift/com/twitter/ads/schema:common", + "src/thrift/com/twitter/ml/api:data", + "src/thrift/com/twitter/recos:recos-common", + "src/thrift/com/twitter/simclusters_v2:simclusters_v2-thrift", + "src/thrift/com/twitter/timelines/render:thrift", + "strato/config/src/thrift/com/twitter/strato/graphql", + "strato/config/src/thrift/com/twitter/strato/graphql:api-media-graphql", + "strato/config/src/thrift/com/twitter/strato/graphql:topics-graphql", + ], + generate_languages = [ + "java", + "scala", + "strato", + ], + provides_java_name = "cr-mixer-thrift-java", + provides_scala_name = "cr-mixer-thrift-scala", +) + +create_thrift_libraries( + base_name = "cr-mixer-scribe", + sources = ["*.thrift"], + tags = ["bazel-compatible"], + dependency_roots = [ + "finatra-internal/thrift/src/main/thrift", + "product-mixer/core/src/main/thrift/com/twitter/product_mixer/core:thrift", + "src/thrift/com/twitter/ads/schema:common", + "src/thrift/com/twitter/ml/api:data", + "src/thrift/com/twitter/recos:recos-common", + "src/thrift/com/twitter/simclusters_v2:simclusters_v2-thrift", + "src/thrift/com/twitter/timelines/render:thrift", + "strato/config/src/thrift/com/twitter/strato/graphql", + ], + generate_languages = [ + "java", + "scala", + "strato", + ], + provides_java_name = "cr-mixer-scribe-java", + provides_scala_name = "cr-mixer-scribe-scala", +) diff --git a/cr-mixer/thrift/src/main/thrift/ads.thrift b/cr-mixer/thrift/src/main/thrift/ads.thrift new file mode 100644 index 0000000000..70d4ad562d --- /dev/null +++ b/cr-mixer/thrift/src/main/thrift/ads.thrift @@ -0,0 +1,33 @@ +namespace java com.twitter.cr_mixer.thriftjava +#@namespace scala com.twitter.cr_mixer.thriftscala +#@namespace strato com.twitter.cr_mixer + +include "product.thrift" +include "product_context.thrift" + +include "com/twitter/product_mixer/core/client_context.thrift" +include "com/twitter/ads/schema/shared.thrift" + +struct AdsRequest { + 1: required client_context.ClientContext clientContext + 2: required product.Product product + # Product-specific parameters should be placed in the Product Context + 3: optional product_context.ProductContext productContext + 4: optional list excludedTweetIds (personalDataType = 'TweetId') +} (persisted='true', hasPersonalData='true') + +struct AdsResponse { + 1: required list ads +} (persisted='true') + +struct AdTweetRecommendation { + 1: required i64 tweetId (personalDataType = 'TweetId') + 2: required double score + 3: optional list lineItems + +} (persisted='true') + +struct LineItemInfo { + 1: required i64 lineItemId (personalDataType = 'LineItemId') + 2: required shared.LineItemObjective lineItemObjective +} (persisted='true') diff --git a/cr-mixer/thrift/src/main/thrift/candidate_generation_key.thrift b/cr-mixer/thrift/src/main/thrift/candidate_generation_key.thrift new file mode 100644 index 0000000000..4f2a4a9eeb --- /dev/null +++ b/cr-mixer/thrift/src/main/thrift/candidate_generation_key.thrift @@ -0,0 +1,21 @@ +namespace java com.twitter.cr_mixer.thriftjava +#@namespace scala com.twitter.cr_mixer.thriftscala +#@namespace strato com.twitter.cr_mixer + +include "source_type.thrift" +include "com/twitter/simclusters_v2/identifier.thrift" + +struct SimilarityEngine { + 1: required source_type.SimilarityEngineType similarityEngineType + 2: optional string modelId + 3: optional double score +} (persisted='true') + +struct CandidateGenerationKey { + 1: required source_type.SourceType sourceType + 2: required i64 sourceEventTime (personalDataType = 'PrivateTimestamp') + 3: required identifier.InternalId id + 4: required string modelId + 5: optional source_type.SimilarityEngineType similarityEngineType + 6: optional list contributingSimilarityEngine +} (persisted='true') diff --git a/cr-mixer/thrift/src/main/thrift/cr_mixer.thrift b/cr-mixer/thrift/src/main/thrift/cr_mixer.thrift new file mode 100644 index 0000000000..2fddf1cf82 --- /dev/null +++ b/cr-mixer/thrift/src/main/thrift/cr_mixer.thrift @@ -0,0 +1,104 @@ +namespace java com.twitter.cr_mixer.thriftjava +#@namespace scala com.twitter.cr_mixer.thriftscala +#@namespace strato com.twitter.cr_mixer + +include "ads.thrift" +include "candidate_generation_key.thrift" +include "product.thrift" +include "product_context.thrift" +include "validation.thrift" +include "metric_tags.thrift" +include "related_tweet.thrift" +include "uteg.thrift" +include "frs_based_tweet.thrift" +include "related_video_tweet.thrift" +include "topic_tweet.thrift" + +include "com/twitter/product_mixer/core/client_context.thrift" +include "com/twitter/timelines/render/response.thrift" +include "finatra-thrift/finatra_thrift_exceptions.thrift" +include "com/twitter/strato/graphql/slice.thrift" + +struct CrMixerTweetRequest { + 1: required client_context.ClientContext clientContext + 2: required product.Product product + # Product-specific parameters should be placed in the Product Context + 3: optional product_context.ProductContext productContext + 4: optional list excludedTweetIds (personalDataType = 'TweetId') +} (persisted='true', hasPersonalData='true') + +struct TweetRecommendation { + 1: required i64 tweetId (personalDataType = 'TweetId') + 2: required double score + 3: optional list metricTags + # 4: the author of the tweet candidate. To be used by Content-Mixer to unblock the Hydra experiment. + 4: optional i64 authorId (personalDataType = 'UserId') + # 5: extra info about candidate generation. To be used by Content-Mixer to unblock the Hydra experiment. + 5: optional candidate_generation_key.CandidateGenerationKey candidateGenerationKey + # 1001: the latest timestamp of fav signals. If null, the candidate is not generated from fav signals + 1001: optional i64 latestSourceSignalTimestampInMillis(personalDataType = 'PublicTimestamp') +} (persisted='true', hasPersonalData = 'true') + +struct CrMixerTweetResponse { + 1: required list tweets +} (persisted='true') + +service CrMixer { + CrMixerTweetResponse getTweetRecommendations(1: CrMixerTweetRequest request) throws ( + # Validation errors - the details of which will be reported to clients on failure + 1: validation.ValidationExceptionList validationErrors; + # Server errors - the details of which will not be reported to clients + 2: finatra_thrift_exceptions.ServerError serverError + ) + + # getRelatedTweetsForQueryTweet and getRelatedTweetsForQueryAuthor do very similar things + # We can merge these two endpoints into one unified endpoint + related_tweet.RelatedTweetResponse getRelatedTweetsForQueryTweet(1: related_tweet.RelatedTweetRequest request) throws ( + # Validation errors - the details of which will be reported to clients on failure + 1: validation.ValidationExceptionList validationErrors; + # Server errors - the details of which will not be reported to clients + 2: finatra_thrift_exceptions.ServerError serverError + ) + + related_tweet.RelatedTweetResponse getRelatedTweetsForQueryAuthor(1: related_tweet.RelatedTweetRequest request) throws ( + # Validation errors - the details of which will be reported to clients on failure + 1: validation.ValidationExceptionList validationErrors; + # Server errors - the details of which will not be reported to clients + 2: finatra_thrift_exceptions.ServerError serverError + ) + + uteg.UtegTweetResponse getUtegTweetRecommendations(1: uteg.UtegTweetRequest request) throws ( + # Validation errors - the details of which will be reported to clients on failure + 1: validation.ValidationExceptionList validationErrors; + # Server errors - the details of which will not be reported to clients + 2: finatra_thrift_exceptions.ServerError serverError + ) + + frs_based_tweet.FrsTweetResponse getFrsBasedTweetRecommendations(1: frs_based_tweet.FrsTweetRequest request) throws ( + # Validation errors - the details of which will be reported to clients on failure + 1: validation.ValidationExceptionList validationErrors; + # Server errors - the details of which will not be reported to clients + 2: finatra_thrift_exceptions.ServerError serverError + ) + + related_video_tweet.RelatedVideoTweetResponse getRelatedVideoTweetsForQueryTweet(1: related_video_tweet.RelatedVideoTweetRequest request) throws ( + # Validation errors - the details of which will be reported to clients on failure + 1: validation.ValidationExceptionList validationErrors; + # Server errors - the details of which will not be reported to clients + 2: finatra_thrift_exceptions.ServerError serverError + ) + + ads.AdsResponse getAdsRecommendations(1: ads.AdsRequest request) throws ( + # Validation errors - the details of which will be reported to clients on failure + 1: validation.ValidationExceptionList validationErrors; + # Server errors - the details of which will not be reported to clients + 2: finatra_thrift_exceptions.ServerError serverError + ) + + topic_tweet.TopicTweetResponse getTopicTweetRecommendations(1: topic_tweet.TopicTweetRequest request) throws ( + # Validation errors - the details of which will be reported to clients on failure + 1: validation.ValidationExceptionList validationErrors; + # Server errors - the details of which will not be reported to clients + 2: finatra_thrift_exceptions.ServerError serverError + ) +} diff --git a/cr-mixer/thrift/src/main/thrift/frs_based_tweet.thrift b/cr-mixer/thrift/src/main/thrift/frs_based_tweet.thrift new file mode 100644 index 0000000000..bb83397b6e --- /dev/null +++ b/cr-mixer/thrift/src/main/thrift/frs_based_tweet.thrift @@ -0,0 +1,35 @@ +namespace java com.twitter.cr_mixer.thriftjava +#@namespace scala com.twitter.cr_mixer.thriftscala +#@namespace strato com.twitter.cr_mixer + +include "product.thrift" +include "product_context.thrift" +include "com/twitter/product_mixer/core/client_context.thrift" + +struct FrsTweetRequest { +1: required client_context.ClientContext clientContext +2: required product.Product product +3: optional product_context.ProductContext productContext +# excludedUserIds - user ids to be excluded from FRS candidate generation +4: optional list excludedUserIds (personalDataType = 'UserId') +# excludedTweetIds - tweet ids to be excluded from Earlybird candidate generation +5: optional list excludedTweetIds (personalDataType = 'TweetId') +} (persisted='true', hasPersonalData='true') + +struct FrsTweet { +1: required i64 tweetId (personalDataType = 'TweetId') +2: required i64 authorId (personalDataType = 'UserId') +# skip 3 in case we need tweet score in the future +# frsPrimarySource - which FRS candidate source is the primary one to generate this author +4: optional i32 frsPrimarySource +# frsCandidateSourceScores - FRS candidate sources and the scores for this author +# for i32 to algorithm mapping, see https://sourcegraph.twitter.biz/git.twitter.biz/source/-/blob/hermit/hermit-core/src/main/scala/com/twitter/hermit/constants/AlgorithmFeedbackTokens.scala?L12 +5: optional map frsCandidateSourceScores +# frsPrimaryScore - the score of the FRS primary candidate source +6: optional double frsAuthorScore +} (persisted='true', hasPersonalData = 'true') + +struct FrsTweetResponse { + 1: required list tweets +} (persisted='true') + diff --git a/cr-mixer/thrift/src/main/thrift/metric_tags.thrift b/cr-mixer/thrift/src/main/thrift/metric_tags.thrift new file mode 100644 index 0000000000..dd4fb50125 --- /dev/null +++ b/cr-mixer/thrift/src/main/thrift/metric_tags.thrift @@ -0,0 +1,44 @@ +namespace java com.twitter.cr_mixer.thriftjava +#@namespace scala com.twitter.cr_mixer.thriftscala +#@namespace strato com.twitter.cr_mixer + + +// NOTE: DO NOT depend on MetricTags for important ML Features or business logic. +// MetricTags are meant for stats tracking & debugging purposes ONLY. +// cr-mixer may change its definitions & how each candidate is tagged without public notice. +// NOTE: TSPS needs the caller (Home) to specify which signal it uses to make Personalized Topics +enum MetricTag { + // Source Signal Tags + TweetFavorite = 0 + Retweet = 1 + TrafficAttribution = 2 + OriginalTweet = 3 + Reply = 4 + TweetShare = 5 + + UserFollow = 101 + UserRepeatedProfileVisit = 102 + + PushOpenOrNtabClick = 201 + + HomeTweetClick = 301 + HomeVideoView = 302 + + // sim engine types + SimClustersANN = 401 + TweetBasedUserTweetGraph = 402 + TweetBasedTwHINANN = 403 + ConsumerEmbeddingBasedTwHINANN = 404 + + + // combined engine types + UserInterestedIn = 501 // Will deprecate soon + LookalikeUTG = 502 + TwhinCollabFilter = 503 + + // Offline Twice + TwiceUserId = 601 + + // Other Metric Tags + RequestHealthFilterPushOpenBasedTweetEmbedding = 701 +} (persisted='true', hasPersonalData='true') diff --git a/cr-mixer/thrift/src/main/thrift/product.thrift b/cr-mixer/thrift/src/main/thrift/product.thrift new file mode 100644 index 0000000000..6e23a10924 --- /dev/null +++ b/cr-mixer/thrift/src/main/thrift/product.thrift @@ -0,0 +1,19 @@ +namespace java com.twitter.cr_mixer.thriftjava +#@namespace scala com.twitter.cr_mixer.thriftscala +#@namespace strato com.twitter.cr_mixer + +# In CrMixer, one org should only have one Product +enum Product { + Home = 1 + Notifications = 2 + Email = 3 + MoreTweetsModule = 4 # aka RUX + ImmersiveMediaViewer = 5 + VideoCarousel = 6 + ExploreTopics = 7 + Ads = 8 + HomeRealTime = 9 // Home Real-Time Tab is considered as a different Product surface to Home Tab. It's in early experiment phase. + TopicLandingPage = 10 + HomeTopicsBackfill = 11 + TopicTweetsStrato = 12 +} diff --git a/cr-mixer/thrift/src/main/thrift/product_context.thrift b/cr-mixer/thrift/src/main/thrift/product_context.thrift new file mode 100644 index 0000000000..29e2d96873 --- /dev/null +++ b/cr-mixer/thrift/src/main/thrift/product_context.thrift @@ -0,0 +1,21 @@ +namespace java com.twitter.cr_mixer.thriftjava +#@namespace scala com.twitter.cr_mixer.thriftscala +#@namespace strato com.twitter.cr_mixer + +struct HomeContext { + 2: optional i32 maxResults // enabled for QuaityFactor related DDGs only +} (persisted='true', hasPersonalData='false') + +struct NotificationsContext { + 1: optional i32 devNull // not being used. it's a placeholder +} (persisted='true', hasPersonalData='false') + +struct ExploreContext { + 1: required bool isVideoOnly +} (persisted='true', hasPersonalData='false') + +union ProductContext { + 1: HomeContext homeContext + 2: NotificationsContext notificationsContext + 3: ExploreContext exploreContext +} (persisted='true', hasPersonalData='false') diff --git a/cr-mixer/thrift/src/main/thrift/related_tweet.thrift b/cr-mixer/thrift/src/main/thrift/related_tweet.thrift new file mode 100644 index 0000000000..04e797b1bb --- /dev/null +++ b/cr-mixer/thrift/src/main/thrift/related_tweet.thrift @@ -0,0 +1,24 @@ +namespace java com.twitter.cr_mixer.thriftjava +#@namespace scala com.twitter.cr_mixer.thriftscala +#@namespace strato com.twitter.cr_mixer + +include "product.thrift" +include "com/twitter/product_mixer/core/client_context.thrift" +include "com/twitter/simclusters_v2/identifier.thrift" + +struct RelatedTweetRequest { + 1: required identifier.InternalId internalId + 2: required product.Product product + 3: required client_context.ClientContext clientContext # RUX LogOut will have clientContext.userId = None + 4: optional list excludedTweetIds (personalDataType = 'TweetId') +} (persisted='true', hasPersonalData='true') + +struct RelatedTweet { + 1: required i64 tweetId (personalDataType = 'TweetId') + 2: optional double score + 3: optional i64 authorId (personalDataType = 'UserId') +} (persisted='true', hasPersonalData='true') + +struct RelatedTweetResponse { + 1: required list tweets +} (persisted='true') diff --git a/cr-mixer/thrift/src/main/thrift/related_video_tweet.thrift b/cr-mixer/thrift/src/main/thrift/related_video_tweet.thrift new file mode 100644 index 0000000000..18a987e7d9 --- /dev/null +++ b/cr-mixer/thrift/src/main/thrift/related_video_tweet.thrift @@ -0,0 +1,23 @@ +namespace java com.twitter.cr_mixer.thriftjava +#@namespace scala com.twitter.cr_mixer.thriftscala +#@namespace strato com.twitter.cr_mixer + +include "product.thrift" +include "com/twitter/product_mixer/core/client_context.thrift" +include "com/twitter/simclusters_v2/identifier.thrift" + +struct RelatedVideoTweetRequest { + 1: required identifier.InternalId internalId + 2: required product.Product product + 3: required client_context.ClientContext clientContext # RUX LogOut will have clientContext.userId = None + 4: optional list excludedTweetIds (personalDataType = 'TweetId') +} (persisted='true', hasPersonalData='true') + +struct RelatedVideoTweet { + 1: required i64 tweetId (personalDataType = 'TweetId') + 2: optional double score +} (persisted='true', hasPersonalData='true') + +struct RelatedVideoTweetResponse { + 1: required list tweets +} (persisted='true') diff --git a/cr-mixer/thrift/src/main/thrift/scribe.thrift b/cr-mixer/thrift/src/main/thrift/scribe.thrift new file mode 100644 index 0000000000..61fdb5eb99 --- /dev/null +++ b/cr-mixer/thrift/src/main/thrift/scribe.thrift @@ -0,0 +1,168 @@ +namespace java com.twitter.cr_mixer.thriftjava +#@namespace scala com.twitter.cr_mixer.thriftscala +#@namespace strato com.twitter.cr_mixer + +include "ads.thrift" +include "candidate_generation_key.thrift" +include "cr_mixer.thrift" +include "metric_tags.thrift" +include "product.thrift" +include "related_tweet.thrift" +include "source_type.thrift" +include "uteg.thrift" +include "com/twitter/ml/api/data.thrift" +include "com/twitter/simclusters_v2/identifier.thrift" + +struct VITTweetCandidatesScribe { + 1: required i64 uuid (personalDataType = 'UniversallyUniqueIdentifierUuid') # RequestUUID - unique scribe id for every request that comes in. Same request but different stages of scribe log (FetchCandidate, Filter, etc) share the same uuid + 2: required i64 userId (personalDataType = 'UserId') + 3: required list candidates + 7: required product.Product product + 8: required list impressedBuckets +} (persisted='true', hasPersonalData = 'true') + +struct VITTweetCandidateScribe { + 1: required i64 tweetId (personalDataType = 'TweetId') + 2: required i64 authorId (personalDataType = 'UserId') + 3: required double score + 4: required list metricTags +} (persisted='true', hasPersonalData = 'true') + +struct GetTweetsRecommendationsScribe { + 1: required i64 uuid (personalDataType = 'UniversallyUniqueIdentifierUuid') # RequestUUID - unique scribe id for every request that comes in. Same request but different stages of scribe log (FetchCandidate, Filter, etc) share the same uuid + 2: required i64 userId (personalDataType = 'UserId') + 3: required Result result + 4: optional i64 traceId + 5: optional PerformanceMetrics performanceMetrics + 6: optional list impressedBuckets +} (persisted='true', hasPersonalData = 'true') + +struct SourceSignal { + # optional, since that the next step covers all info here + 1: optional identifier.InternalId id +} (persisted='true') + +struct PerformanceMetrics { + 1: optional i64 latencyMs +} (persisted='true') + +struct TweetCandidateWithMetadata { + 1: required i64 tweetId (personalDataType = 'TweetId') + 2: optional candidate_generation_key.CandidateGenerationKey candidateGenerationKey + 3: optional i64 authorId (personalDataType = 'UserId') # only for InterleaveResult for hydrating training data + 4: optional double score # score with respect to candidateGenerationKey + 5: optional data.DataRecord dataRecord # attach any features to this candidate + 6: optional i32 numCandidateGenerationKeys # num CandidateGenerationKeys generating this tweetId +} (persisted='true') + +struct FetchSignalSourcesResult { + 1: optional set signals +} (persisted='true') + +struct FetchCandidatesResult { + 1: optional list tweets +} (persisted='true') + +struct PreRankFilterResult { + 1: optional list tweets +} (persisted='true') + +struct InterleaveResult { + 1: optional list tweets +} (persisted='true') + +struct RankResult { + 1: optional list tweets +} (persisted='true') + +struct TopLevelApiResult { + 1: required i64 timestamp (personalDataType = 'PrivateTimestamp') + 2: required cr_mixer.CrMixerTweetRequest request + 3: required cr_mixer.CrMixerTweetResponse response +} (persisted='true') + +union Result { + 1: FetchSignalSourcesResult fetchSignalSourcesResult + 2: FetchCandidatesResult fetchCandidatesResult + 3: PreRankFilterResult preRankFilterResult + 4: InterleaveResult interleaveResult + 5: RankResult rankResult + 6: TopLevelApiResult topLevelApiResult +} (persisted='true', hasPersonalData = 'true') + +struct ImpressesedBucketInfo { + 1: required i64 experimentId (personalDataType = 'ExperimentId') + 2: required string bucketName + 3: required i32 version +} (persisted='true') + +############# RelatedTweets Scribe ############# + +struct GetRelatedTweetsScribe { + 1: required i64 uuid (personalDataType = 'UniversallyUniqueIdentifierUuid') # RequestUUID - unique scribe id for every request that comes in. Same request but different stages of scribe log (FetchCandidate, Filter, etc) share the same uuid + 2: required identifier.InternalId internalId + 3: required RelatedTweetResult relatedTweetResult + 4: optional i64 requesterId (personalDataType = 'UserId') + 5: optional i64 guestId (personalDataType = 'GuestId') + 6: optional i64 traceId + 7: optional PerformanceMetrics performanceMetrics + 8: optional list impressedBuckets +} (persisted='true', hasPersonalData = 'true') + +struct RelatedTweetTopLevelApiResult { + 1: required i64 timestamp (personalDataType = 'PrivateTimestamp') + 2: required related_tweet.RelatedTweetRequest request + 3: required related_tweet.RelatedTweetResponse response +} (persisted='true') + +union RelatedTweetResult { + 1: RelatedTweetTopLevelApiResult relatedTweetTopLevelApiResult + 2: FetchCandidatesResult fetchCandidatesResult + 3: PreRankFilterResult preRankFilterResult # results after seqential filters + # if later we need rankResult, we can add it here +} (persisted='true', hasPersonalData = 'true') + +############# UtegTweets Scribe ############# + +struct GetUtegTweetsScribe { + 1: required i64 uuid (personalDataType = 'UniversallyUniqueIdentifierUuid') # RequestUUID - unique scribe id for every request that comes in. Same request but different stages of scribe log (FetchCandidate, Filter, etc) share the same uuid + 2: required i64 userId (personalDataType = 'UserId') + 3: required UtegTweetResult utegTweetResult + 4: optional i64 traceId + 5: optional PerformanceMetrics performanceMetrics + 6: optional list impressedBuckets +} (persisted='true', hasPersonalData = 'true') + +struct UtegTweetTopLevelApiResult { + 1: required i64 timestamp (personalDataType = 'PrivateTimestamp') + 2: required uteg.UtegTweetRequest request + 3: required uteg.UtegTweetResponse response +} (persisted='true') + +union UtegTweetResult { + 1: UtegTweetTopLevelApiResult utegTweetTopLevelApiResult + 2: FetchCandidatesResult fetchCandidatesResult + # if later we need rankResult, we can add it here +} (persisted='true', hasPersonalData = 'true') + +############# getAdsRecommendations() Scribe ############# + +struct GetAdsRecommendationsScribe { + 1: required i64 uuid (personalDataType = 'UniversallyUniqueIdentifierUuid') # RequestUUID - unique scribe id for every request that comes in. Same request but different stages of scribe log (FetchCandidate, Filter, etc) share the same uuid + 2: required i64 userId (personalDataType = 'UserId') + 3: required AdsRecommendationsResult result + 4: optional i64 traceId + 5: optional PerformanceMetrics performanceMetrics + 6: optional list impressedBuckets +} (persisted='true', hasPersonalData = 'true') + +struct AdsRecommendationTopLevelApiResult { + 1: required i64 timestamp (personalDataType = 'PrivateTimestamp') + 2: required ads.AdsRequest request + 3: required ads.AdsResponse response +} (persisted='true') + +union AdsRecommendationsResult{ + 1: AdsRecommendationTopLevelApiResult adsRecommendationTopLevelApiResult + 2: FetchCandidatesResult fetchCandidatesResult +}(persisted='true', hasPersonalData = 'true') diff --git a/cr-mixer/thrift/src/main/thrift/source_type.thrift b/cr-mixer/thrift/src/main/thrift/source_type.thrift new file mode 100644 index 0000000000..913739fa31 --- /dev/null +++ b/cr-mixer/thrift/src/main/thrift/source_type.thrift @@ -0,0 +1,123 @@ +namespace java com.twitter.cr_mixer.thriftjava +#@namespace scala com.twitter.cr_mixer.thriftscala +#@namespace strato com.twitter.cr_mixer + +// Due to legacy reason, SourceType used to represent both SourceSignalType and SimilarityEngineType +// Hence, you can see several SourceType such as UserInterestedIn, HashSpace, etc. +// Moving forward, SourceType will be used for SourceSignalType ONLY. eg., TweetFavorite, UserFollow +// We will create a new SimilarityEngineType to separate them. eg., SimClustersANN +enum SourceType { + // Tweet based Source Signal + TweetFavorite = 0 + Retweet = 1 + TrafficAttribution = 2 // Traffic Attribution will be migrated over in Q3 + OriginalTweet = 3 + Reply = 4 + TweetShare = 5 + GoodTweetClick = 6 // total dwell time > N seconds after click on the tweet + VideoTweetQualityView = 7 + VideoTweetPlayback50 = 8 + + // UserId based Source Signal (includes both Producer/Consumer) + UserFollow = 101 + UserRepeatedProfileVisit = 102 + + CurrentUser_DEPRECATED = 103 + + RealGraphOon = 104 + FollowRecommendation = 105 + + TwiceUserId = 106 + UserTrafficAttributionProfileVisit = 107 + GoodProfileClick = 108 // total dwell time > N seconds after click into the profile page + + // (Notification) Tweet based Source Signal + NotificationClick = 201 + + // (Home) Tweet based Source Signal + HomeTweetClick = 301 + HomeVideoView = 302 + HomeSongbirdShowMore = 303 + + // Topic based Source Signal + TopicFollow = 401 // Deprecated + PopularTopic = 402 // Deprecated + + // Old CR code + UserInterestedIn = 501 // Deprecated + TwiceInterestedIn = 502 // Deprecated + MBCG = 503 // Deprecated + HashSpace = 504 // Deprecated + + // Old CR code + Cluster = 601 // Deprecated + + // Search based Source Signal + SearchProfileClick = 701 // Deprecated + SearchTweetClick = 702 // Deprecated + + // Graph based Source + StrongTiePrediction = 801 // STP + TwiceClustersMembers = 802 + Lookalike = 803 // Deprecated + RealGraphIn = 804 + + // Current requester User Id. It is only used for scribing. Placeholder value + RequestUserId = 1001 + // Current request Tweet Id used in RelatedTweet. Placeholder value + RequestTweetId = 1002 + + // Negative Signals + TweetReport = 1101 + TweetDontLike = 1102 + TweetSeeFewer = 1103 + AccountBlock = 1104 + AccountMute = 1105 + + // Aggregated Signals + TweetAggregation = 1201 + ProducerAggregation = 1202 +} (persisted='true', hasPersonalData='true') + +enum SimilarityEngineType { + SimClustersANN = 1 + TweetBasedUserTweetGraph = 2 + TweetBasedTwHINANN = 3 + Follow2VecANN = 4 // ConsumerEmbeddingBasedFollow2Vec + QIG = 5 + OfflineSimClustersANN = 6 + LookalikeUTG_DEPRECATED = 7 + ProducerBasedUserTweetGraph = 8 + FrsUTG_DEPRECATED = 9 + RealGraphOonUTG_DEPRECATED = 10 + ConsumerEmbeddingBasedTwHINANN = 11 + TwhinCollabFilter = 12 + TwiceUTG_DEPRECATED = 13 + ConsumerEmbeddingBasedTwoTowerANN = 14 + TweetBasedBeTANN = 15 + StpUTG_DEPRECATED = 16 + UTEG = 17 + ROMR = 18 + ConsumersBasedUserTweetGraph = 19 + TweetBasedUserVideoGraph = 20 + CertoTopicTweet = 24 + ConsumersBasedUserAdGraph = 25 + TweetBasedUserAdGraph = 26 + SkitTfgTopicTweet = 27 + ConsumerBasedWalsANN = 28 + ProducerBasedUserAdGraph = 29 + SkitHighPrecisionTopicTweet = 30 + SkitInterestBrowserTopicTweet = 31 + SkitProducerBasedTopicTweet = 32 + ExploreTripOfflineSimClustersTweets = 33 + DiffusionBasedTweet = 34 + ConsumersBasedUserVideoGraph = 35 + + // In network + EarlybirdRecencyBasedSimilarityEngine = 21 + EarlybirdModelBasedSimilarityEngine = 22 + EarlybirdTensorflowBasedSimilarityEngine = 23 + // Composite + TweetBasedUnifiedSimilarityEngine = 1001 + ProducerBasedUnifiedSimilarityEngine = 1002 +} (persisted='true') diff --git a/cr-mixer/thrift/src/main/thrift/topic_tweet.thrift b/cr-mixer/thrift/src/main/thrift/topic_tweet.thrift new file mode 100644 index 0000000000..46552d4541 --- /dev/null +++ b/cr-mixer/thrift/src/main/thrift/topic_tweet.thrift @@ -0,0 +1,28 @@ +namespace java com.twitter.cr_mixer.thriftjava +#@namespace scala com.twitter.cr_mixer.thriftscala +#@namespace strato com.twitter.cr_mixer + +include "com/twitter/product_mixer/core/client_context.thrift" +include "product.thrift" +include "product_context.thrift" +include "source_type.thrift" + + +struct TopicTweetRequest { + 1: required client_context.ClientContext clientContext + 2: required product.Product product + 3: required list topicIds + 5: optional product_context.ProductContext productContext + 6: optional list excludedTweetIds (personalDataType = 'TweetId') +} (persisted='true', hasPersonalData='true') + +struct TopicTweet { + 1: required i64 tweetId (personalDataType = 'TweetId') + 2: required double score + 3: required source_type.SimilarityEngineType similarityEngineType +} (persisted='true', hasPersonalData = 'true') + +struct TopicTweetResponse { + 1: required map> tweets +} (persisted='true') + diff --git a/cr-mixer/thrift/src/main/thrift/uteg.thrift b/cr-mixer/thrift/src/main/thrift/uteg.thrift new file mode 100644 index 0000000000..2f5c4198d2 --- /dev/null +++ b/cr-mixer/thrift/src/main/thrift/uteg.thrift @@ -0,0 +1,31 @@ +namespace java com.twitter.cr_mixer.thriftjava +#@namespace scala com.twitter.cr_mixer.thriftscala +#@namespace strato com.twitter.cr_mixer + +include "product.thrift" +include "product_context.thrift" + +include "com/twitter/product_mixer/core/client_context.thrift" +include "com/twitter/recos/recos_common.thrift" + +struct UtegTweetRequest { + 1: required client_context.ClientContext clientContext + 2: required product.Product product + # Product-specific parameters should be placed in the Product Context + 3: optional product_context.ProductContext productContext + 4: optional list excludedTweetIds (personalDataType = 'TweetId') +} (persisted='true', hasPersonalData='true') + +struct UtegTweet { + // tweet id + 1: required i64 tweetId(personalDataType = 'TweetId') + // sum of weights of seed users who engaged with the tweet. + // If a user engaged with the same tweet twice, liked it and retweeted it, then his/her weight was counted twice. + 2: required double score + // user social proofs per engagement type + 3: required map> socialProofByType(personalDataTypeKey='EngagementTypePrivate', personalDataTypeValue='UserId') +} (persisted='true', hasPersonalData = 'true') + +struct UtegTweetResponse { + 1: required list tweets +} (persisted='true') diff --git a/cr-mixer/thrift/src/main/thrift/validation.thrift b/cr-mixer/thrift/src/main/thrift/validation.thrift new file mode 100644 index 0000000000..96a04be3c5 --- /dev/null +++ b/cr-mixer/thrift/src/main/thrift/validation.thrift @@ -0,0 +1,19 @@ +namespace java com.twitter.cr_mixer.thriftjava +#@namespace scala com.twitter.cr_mixer.thriftscala +#@namespace strato com.twitter.cr_mixer + +// ValidationErrorCode is used to identify classes of client errors returned from a Product Mixer +// service. Use [[PipelineFailureExceptionMapper]] to adapt pipeline failures into thrift errors. +enum ValidationErrorCode { + PRODUCT_DISABLED = 1 + PLACEHOLDER_2 = 2 +} (hasPersonalData='false') + +exception ValidationException { + 1: ValidationErrorCode errorCode + 2: string msg +} (hasPersonalData='false') + +exception ValidationExceptionList { + 1: list errors +} (hasPersonalData='false') diff --git a/docs/system-diagram.png b/docs/system-diagram.png new file mode 100644 index 0000000000000000000000000000000000000000..2a18fc4153bb389951e617c7c9abbe2558d9d2b9 GIT binary patch literal 95955 zcmeFYcT`i`*Dj0&Q4ry%h|<-gC{o{R zI^3KBoNR1t+z;;GdBVoVk;2AyNa5%a);I0fe|=_s9P_wu>cz&!b!zwT09$hES=N^a zyq@UXVk_zvnqyrYcGT9>W@9UjbZKd{ z7yOu!CIqz*G&Hw5eZPG##*YT{h+9~7GHWEqf;*)Kppzjd>2riNMA`6j=s(9ZHy*5+ zVS}P(<%zvPVMz<>HjBmS#dP(%ZOc^E4%!^HV6;?r&YxlNTeWbn{X}y4ntoL5dQ6B~?&3I$=o)jM>1Zxb1Ac*O^2Osl0-{ z_n$x;FK%MID+?RY#?5W?W-^0bS+W$ArYWO&S=h?zQJ?bPu?8ekUg{&xd>`BIR< zP_;8`-x!nQ>Iz*&c816ZFH-m7Azi2#JVoPg=y*fW6ow0Swszf5*Y)>T$x+`L{>Zx; z{>z;Yb-@L1fpyjWfP*gT?CP%=9KFHobnfB`_l&>{>S(?jU&+`hwVm`#`aj?4r=zd?3Tn-QHES7L z^gIHpa!gIRO7As3H;WFa?^$0R{5aE{LS1Aw>>Gd_G~k@V&TN+Gca7esCD|_O(66eS zuiuvP$TK+#Ha)zTsUakd@xiFCoDOHC{QD5uJZfv_OEs+5I@Kao-%`Y@dBOno_lJ+m zSxwMmWCO;Gx98NgHEYK=8txB&P63JgXH=0zf0*|Qg%~_2eD>43E)Zsv@P#INB8AUY z-2+CTU*eZz*Di0J&OomvK3HTL@WrIndMmIw8Wii+%UpR1WvS>SsL0+pwzf5Wt1{1Z z%eW|-D{)ORYWwoBsNO~O&zCoD3?AHh1^eVY*T3x(EIgG0+lj$3b@iBd7byFLnU&So zYM+(A{Z6Q$ITm+()7At7i9*m8zS2_3U)x*!GSM%Vo3@#Z%X=mhnk!X|a}v)sY=c?n z^W0a5DoJd{(=i?I2~nw^4r_RkbjpPuFM46n=96*XhVJ;48 zqFZx$LAIEl`QPu967A&Nd<>TW_(0+V*9^)fOHt^#He(}3X~D&auL_|`JOPHpylBqtgk(0S998MtuffVqG!KKmciR$>C zmZ0z%J?;U|lc0?xnq-WiW7RX$tnx7~KD)VSCC(J@0eQ;>y#d{5$)LR-|c{WpI}El#_8epI47^fB`x&i-Jm~ zP+1K&$PWJSE*p~7J^nkI-CNxHG4FgTk9L zztazkVseeH&K58$GJY5uPD^h84oNsW{2?*1#;_Gl{{nhDi{Wlw?tv|rRoPyZTgg)L zQaCg+LF72Oqp$>v{s;c%RDJ>qFTFX9va`CZELaM}kriv`pZbiy&?C>X5BNdBH4Ia# z*ThB&v~%fA>tBbb1F4TcE~@^N!Un0> zFQu_Y^?OtK1D|0&A9d8~u8Pch;!elxN4F5H)BRH+)c(}I1l1|fsD=lzl3*ftS-7){ z7~FE-*nSTEPd28KtvlT^!;7RzwM>~88n?o;6m3*Wde5xSl6g0a{3?_Ab!7x_-}rKd zf$1uWxW+Rul{hC)F!Vc!SncUUzkSC(*0COtyq5lpP9?cFkU>P0q9yE-8}lhO<&}3) zi00cUru&F+(Jtt27t= zu(@Hzx5d^jvkh$_1_M6B4DP#TWXuIM%VY$#)^l8Ke>^^UYa#sp*@SDU$3)xE=#AI6 z4X9xgH^{k-4;kM0(|2-D?K<+&<)=0X+Gxm+18j~u#Rm1c=6uU9V~Tu`;94u*{Nws- zxD#lY+-ZSYnc7Qr-Au+4s9U}L6H00S$L-rx;lKiZZvVGg2y|r+lyeWmNBe zIh%e7p0c-!LTBT-|=FK9YSo|lbRjbY;U#~#|M^*@4c z6H8Rzf6Fa?o%w$hy=Ph`B&fQ!q`04B`w~sxcST~F%Oj#Er}c(^YPyOkvcjnn!OJ0p zneL#m6>h4amfO<5yXddH;Ozw-$6SwljVp~L^twH08S_w*f^U1M%Mb`%=q2Y@RK&4D zsJTVb`3MfvhH>bjcJ497{ViwfLr(^C;^%2ok;lkB?j}TqyH(Kmb`S-VB7QEk+>hM# zp1`yqNUnSg(l01;J3@)dmN`*U~2wE zw&!vG8P+u35vC*?AnWLLv!e2FS)tf-Y_*qor*WM-n`tX!UyJb>W{r+8ClgC#AMIXT zO|wW96~8E=<@+;@Uf0DO3Z`@CEL&CYse-or-&C=)z%$obQ1Xcf8aCPslar(0(m*x; zZ-#u{izUJ8YenMZZ7VU3-Kfj$so2zw2lB((cXwxba4_ z!=FZf{%N)T|LDfoySeK*{Q4LWkFrf-1oU3wH{EHdCl`ACnk}_X<0%ZPEW%7}75Sjz zT!TDEpCgDxU%n%#l{~uUL~5t##SsMo5X1LU;v3A6LI}D8WuVqTyxbSrfJJw8TQx%o14&BqNyx%f6c<+m}mg`yiL zJ90U=PCa8GI_CBSWWyk%2-+cVM-7!M4*r;?gg`e2-z76lyrc$=R>7KAl0hrA^OKFziqA02rxOc4=gTEBwY7B{r*cqhz#Lb}BE9gRG7p zN9I!lJIYtjRIQ4tEj<&w;ab9#jL5z0CLB2-5YgSSqyNqJ-(@bu+LYPYzI0S}iv+dU zd-BzBJ{bBDaJxbtb;Su-739ZjGH=P(@anV^yv+A8@5|iVZM~%xX$J?hADF2tR_Ikd zqpwwt`5q|f)*&TyZ{N(B^0|`F9et6n({GsRN0<DC1<7J?U6=G5Bi-kDPhC+ z%TmF$1bhggoJO>)Z0J_t1hnl;15oSeL~#f{^@siKi>v3*!iRC6+{&F-cdCNet6kVN z+y;*d@*`Ti{~iYT(EG=QzE#?JFGh#22CctxG#^GZT9OdFqg`ors}aM)&B2TXI{A~P zim+}r=?mv7UGgI1p;~(D>S*>(vEF6@S!!sbzTUTfKkWNQM-als7Am*#!>UW&TvYVe znC!Fz-G7uXsoHnYEq(lj?_jcl|J;vCNcSq&v!Mp=$9!(NTM8P3W4|r`_69D&6^2}L z5L)I_527T>h5}U{@%6v*)_3_-FLgAKXj$7RwYqwPK^yMC*;V$G`1pRDF+VYoI&$$P zcPUZl+lDvpxjWabMH0-2!Oq6)mJw9=zOTW-3%g$K{(GL+l^oi3&zyu#yyHg;I7io? zKkDscbX^i&#$6eGCK6t0knF^xxm;MVsn^yPq#5H1dKWUgQ298r*eTXQnDJ@jE!(gQ zr@5eW%=ys^Y6;uWz&v@!^4JNv?WY80n&_4s@UJ3f!KP^&Ajnl6<0+5!nGEhL$i`N|oc|W0io79k z%%t<=pE@+B%VD`t)Wvc?qi2J;T#@ogImal?$PDV%_+Kfq?qU0reP1+T7X@_Y>?tk$ zVbVn?*SuwD;@xDhfIG4OKoN85_0eD-u3Lryb`YLBINZ{RyR*{Cqm7&uJG!ikB zg$az1d3CsVQ+=IC%xOC7Z6TT=+k-sELlsZ^FORIUyW52xfAk7vhY4;8>3+tZpz&yE z^k)Y=cFiQAj#K4*WEW={AL?>}8&K5=CWLL+RP~cL8QOBKQ1>l^Je4n;pS%O#8d`tq zGupdNy3HwvH`*VC&U0Qj-93~s(!8{M* z9Ys-8ZP;$gR`}t^fv(e7zshY35dPEp){+cf26hAg^0xtz z8zRR48wd9fh?^D!$Q9#P-X3Vzs7Fbpo7(I*4&C~oXj498pZob-yA(hPtnoWIE8tVU z(~;-8eBH@B!u!qQVJnwifdJXiQBc)({B-o|5%;sZ=fUOw%DG~|3e_rljtI1ct8u1a zAsuMcxRy~x=zqmoD^ElsyM~U4(Fe;ktP1fz&AYU04Lbw>6?(fP<-eM;zU%e>pFx=9 zE@Yqj7Eo(|dq)vpNT7&A+Oh_Yh{;CO@ZoZrAe&3i$OB>2rh|W;C#U!aw?C)iU)SnT zl#dz_gmX83!dKBh)Yg}5ROtzuFWe&OLpb+=|E@#3X@l00({cuu-rp4#Rqffzbgae< zD?>M?@TZjaA{n_;zu~O2_E3rBW65Eoh6&}CwtK=miTshZzhZUC+05~faxQlG2SI#4 zozN5g^Xz1n7&qCf=|5HGis=c`(-qjNEA0gTI2qOxyAJH#p+K=Oa>%^# z)n0wCn`Dju52Dn<_V`liO9`trpes*b3xkV3DqF@xbZbzcnqW-7o7wB_6LYKl{He83wFZm{dX<+!8a)F+@#qWw!Xl(oGWqpCHKmG8O9e8<#}oqU-fHU*n8u;tso2@ zV~(D?W16SCT~czul7!W1qlHvr zh&~)XzFPzRb7XE@bnOmpHefG=KLZ|x^9~s+iFeDJhcGu3rZyL)uMkjF{}PKJMbavN zVA6MhCgb5VL>@1QVce+7%!xB^B$HlYwxJ(|#^~W|9E1iw6QZvvzaZUYb0{K26GtyY z=#0`d&eBQpQUVHoCt9<7>R)LqZTAm0xE79a}_iUIkmclWQ<&|Etwxzl5?Dv4>ewpWN8WCKp04yqdi940T* z2n>Q380HxTM%TNori=9wsNrK}9T#1H@bI@BoW!@n#vpm|#&j}j!IN%QsoRE9kUSdk z?X-hIol7WJozFEyCFR>8=dMn*@&y;aU_?RaNF0eKW@23H5p!3nnSeao`;CIu<&8b{ zUnUhVdNf5g^EDAmog>|P%3d(}4WkobJi7^dH?tVU{MA!xXfpLgiTrKyrIXskjn~Z!bTrKK@6i+N zb)0oRj{tIvdCDinkjv9&y}6>O8YZ;Ptw0>%Bykbk)B0e>N0+pt%2CX=ph8aq#ZKDZ*W?%Bi^>*7JA#O61u1g9mEsQhFNKLj{bAP}I>d!4~6e?x3d8;-lnfV@W} z!9OcMq9DXy&ggc3h|Op7eJ#YAJ1 zXT5^XEy6DvGv>^xfpaTu!E#G{laMm z=+0+Knm!LylH<#4eOa zgV6-#qltXC49@m2!#Q#5`4pE06(n=>IPS@TsJ_m z{2qoXYfBkzL{ou285oxYpTP5MRCi4`LMBzrRRXh_=^}R&}6rx!b}- zH~^p&#A!dIA9(MInlOEsW}PM?{boq-^rSv)F5ASUAPA8shPozT62;mvd*)2w4}p^8 zrOmf+@-5vcRNv^pt5pqf5IdISEIohPwzrCT_B4>eyqbwA`#$dyN#zJP9E~%^f;V+V zF`o(q+I3tM)fX47aIF2-!sq!$nsIBo(X7gJ#+i}tH@GXCfmNy#@G!SnN5biw#y>6p zI#w1TW&#+PaGAWyBZc}jP!!8P!ublomv&6J%?Pw`RnuH3s8B0ll;C9{hG-la@EkN# z-CFOnUf}HL3{ScByxAc$M8j#Uiu$2z(-tYLW;qo;>cK`heaLZN7g7e=dW|bu+9NQK z?c*SFR$pQtr@p+ECtSiBT@*gF*$hCVH)%eX8G3gQRE0l7_MC7pF4EL7dbD^%0}M1XGN73D7XRTmsIn_zrORZK=F zC-7hx!A3Cj(!-4IE#!xM?w#8A1#Tkp)Od=LIfK}#A#66%X+<6n5!s54zQB zE?ckB6X%{W^kYV+NWwFr>wUg!H&=q6!sy1ngCU(s=yKH3&d*3v29YnL%DOQjM$3>& zKDMZcXUJCIKdIgRH-&HiQhF+zJ^iG89&ea&HFEHiQp!}0;FCPr%W0~tV#=pOZBuBT zD%qpy=;a@ejhBUTH&XdZBr?Z5XDbk#w=kFR1{hNkE(C#a=RMlV%C%fOQ16Re%ud@8 z*K6^9AM2e4|L*ygIzJn^Hq&?=W#oOO_0%Qm*7oP zM*oeB5f3#ts}Kuy8CmR?khs2KpA#xQF+m0`v}yYJ@(BCwz&FPaE_Ft%$;Z0B7~h+0 zt%eJ(ZdndejKQvq!XLKmot&|JYUEtDfFGk4tOQCRquDSr4gM^>I`;>VqQb{zY8-M_ z^n&DYnnoRh>m%PAODq}*Mj|M|>+nnfADZ(CGW{U1D`9KxfVc&F$#06uIBm6gW=f&V z_$`+`-_pytEQ;Te&eLYP87BZf2Q7tlYAF5a~A`4#M4+%cxg$A7r( zng^qP8yjUCwzMUP!)F#nE-vR<@1Sc-BLuB3SWyb0E*7wu;I-AJsbsGjgenU6zLqST z``zLYAZdRh@5hu>M?PsR-1t0n1T^%s?E2AEDw_sF|MvFC*BDvlg)&75b{K zv_UgaQ*pPV+8BA0zxttVjFEPEQAb*_h>xvdWO=vxBl6stu_RI0MmtKn4zmJg@Ku*K zlrg_BOvyigmPiQh&79M#qbZ>4n%I@_wd#CaUtN#aST}Ptk4^dgEpeRlD$BWqBS-27@Qz}3wz~B7S!1zroy!?OTbTqGZT8Zy!-Lc#J;1M|ARyK z9n`0S zyXJ&Oj#D$K-QROxcnUrG7&{UNCQYobnF~_VeXtBdj4(putw%KuU>gn3*-sed{?wu} zUm1dD^Kau0+bll7cH@jZ3Lb(YTR9C?`LQpnIUg8PG)x_>IAf7T?j4A=K{t@s4X~!0 z{;$JPPY!>7#n;;vQ@p7E1@7?b#Rg|>Lonssa8t#t=Za7Y& zGuD3(Q5L2JRpm^srb&jRJ6jZ#ysdxzu80JqRqEExnPH0Z$O!V6Gt-jU6m&5ei`!S{ zVB8Of-Og`$-H_B8O6^R(ZL)esw6LgDzvZn0E<~8U)x}<&>G1`D3h(Ey6!M!W79no{ zYV})i2H=hM^hmdA8Z4mr_x9nZwQj=3t2&(${DWt3NG7NjGmz5#c#SHFzZh2HrXr5) z(LbI(-6HjN87%E!p{r?JUi7vVZ5L$c=1LeunUiX7R=I_2&*-5w_^<*(`;JEKPx&U9 z7a0s(%4e(IT6rh-+m*Bk7%Bt~!yEV_3*Vm+_6#d-?%$YFty=rNE?6w&bAGNBnm4L| z1NMLypRsm1%jy0j5~<7^r*D$a3Pd-9)YI!`|4EaCYZRFPrd#v8LgtfcO+Cx2=Ft+> z=)clnBgJ`{BXL$(lT~X;|8(T#=XzUPT{RAx+z0>N>+)Fu=rx|xADEQ~oP^}dw=W>p zYP_lT(P!$MvUMzM^d2|0i=GQ@eO$iM6qw;YPvy6q8{jA`y|N&58e)Vn1b9h_KDv=t zW@W6MkLmoq;oUY+XpRX#iC*Ol)Bc$v=GF$ydU?9>-6IW^MK zNMTAx!#uvN)oV~eHQ`ZrL4M3W6xhyD@!u=%|1T9%S!m-T5s5s|UY}JTEoJqaIx@oc zYpgen|KB`cZ0EFT`bVWN**U>>|+3Z|cG{qh#CW{K$ z;&%H84{LzC$1h73RC(%XO4z-}Eaa(^fB%$}aD?qsvZ^MtWTP=GqKOjvhc|gGO_e=s zHNHuKMUZ0wxA*zAB>&l4R9duOdc$h8NnfBsu4I^O{Oy5otceD)P*k;U@W1sSg5!`oj4kEg<*5_TlFj zynkRCtwTps_r#h1Q|e>YAYU9JaiBqNdtFEg@8!&2Iykz$NL`iUJIt$lL5EYvAFhCN zb&VDH5F2|pi^g`1fyXbFoR2s%GW*zgZ)Iue-r1AU&WXkKt90KcD2{G*ahwp^Lr+5! z_Zb)6jd9!g%`B!1&SJ-q<=uA=slWF6PR$Yz(HH>MZ6r!mDdeSGBqSf#E?13{*kM|~Q z{925NPbBv_wU6_AQuKb_SJW9b7FR3ZZLfPN&OOoW!r&>ojKn8|J>=i)&4o@Q-AWN; zB$;OjQK9y=8ODXSiTr1F_0x}kQBio=4$ z)L45+CbB%JmaaFsOL2t7p=PbXe(Q>1CP@%b4|kW~pfBTaN=HD3qfB*)C1_l#)cntQVY z&hb!;tv0J(L`MCx0(oR!HDl4fGT8;7E%#Z^eE;t=S%oH6?QAjM9@=11`KdZNtTi)n zjjgx;_&K%qUGw|59jou%)6D#{>VaPXi^CH$`)1K@yk}qW_Bl{S4daI}YYW&t;*0m) z0mo@2S@%7QYj@mldnirS4|Ry+W944ak%glDvAp)M{(g<0BYTFmBCuy~zqCmDX$y_o zdrjEl(fiNU`MS?W+9=%h7`=AEMPNxX2DL~(5SLz9O4m?_=ok!|pw=iR*4PgyUk1zz z+fhZrA*D+mt8+#8n2l_(iZrYW7%<@LBC8shsEadsu*AP0D;S3#Ig6?9O#?w`BB~JT z5$vz6ckOHEFxl6uw$G&QTxE1+7z~TIz1xwl*#2sHw~1OR=G%o?P@XH2`7?C3c!NJO zCdWTkk2W{_^rpEhQoH>VC9~4vaafFRo7%YrRk=j#5vimK-=3fhRhg474Ax!2LF;jV ztB0`W=P>Fg12Z7Qg{%D`^W_(+x~U<7ql-JwN z^z7q5d|Ds2Sa+c^xT{S(!ZV}2gMV)l*h0O+c1vWL$2Nz8+8~s4qeYV2rMK6IqtIYY zjfB<4CxAL5j=Q%8B6qx+EeW3e!ba!G>Q+}!{jrFJ*zpa!FtFEk)zd)9M?09_XZ$C2 zf-NW;3kZ0rZg=zxJUAlL1byo2F^s5)dB!RpRb~KpDM;{v1ov)DX+CX{_{5YK9IJ2l z{T&l*v!@ugc8)*O+j}_w0`0xn!k}JOPII3@Y?^P=Y{@E@6xQwBFpeN;4N#i@1_9iZ zv<6;hRIgrON$~|d@2+}6=a!Ad>eBNxa;ei_8*l&KSn4xc4nyM{I%^sFZOEUP5RoA9 z=Uv|WDdGzIpL$@|cX0sx9fDg~o>YR9CFk;*Y0o69`;bnhdpsr&t2_y~KZHak47_@n zy($Zaxd^A-?*vrWR{nT){&&gZUEfZUS*?c|p6rhfq`#=IW_&3)!MZWUgv%{PuM=>C zLfFYcEryH*4f(EKYH_oGZ7l_+!y~C@p9;1;1c??dDxJ9g<;LV%Gr8B|4HY>FIw}L|O_x+35MgV81s`lA|{55NZt@9P> zK#+@Zf{=OiMo8%yx4JlTXDu7MuSLl}rl(4Fgid*%J6R^eWW31>2e?GY=k|a%=UZCoN z4U#!(BPwDs*}Q68iCxr9z~YE0wHIvrhbz)s=Sgph)E7I312cpqpJ#2BEJm`5htp2K zPtMT|1J*0lx7y&tUhsBc`~t}f*bZB5R4CdE@P6<%X3@Uy#=RLA+P_q(vvEem2_(T|V%HS$Hm1E42)T{CdSoA&~LbuaT%%v%rBu#M99J|surb%LUj0rE8Z z?{y0(%Lf|k6%q4_XZAfnvg)qm4880$fq7q4W<0U*=534gFu*$H{!m>_WOcyH>{S&> z>xSuf7&lGnI__{&O+POQQPJoZc}c1KXe&|k=!3(@_=mWoPl4Q{NI;nF)}qxxf2jN_(@LVGuSs70Oe$+tUa++&&r@*4Q16q*asP;_ z@^SYV3`-rt-+^~T5c&2TJR92z`|g?6soYRs+RG`oFRpm4=J{g~#ST$&*CrqC z%$Ep9A~VRQ2jCKJMt#9Qap+sl#8}n7OBP&G!WvE(C1|a?FbI|#?jEBtSDKsc}9*E zw0KyxGTZdoD+i9Sy{hrPhAREWFikqqf?`G7pvvisc-U7Jx3sK7GDi~u)ui%((X2z# zCEMz%`0Z~)D7RG|*4s;@))O+^>G-Ku_RP7*9vg1FPH-QZJT%(ei44+_Y&2(d~j1-SWzvzb_1ad*xYZ9VvG6Z#Pf0I0OgBua#{P8isn~XOV9T@Ze znuY{zUP^IHQ7LP**afdX4@*c>PK9llh45hvdSmrf)fyX{(<4D5-i3lsFxe%*UZaj7 z3f7d+1;WaKKT(wR%x+XjEsG~w9!Fm+*G@7QqSq1ck^jc)OnR=>VYsL|Q^P1*V{gs+ zXFbjV68uLPYC;R*BOjkodIpw_i$S8P19^)9F$P$%PwGwE{OFodrrPYg=v^Z06Yi_N zF()^`@e6@uH5_{nwh}t2@vcufH;CCHqe~*bO!6np*PdZ*BJG+lx|{=<=^@-lx_6^a z&p3!8+tO)TqCtH>D`Wh+I?4*xAe-66)fue%L$BrLGOyM(Z#aCALBjH{NBh;xE+uU5fD9eIjA)WZ#F$~ zKgcz*k<)?C+9hPh0IpxF{7$~7q2^ z`X}4X2b<(W)_I1Gur*d2ZV!mNF7RDkNWWS+XsR(+TvYx52DRY5;8xO>&IF`En7%zb z8JAFV^*g2F+Z(l>J({YpFw+>*7N}pXm=WXA;Kid-uuW!=#aH;6=85KqXi;^`E!=Z5 zF)xWreQmfn!_~^N#c#0;OEiP%M&P3jm_MbAu5bB!oDhugq|N%K3oH9xJ9@vsb)TF} zP+`&2tmAt60x*Z$79i0(;7>>yXGc0wJx6~W&g?qc>M_?CJvtSY;g1nkyirG!J7bae z6nlMSEo4%M7!%o*k4n;RI?Ebq;s@*R+I85vmP7+R!xDEIyhB2uG2b{kL#kUB zlAO(Zb{rlosk#W77r(PkGayi%?|G;{VP_dQ>rD=p{g|-`SU%FPg0JIpzVu#ZCDtM7 ztF-N56+F$JrtO4sHkZ+(*mA_CmAxwW!&T>cXb58OFq~rnQ@a#57As|;A=IO22l6XA zt=1RM(M0LA5^(E^KcPxwkW0a5Y)G)mu=MuXkgcF>QZ@=hT~wt=R1P?UpmOUy6(|D? z9B*|9@S!#!&%k6sVAfxShpF!^2j_~J+V(0GyTxr;2D8^RTwi(s$Vwx| zqHt`XGv|d=(yvB-;54yc26u*sdSyv*9<+}vw1KAsdJDx&!b+~p)=@d)s@A@pU$Nj( zJ6&n@cvvV`3=jrZ;st8KGoSK($N948!7Ta_%NS!RjET04W4Go5Sh3~!2|Td66Cl`A zrFin|{PMh5p*{=Na17-3fo{n^;Y~ZcNPPuVXEdrLENw8zJDZI|rQp>22g(lm!}Ly8 z^Qu+FPE6ekUvDE5PO!{om&KrGFEa#WT;2L@v-+XB7Lh-D^2LO$j2lU!uty#c4!*`| z*5jbIOS57}u~6| zkI30=6k~kAU7`b}D})C7F!NkemZvAHXcPHP%%>!oHH!$lDd$fl4(jCOQ74*f7S+g&k3% zyy)*ooPhhBy840-i>ncLw5^GE*&b0C?=fW$$ zHI<0g(k9yS=A98qXSZIq6_+}zvVWiOy06ye7Y>>XX}0Z01r{=2wDC2ZFcbIGoB(S; z9x3`Ekm=~$KubKC{&;z%&8NugHk!)<1Sx>~qIZ@Wyj?Yq>`-rE?X*1z7^qgt&+@F- zGom|73|btK5f(IRe$eqjp*e6Vc$Htl|R3K%!`N*CScT2)*~~+&ykxVDw$DT z8JNabev9{_9((?~v;%5FwVdwuhjl)|uf3rcz=N z!n^Zn0<(K2ZgiW8Qeh!+<;q0)&>3C67CJ0{xC=Y_lyn=&^&D_Vv-}mBg*A~94-VwG z*AvDA-HSo)!nc|xSnsN66g(tVV()FkwBw3==vm6~L8-8GuG=uQTZDyzcL~=gLAa2< zLy&jujHmHn8g8~)XXh$yHK*yu&_l+AczUUJQ}v~kfok7i{#s`Kys6FVI=W8cXrWB` zOFM?ZABkVT4-{VEZIJTUe-hax@a+w`wv9%LQ)W@zo=x>@ zR9HdpfX=*tYuv%qsAS;0>KY*h*L_AX7u{6TyzGeLliiOe5Q!U8&V}3l6E`-P>sljJ z@u4cF&(SyrSH14D19h$0K771nbC)Ea5-wPYJne4iQ*^YTBlT!E)x-NZf#4rJj__Jy z2+-cs2|XbTlQ0?1|SXu+vj+9%Y{1)mV0e7H{j&`&5n#7Ik zzzO(4V|ElLm`}~G8jj%dNAggxGDf@$q(Bh?ZhwAhVygF}^6-*?rj@R%5vJG~)w9vf z^Xnx6TD;d_8ono3r77e*CP3mkj25I=1sVDrbW1MPm$CK8u@cejo|X}t1soID37-ay zj#|s+SwRNeL~N!Ow{!4lQ6023G_62{Eby{&$n!4wE-DR7R*dd0`v-WDM2r`p~oc4^%dnDbVbOpdQyR_Al?7p7+HzY$|XY~zrlwv05 zj;k9&f)kg7zK!HE;W@wi6^SQqk=Wf#ztkviq;_rRmToq#wZ&=L@*l>JhOB=7($?N8 zU)YmNN8xBca1W=qbCwT4C+oT{hLcAOuLsjVfSyr0+hXC|iBPheMK+G9xj558n@?%m zC3kH^lqg4^{_;B9E;7N^;*!VTPQaYPW(W57+X3vUoC-c%QFWYGS&URu{-->Qj*RSt z9|XcSpH`3y03iWlFSqZb5O=m)d@#Dd=y~Hql?Sj@mgu~}$Z9i@bWzi(`Xvs~=uP%9 z&W6sixpD82?_;F`&q0aGh zSAE0YMRTzaPXH&S0=j3gmR1fsjt^+0c?zMxkymM_mpS_35Q)V(dEWH(;1l-J&mO=q zgv6U~VO|o+R1-iiF3QYZ(jUOP@XelL)p`29q!Yk{66>Y3mLypA_(0)x^OpqLBUhO0 zq7cOP<(gMh`7K*b&5C9U>xr?Df_ZRSTz8W@`5PE;CyL~U(eSpcN#rbW-W9>DUQb6#H-n>@qXF2(FuwVJ@j0KJKrFZ>hLGg1%WGL`x zaX^%4$ks^J##{S)yf_*e-w9yAPi~4g>8+k}XT6BiAe3q)F;ew%?J=IbS=2>&FOG|B zs`3%m-2P?6XscZ;c{xkZB&L|f@^AvQ!PP~dc+!~LaW&w|&aU??-x!)0_F%HC-MC@% zA&@-=9t+@Ay;1R~;!!aBO2623VCYh;nVE&XUc>5 zwmRS+r@u%T0)Qi57j|#UA2= zk_Sw}Y@dB(ERPwG3xjMb`Y}CowO6wGu|}85Z<+UW9TIx+mDAW$(CmRgO$)3i$l_C=b=B8rw0+x#(ugQuJeAlXOV1({Scz{U~!1luV zu@zMvd@6A}e+&XRme#Q<9ZxyuY?1b;aAl)+1nH9@zr&;8;IFB;Q!X0Syrh$1W#K+6 z`00P~^%h`Fe&OFZ1|pzffT#!tA_@Wu(v6suilbAyyBP?IfJljSi4ro{MvV?Z1f+Ao zHaaG;(F_>zKLfx2_xHa4c)Ttzwmo}xo^$T|K6iZX&-1+;e)_cC;{G<`1A`h@85;E( zf|4hObQ@|Lt$trsZ<+}`SQkCm+O8;5759tRgRtkOkZNgRyar!`4a@sZue>qW9s|83 z7X!MR{a)Jhv(g0^QgioryFC-gSUBh%Wt%5l>&vrG%dyZgNXVVky>L1WPeVrU5o@hK zCSf)E%V00mu@)%8cc17fk@PyQ0MI;(HYJzVKx-$ zQ7zQ?0rrjgnl;Yo4@T2E0rtgJ`SqXS7dlQUi^HX(ud+{<`i^SP_>rRKH~p(q8?TH3 z*=oM}zFjNJ@U?$Vt#U$TD`bK@{&g_S4MV4X9#2&6-_V}lez6K5tj7JlA@j+!sL%~a zozb7z%x{s9^p)cK!*8y`8*`y#F&N|-+)!t-ud2mY+5~=;A=Q<@^tko`O4UXN222zc|i#QmJrGSMA$k68Hp_yzbKzsT!Ds8o|4 z)CVH0#8Q-7(RQU2H%{zxbRhCP^2%RrFKZQi$o^*9ED^oEzHpADc?1b7;d+cWL*shR zH=S~^MRrzhnBVzjQ>%Nukh?}rYSA`S{Y6u%jP7*!ymk;x(-pzHJ@UHv41E0nWsj0| zliX=2a*Plfbpla)6Srv`!UtQk`tAwN85WQt;Q6L{^~k&@2-=2U z*ci|iYt$Bv*zSY2U2im@@I10q`=dy)D&#$R3tr8Vwyo9U>ZG|?8Z zH?8af8>HW=*eJqXi?32w!?w}B=W>IfvZF}rBoO`IT^2VHAexMX?1lRTR zIXj1vm1_-lo6OEX2(9nJHZeMq62FUG`L;c$|57%w{ioI{2hSAR(VjkhA2+e?*u-1A z+A`?%XR|z&98_-#d1#HzFT4hD+{_kaXnVXkhG1~r!9&{04-vJiQT|nn{MwMMh7Y#k zi|~JLpEB_N+P$>Bzt+TyMKj3oc2eHtIpObOtf0n(K_UFM{j2NzSVwn@J)yJD@IrY{ z2<7l$=av!j+83zP-O|+y22T3vVRH}*P_mg0r=_Br$fUdRs}u0zfl z2-^f#a)!4DVM2DwGd}-WSCQm@=E_!OG9craw6v~@s%;pJ%BZ+7KFSN^ZH3jyIrb6E zjRVp)PG{E6C@F~9_B8q-?ujl`)x%ms_Fx}1R*(6myN6iE!3ivRqjMWMKZiO|dCw=F z>dZa0wrdZ%TSKSaS}YTo)q5Tm@W>xo!pj)m&9A9JR%=}>G)!RB?mhhk+DZVe0Pv!) zja=icX>ry4)wUO`3l3PjFo^mLx+P`-K{aMPmEeRU^eRfy7(F1$>MoUF?ny+2qld%4cHcO4l`BxynSm&Bk zf~^r~fIMc=3~P_CzA?Qu8*eJ|1Xj*kGWv7v;v$j0en!SD zJLw4`#o(18$*-WSL+mtTy*YKej9)g!?MjQD!-$mV-e3JkawLFczD#}cGo*)`3V zZ^nj=p9PQAb)+nvHedbB70t4}uk`A>Rc)W=%&Bu>Gmy>842I0lVbrB8TG>JL+R#B5 z)oViOuOrsKTfGC;rO)edX$|JdYGLoZKE0^g)VG~*yoAnz>v-Mv;I%yB)$Mv@{b+$) zGp5!$>Dr4?miHyU<)}vgG>-IkQcu8~M%)}t5aw5ztPQ-bAUxd2v`4Y40gO5QDms(u z9<1!8wI9rNq>_n?H}u38X}ecn_HUF34$@4l6Z?CI6Qe2pW@a(Fne;~HDz)Mzx<-hj zMyTEnYJ!Fw@~AmPBdA?#hf9Mw`6G7(PpZ~UZ%+Ze)viyU_65r#J<`rjRR?g0QaiQht!2bn4C+$C97S%Xj6SOH z@kmvL7hC^-VXnW$c3Umqg_(?OjPHGMzHFWpo;#`d20ORCd0(VB%s{oSkTD?|{{5|{ zoZCdrj;mO-wUaRRn`3{TPyPr#X`3AA-jo|QkS?FT@awzc<=wI&CuPg1VW(%Tc$VL`u-S2&}9pQ(+4S=AW7NJ?fuR=(OMM93?zoYyg}+rKb)UMB57 zOhRzW%&nURP!|C@X;!P>d9FRDpY{$JKz(i}J!P&gHZn)FW@a|E3Gu9t&pFG`Zgo|gokQU<7l-%K^%>}vWlLge>)_=yz>P)JB530*7 z4+cM3v#)5YCo1Lr5LL~R(m8}A4OUp?eo|)|z5n!xeMTkJ73<6!%jU4UmBs6tO!IRik)2W51sKNu#FKy`;K3{v?FWEpB2>q#zDS+IyskvFv zM$>L(X-iFwfRMQA{LGnWU@55j5slI`mwE1|Xs5t3fDmXQsJ6cL_u2pW)YYkS4B52$ zZgTW~+2F{qd&`9rr7x@L^_T%fW$)osC}(EMjV4tguGJ(tQFv1JYL_pPAsh*XuuHOi zYVTpMv>$?t!AI#Ei5ORdzJGOTswsdB>u3pB57PCvhKT0UX)!VJTm0E4jp1=k=eGP?QIP2{aw+P5-y-TP4M#*bs zQo@L?z!Ewe<__+Le&L$w;#_V!YztymW2)Zzs!65e@$0LCpa|sJ7KF5!Z_HSCy8=te zd}LcJc)uzLRuVxy`N<}m&NfWXs1%6+;QRNW`biiJa^-iLw@3=w*Tkkw?R|cl`w!Md zRW3L9sb~hTtiFU}+BilfE3Gz``*B3E;xTtn5XoZts2+qi_j9S<{Ly)A0mMdcmK835 zcG^$AUFBNma^^PUyndywZC|!$#}k)ZcN^C)mGgdmgi4kxeu9J1E~wpm|4y9 z$N$Ep=U3PVHJ9-@%DEA|Fxf}VQjo|6UuO-UR${4J$|2|s2#C`G5bSHh%UlZxRi$D1 zLmmxNkWPjFAiMoIZP2fx4a$8(DltDpd{Fsm-?hlfrz}w6X*y!e`&FY67(Jd-L~PEr zlfUK>qou}jcYh*Ip)e}R=0iclIgN;WSfulVt+lQLNd!Oj{-7A2@G>~>g~#hoc@dG7 zdXj3{Pj!E!y{Vq(rpw)@>eH)RoMwW)1~G*#n}7d2m4$C5_Ft&HPFL~E=XY^9k2NW| z#CjX$1TXI=N_3+#;d<1laLCrk@|PyZBTf&EWyYJBYOCJYx*S4=-X>d75PcTvBRFPS zd*-Q+_V(}>8mq#&-$ZqgUlh^746 zDR|Tw6|aBaL>s^2C=mKKsxMC1hl=WyF5oxr>PM^ zUit@5r7Db29Rt6fcqy9D#(Iw(q{l6$UhkKv^CqFQOL5=fq zJH)ho=m+Ze_*3)f(V2s|gVukJlW(c4s_Y^UXiTHXn5 z%n;9S%pfqvcXw84S?)&6X_;iQX_ueDqWT%aV+QNJHX0{|tGv^77NfsMK{-oC)u}fC z!osXgCOFJvU_k%UPJmN5$N4mR*DQ zXLLIunJDc-{DaO}y!@}*O;qAPRM&w#{+|rN2XJUvOdm`@U7(w#Ly~mW;I_-x^qWA;E*+x6fUA@LuF2YFCee!0H%66Q z4w_`jvGhq1lx8wM7h+%5PjJbVO9lo1Af4$4j0Nw%UTPMo&bq}fJ)v)$uuUPb*gnk) z7V4|mwSO=(i}cssRgR-~jzvOYPH5}9Y7`8i|1lb1?f=I~{%-8j6d<3mSJ=N6v=s<| zO7B*d0EPk)P`8*41tVburK4a7vb}=N$}s?>tx=M859E-1ASRw~CL+UyK?nJ< zNS%k?r6`~K%P@Zmq>=KPpGbjHS$GK(rEIC3>_KCZx^MaTDL4k&6MNi$mj|ng34d5+ ze;p+^Oh@P$6NY*etk24-X;4|*^;`{?xDKXiN`Gs87$Ch*0$@nF2I#&AvDw=!^IrWK zs}q;sX`?xuX(9$Rf5W(wiZ_GXM;wyBUOL?)Im>S&ZKu&DF||kdJ9>ud8VMj6*}WYl z0P8?M)l^6Em{6=L?|c&jKMFKhpOPgi(d!!*=h1+{QklqKAD?>kBwmBvTxs78%(48~ zDXOq*NUvvK4E%ic+qvi;a9X8FaYVopt3l#JFzs$62cQZt6wC@g)MtVF&dMlT+!GRY zG8;sEe)JsGwbB)C8{(^17GM{RpfkVL`no%BKt1mBPSSh^=+oQSo-BPee##_)J8>N| z*K^-*#`9n#Q>mD6weC+g-L;u9m8zGiuDO4_+j-j|)pL@X?@F%6maz^=4&pEwasHZ` zRh*RL(X{LFx{{R)pKQ=I>tn!3wyy&G1abDOs~VKJST`W4872{s`qmf{BDg z&`kB##>7{@am-Ml)<7M+ zY{R|<9imR2HA&2*c3UW%2Dz>pYn=rO~xiv;^SbM0>s)((wpr=3kA&*F-a@`=H zEs1eq3#)Z@+$lU(=ji0gH``Sv^S7EnnH2!+(up z{%mW}bz&&j{@)}%8MClI22FoLBa&C?C@}eA$wQbe$dl0;Q17@SufT0BtFboIi0F-# z0r4_?5t}d)^v9mqpZGf6!y`AiQ$K@o>!(Zhb2pyyF@wVk*@7WAMgAhe7{=wp7lJS6 zKKU`u2FZGXr)p#M{`|#G8&P(8LjMMNWlYSk19l69pP=9HsSA>>($aS~ zQY$J_Bs5a8Ew0*+;O@Jn#<^KDOFT&_@?}V)K%KiT{s&H$Rl!v*#MA3A!?$C9nV*b&3ZEWX=gXFB} zPF179*7~@nY5oR$&XD?7LP<4|?UTDnF>EkwH#7>6pbhAD?1@1Pa!tz+?*d3B_#9JJ zz1|L94bEa2RohAvEG@aw`iXJVOAF-s>+5jme)vj9hn=5Bn~Vw7%Da($@h7!@pZ}11 zDV><1w-LDkX%)Zg*%na8AvNtR4yuojw3t zGE4&uBc-wC1H*#Rv>LLQvfF_f8I$kw4AS-RvK&z7yI3Q6(gOm|)%guG=DSyo@SEa* zu?P>8gs4YKS}VuoZbKl&^hPuad;7DD>gvBic6h={hGu@2eK#hKq0+n7p1Lgv#(~Wx ztcO2LXm{TsjN8uNReOtgA*N)!W>l~VGodE<-mQ3S6|vhK|5+R_=4?xN_6QSj)xEA9 zp;4m!_Lzm&n(290;Ygq}8B$ziRGs&`nnlDKMwKOpm+LDjI~kv4Js)O2Q8o+I_>_c zgpUnN$<%(0>@1ZHv)Hh*x*@md zYE?bFys%#jubptw85LDWVAXc(Yq{03rjqjyj%fj3BAbZH9MG(J*(FINmLG4IjcrdX z#Z;H9U8cAotr>0z8CJCk?zMWRuWh<&%=EANxP`S|@W5nMf*8}IU*5}AAAuo!ff>}8 zmfU2anLN6rWQ$?3bSK#7H2U5Cy&`Rys)2uVtgHUV>DrOfzQ+cOyk3h(slMI;bf$nt z3)b+zS|h@^6CokA*T@u3Ei($?&sWomDvfj~RF;*ew#qi;o~GPK1Gldj+|IXeXpGTb z0rJd7KNMphKO^9~)yVMdAs$%n@<~gbKwJ%7xqWfH*zlMLE7Pmi>9i1g;lG{ce>ToL zl0pU^IxB_N>Qk*m_{n%Uw$hR^FIFSB27GT&cW=My>>Vg}w%v>lX=656KzA$av2XXV z9ifVRvS%!nVmB}e0xXvpSDA1vdcIH&x2J#i>bo1=d;#b zbeQ2;CnD1xZ*V&;((Jh+NS{C!TEAR$&lSnP@>i_-$=~WFz>C0XJB%b53^73wpcm7y zvRBx%J8BTO4CueY{znrDpe=I-J<`Bg9PBG7CiR zoRl_G7+AM8X3kU3%lk-ugU+zHGh5eH-!{UsbObgvOuWI|-j67>-`iwJ-(@daiQ-%@ z8S4;EDeX{;3BFt_LlcG{F>RXYSY<`Bqn-q;GhOr4UhFUwd%KfCk|s?jSS0yX`&km* zpjwX?AvLD^uDe#XpPYA*jFr>lhC^!1MJ2uDEJmads;*D27~q;s{ufW$H5-cW7R!kK zk1Xxl$t3ewJzKCq7M@IRag{sq0B1A45Jxy!AKl;?&YXzb!n|bpWT7X8_6m7j?*6;# zVZkmZQ4t#>4ZUW^8i;}Aso)kD!ceQiwfWO!mHkif-*g_T6Qv>Mr^c6+=>pO|bg0$6 zfZ}H}#Tnkt;IocIsSjGE6K0{$^ca|~%!dk2=q$`vx7YIjtyZ}1x*00ES8n~zVS8-3 zw~mar6k#a_fuM=v{}+#2u??MMkL_+Uu_A=aZk7Y`11dTOuev3PJvn82mQ=+l;x{cS6)RC6M_4L0b|e_tjP~5AWYz!kN2sVA z7`Dpiz)p28sdiY6-nQKv={86gECNX01~%;3j(5QU!2xFd@y1qOI>H%?~ z!Z%m9MwIJ7L40zKS*F$+CJm*Rfq<7@8D5f90daCs|&#&}`9g_QIQy-6Z~#$EW= zlLWqCk{uf*nTk;zTc3pJ4B27_5aQyp# z>cO4=@&H|uOqY7lwbs_|wYH-nf?j*Nmq0KnQbG;72=|O%hfsxO`gL}kMilKk#1Le@ zY*l^n9mTp`{jsVW7ycWa`4C$M<)7;7h`jo+f=*(>>PlkKga+tub34FhU2Z0s8xmDA*U(|$?aPtV!J z=dA7|eqJ?DNau4mxVQ4eHI-F-?5O*~Qcdfvu>2JXWbw!B(fmtA8fB({2j>JUXjVR; zzX}oeFS_biXnb(TdkDe+#>`&Sj>>7iXO+@DN2z~4eolvOMK0R4marE9_N6ye0!4rt za~_utCzmS*y-5-^`%3ogm{>1xEg_89;}f}4TNlfEvShXsDQjgR20E@A?}L~gyqe;v zH_OOCW9gCspwzeVrFbcNDnR-M^|= z!e7!^cwf&}=^o!DXGiZON2m#(l?-2)xpsbdTDCSmQ!G6ReZ^2rY*|SO>V< zEfHqbjg*DIWRTmMzB1og3TW8O&j-0U8LbXpEAFrAtl&fdwo)CI$Eu(cN(#BrpE{$1 zHV&-xQkl>A25mpNkz+g?Z@oy)qYwSXbjEtCjg{X)t|&a$LCaYsBJ4yPU2)Un(%!)b zA85NLRj%~Z!E^JqU9J!mo796(UxU)c2*nl7XlakwM%hiGc_#7kWJTa5$KAb;}2VuBbJ##2c#9s2DpG`y@?onGzey$OaS zF~Z!KD0cn|CBq`S_v)g$ujjKf;mp`JAxMcOk~>9}^=%u4EoMf(maj~2BsxN`^9y1! z{GreZlh6uao1B7by_}vJ72Ym}gjxpHn({-Lyhk){^W?7+esDzO>TV`o#93V(?-J>@ z?S5`d-6NfWu3=&)e>8pBeC1MS&R;Vjpy%tQNgAC z;NDVtz_e$TYJplr-_LFvW|jz_F#=^NZ}k^@e+!|)Q0PrjjcFGTJ}atOof0?i2iL-J zOxKXus-_JP?@+YzJ}8ivU}~n_5VtoGz>l}i++A?Dr{poflr;($nnxwACjPuM^wm#s z4hMEhwh>doC6xzX`l}j4{1M#H-!8?UcJVz>DR^E+m6JRXEamS?#<+; z{#q5vHr9HOpDLQ6hI0zU8Pz~Z!0@;j(`#sPO=A5+<(*~RP{&D=%91DEuH_VJWhq}C zCl1&SO$HbY^Z10f4O9(Z{snr0jGdnxZpF#;6k{PbYytgL^u0WB-JX^zQU?IS3;}$> z7@iFoCgZT=VJB+1uZjtT?g>*_d17m0GK+N*oXX2$@f@khjxO32>AIG@kquP^Nl)r3 z>1O4)0ApJT-oZ4juw2a=LgDIM9{%A1E#H^$_oH-;oL9uj#yq-8HJ4UZDK%%$pBU5(Feq9q%_yKx;~*|jsI<6Rj$h~d*MfbUwC z2iS+HrWZD5l>o6@eKl{JQVYw-G>~HMp?bAyWfJ;Ai=_ z$o!p^bKdha;cmS@(ZWAiubO`Mg-To~i`?2;CJfiI0Z6CC6>6a)t;eWM_ie-el-6G` z$Vq;l@vYppdOovArDCUs>FvP4N1b_vM!%|&b$}FUrAyrP8oI)tn3kyJ7lroHs~{CV zqSr6({*K7c`e8T)^01RXw6+YZJ4Mz>^vJc@V<)~kz8)Fck+8w7!v)luzEg-4fK>MM z>+bEJ1=88|n0(6gTp{;Yy~`=8TQ=xt8Wm)1mXvqmDZ88>y1`E!2r+P!pc3u`Hxkrq z9q=}EaC$}jd!MT7)6UMHY}{$kTpf1E%A@40^2gt*=v~w5n~n4TPV)>n2&iZiL6IVh zfv*HmCBJC}?yG5$h&rf8SZA-?23bM90i&79{r^=CkN>o^0GLeuA*V`%+lk zpp*R-Y9@a-`9}w@Z)|?t3VJgMiXL!Cp14_&$d!N3L80PhJZzs_ZH91ynHT1}N%5FL zsl?^XDVCw>I^}VPY+=+9Wc`>ln@+idr+I#=yQj;p^}B0cD-#@dA~J1vt2bgUDgocw z)g~ohPLG}g>uRWe;qXqrW*+2TE#zOf=K;pwWsu3YX&Qg}RMsshaQM4Jm~OI}iaozhJmeOk zlJ))01hUUxX%ocD!Yb3sv$!uj@Yc@i(x_W<-JRd>_*!p;8je-0?BC}ag#jS#n2)Wz zgdH^+7Oo$2^96AM_Eojg`=z4YaJ_BqdXWLUzMEd{Hwz2>D%R?xx10In#j(O7Xwe&@mJ*6<+>eQ_Jb5+U7}3p`Qo?1&t9De89CcZS-`UE}#4 zCdj8>ck5#-oCW*<#bA<5%{#pjo^!_yFz{w6UlXNgHf-%&1oZ4B+5#wQU2UM)eN`zp zK&R7MO0^ZGL>{P^j*EM6@jr@7Q4x#pK)t&ZT)4w=i%LH%7<9IbRwA|J5Xf`thI!nk zyw1!WX~~5jsxH3Af+k>iF_fMFsSF?l_1Eq$obIo?AF8YAlad>W2tggwcvRAMaR?n| z*$3iSm@Rd?726)jB0BPxqm*KX1ot&*JzPQs>JJS>dCsw6wOxl&_j0=i0;}b2vm!vf zxTYc{Db#ZYGgl_-)?1@iP#z=HQ~(RCl_`V+UHKl17-Eg6%%_A##LdqgxGtkOH~8r= zvda(FL~FahEYWRG;H!wmf5U&9l}#bH|N893&$(k$h2wy1Qez!xiDD%)T7nTES^Dh^ z5+#O$8fRae$s_a;F{ND{H`gGn_JKH13?l|Ojj~G=E@J4c11H9lnET3%5928+3;l!a zuZ27lQ*I#2S_JCcH0BsXkaWQW*3#KOa^@ZwVAnaa4Av5q=X6-Kk-EwHu7!-1ySUDT z;x6?w(o&+a+LoPk9dc(;Gxnf(&;a=PrWzwzCTAfh7}~6Mcvh@ z_u*)kpAUqZ!XM5m|32~+4GILr(DZF@5ywQ}#Ju(SZ9u6{DF@eMo`r^&c|&E_n;`@n z5DfE)s(S;eVa^Q~D!Uqp#st;t8R#X4vy}FoXai%}8;UR)+1e+vutgJxc+(I-$LJ`FvqhAdcVr6hG_Yrav~zQ7?% zpTX~OKZZIIJX4^bWU2_RKf`BVKeMOTn-+TJX96pxa&YAlplt_*xYKMNlt*HGolO9R zLO{m_R01rJY)wm2z9*>q0z#J$8y#tA)$_>97NETQ_qP-!cvtRhtX9%H^56p}_J>X5 z5VqX9REl(U^_8>?9y~~>b`)y^&_*R7v&RME^{{U{+r(J@TZeyk@K9U{}g2CRpC3!1awcnG7-`{?nzhgPy7u5;+{lfUc&EJPL4$&U?l+-UxSO>Io`YW1d3ESAaFV z#AW_0D69f1GCoviYSa@*yZ%Aq;X88)*mB-G4S|Uhh z=h(#lOh);-B09$IE}8zIj}z^?#gr^14FXQ(^l5{oCN2;Dm1~ zlv8h{6?m&Al3Pl32eMG->X1T>*>X`2h7D zEv3PUkQq&egmG?N#V>blA_|}b8bq>j6G3R>Y83;rK&1?StUSZzjt0_AnT2jq)Zh5% zVpTn1R3~kjW^oj|;WX-;2}OG1*-nv93XQy2{cmhH_OA`y1$P?9Z13C5~v1|O{#>JMn)o1vbcHL|Sf?jqwL0OjL+oL*@G}YMf zoq2k@uJ&r&R~;`?&%JT-;)Cnr_J8O^h`U-^C5$5s<1KE%LoEACtkL;PcozND5X@!b zY?{;;OYcs95Sr~&DuDVIM6W4evhbz`uao;!p*Mg0YJBc1`YAq8r_}o1rNlTG9-;%|xM%$r+CN>#Q zj4warTfDVq>4#j6?;p)#bDA~<%I5$NDaspRq$CPI5KrC`} zj{6HQp`zGsLR&ZwPdTAV;ZQA~)zLWPD@o?iv2)7pm;0ghTkad8>GfIxTEPutyS$v5 z;xzAc&9!=-5=5?94})qICta!{E? z!W{NQ;#NaJQ>Mxw})^`CdtyW+l$A#*EC7eudk%dSqMJ zVDq~vdq70Yec?2ZAXp5)E5-Z-M^NXARBYu6Kh~w8l``FXP99oMyhSlA%{(`g0)fSf zVoL5b_0~byXvN*TnHcy~((`6Yx(We948*4uHQ7g8;+oZcJNP#`L&D8BhTxSEc8vB* zPOyn8dy&UOv1#mwKue!eLuqd+K`W&K`RS^=M4c35It4p%e)02<1xNyT@Jf?@? zO9#K?zD@o9{?nT@Q2T|%JKc)qe`)--$am+fmzua9+I4ORn{G4i1NC1J%ecFO#@hEj z)=13Y?Dl>jB}M?Z!A?)EuwSBzW6OJ|WL7)sLrx+8e)vluj8^W7WLCMN#$bFvtEpO} z_ef8~u#ACU6PN!(+Obt)PMVa7e1g;PP2{`ZYrF64!v>3nSoxb|`(#}9Qx1%I!TDSZ z1{}l9WN?F1)PX6K9YaC-c(>A)jlwEnb{DM3ou7{?(vOW-Eqxc;C>nh$fkXNV_OOY& z4AP!GISos<%{7ZXXJ&~lOo*8riA-JN99OJ}f7QEYofYBy<5geW7rul&L9%aiu+E0W z-x>W|ad+MU!_1ZUE9?)CQOZ_XChj3=*zz0jO~r@4+vd&)>w7)q=rCZF`11-=Fcvl- z(NWCEB1*CN4Uen7iM_;9cBbwd{|sK!3Ue+Hk~8Xo z`bM-KtU{lV7@Kvgdl#78OPw4M@9?*;*mUY1&nkFSm#I+hdFo~!y?x?hsuX?h>YG(fY9T9n; z3Um(2Lc6`sf&)%`F1Jrh%8OX-NCBNwfoE>0+;$AJV6UD2@<+L0z9?6<;#c-WtL$1{ zlGiic4z_gGMKSz|ciB+;%;IBBeHm}YKP4Rn)9}QOD{@HYDdkt5KG<%2ec(_%=i^Nm z$H+4&+|E_5I61xpwx?(?rNN@trs<2^QynWKQi~ewkzS7kIn9%{kDGT^JQ^-6F0!-4 zBz(dLiSXFTGl_-$uw&^m9>gyGSkD$n{gq8P^($$+^(9tZ`o8!*LkaexFZ_;1yNS|r;)qes zTd&S$|4NE+FoxFB2p|?Y)q60CPm4MxuA2(3dp|}v>6*;74l6!UQY+%W5FYdvnx%+d zpS?qP=4Pnp%eY_=hBK8pX+FhY+pCeV>^W<^(Z1!7{Bw6?JbFA?WVOZ3A6v*rKVM)= z5C{}ZF@_fg3eNPN?~Rou?T&P~6CwxmNJ;~3)g$mJ;eiE#X|4CcaaF{eJoXNt-si#& z)g0WUg4|b!y?*=0;uI4mhhbSRcZraJE6v4bLM!6Ks9!a*HM0NNOli97Z)t3D$I{5# zw(YFqs*{iE+zqF*)xq{&(u|24)$pJ;wjspp*!$P5hYXs&P99IaN><1!x@zx-YJI$i zA(XEvICoaCDDb{wzg>R%X4oa~#gHd(#`R#arn~`E9FF1reVfgVyqHF_)HC=z(p!7h z#)ckp^>@2c)hh#Zg2FSOd@-^glSb3@y>K%htQ&F1TbA`*cOtlKATX_9k21{(je}LX z(QO+9^uD92Ut)NQS_gM?v`~sb@w^`Ap6J)j{Qwzm9#7G$o|^`{%G49daU3wp8rdqm zq{*w&W0cuwRJ=h$_s4X$@RHGG-N)kDHfw)ORna$|iCs8ap{|vtpyI5~z?!P(JYKdZ z(3P!pmnM$%__<8ww!+|~C#@NJxu?mWOiyzzc|EG(q{&jF@b%rzs*>fOT6nsY(&K`6 z6(>=jW4RfUCBnCU(liP9k{Scrq1rdG>5672Ff=<=C%jeOHm{CRYd&>M5-VEiW?C3X-{xQ3* z;D4eUM@TPKj-;(9_Aaf!3H{_)XEa5ywV_d$W^a zG&{}@e645Xt_N37IcGSs(_SCCgKyvInaAuG1+V*zK&QxMGs)+$`lFUe)zt;Ub0PY( z^ncXrwGYHe+PpF zj&7gu4jo{PFKqKae#IFqMm790d}Ldt!V8HFTEm7%LSX8qSYBsN_Z^idw&gWOYuwD0Stjcv6#1b-^ZQr`qDhnP;rL)B3T8PFlIhu(^eJarl-XPRn=RJil@!7o)qLs@}BUI>%=vI&;i9d}#U3 zon0r!+Nu<=|7Q9b2Rceh@0abG$Sg!icLYyboLorzbdvFX`RdI3-NI{8cXCtokR~O@ z-c!4SHKsP$tXZrR!l-HTd~*>L>HM9_d7iUIDwY0kLA6b%7T5dc)YoUr&rz)GM0%I! zRUo@`*EATR{m{Zk6i3Q-4#K}!le}rz;QG%a854J182g-HsH!m-bfvC;rVjKq5 z1je+MEe(@ktqkX?$q)sV#><^0x%51#ZR2yA}AKkm%EKlAPFjqbY&BeH+IfcLRjwL~%>9)1|4)l&XXA$L4N z@D|NEePJ7--RG=niJHxSG$)5*=g-_uu#V$3u{1x2KNAmR^8GDHbp0pR19x8GU5f?9 zR|b=#z;NBOuS!dhG@-~Z0p7YHBT%AxiKi*dAxnXT_#UEaPal=`bp3N5cMgb8Z)krx zh;6`EgO4w@rtJsyP%=j=ylY-r#T$8S+2>z5=?3p8`JP1`3eXA8gTRe);UkfD2(`-a z!!K|?omtjVefEGp>hu=x*}V6Z+g&(}n+~q+I1uS?hgnNFh(zR>Bv*3dV5u%z>Y&!BllWo_b1a{F za}3{v-wq*U)ySZ8-jOznt}D0Pv&hU2^GWO9&z++C7z|;gIO6I5#`ILctMAU-wt-e# z44?Mx$#VOMH23Jz6t_@Pa=|E19QRbJl|~Oi(y3$DbNX4_$;*1$gYLKq#Bye!Ae(!; zh`CRbboBLD9gRB^AB1>`xBMJ2jjzv?6}@zHckRD1gdjE4Px7g!TB!9P@G{b3mIH$E zGVXjamZQ4a)lO;Dg1orvGJD!> z+>ELKhY`!7n6>4k3|tcL#@6n6$@=)>l8f9{$Bf#o#A61NWnuihZ{#tG>foVHKJs6_ z5Wne4tK=zdO%x|h@MVI#T@J`_0P*&gw@KfBVv_fvz|uIpaIZtG!7T zY@j!-ljBKv*($!X@>S;Za;tt=IS;a`rymAsJ;Ka13;t;q8r6;b{#2<&Uu|W(GgoEMS;kTeXChP|?~+fG854!{&&)a7q>sd7rgs9| z*K4|-4gR(?SHckKdgoajTQtw4=rbCPd1R#T7q0+%^VHJ&G9Wf!<-q@5+||23>Y-Zc z?t`BTyQ-~m$12p__ar7rCpKSTRG2HUqjHZkbjw>EDQ@XBz)Dk{F{}EiZ07z-++?B; z$u_k4R!p1mM#}B+P=)N{9){vRPk&7yQ1*9sI#z}G<5-HE{&xC#^|&uI&@HytY;jL@ z3|2XxvST{K5;W7EC=TWbO5P~vRE#Zinr*S#|3ch*tddR(X6P{-?DNi~og~%Q8HsOyysjrlO2*R_K zei7u%pDq7XeApXo(03Xs2tQR-eh;l}g9o zH_*=UU)Cj(_`bj&>vAMMbLXh7wml1zvSM+KF4fayi6YW57nb7|qDlHK5-*N&Og^(7 z5I&M}alCcCyeLX0ozPwO#;Ec`7`MJhM>M(46mhn{>D<_dv!rG>$P0tV$g*%aZXSQ>wZ;_Oi;4l z?AjfUXM{Ry)w|+RU9T{{clpVYop7*`FN=qZc_T1Qc^=BrTHLw0a_}c6tUCV{1*Q?66UOdY2u#f+pBI`3`PpBCM zKcuQOS)$Jm>gwc5huc!@8G{CR3a5QSm8*La!z-wsgB(aX%x6?Ikdn$gxu z%-=SN3jDgICP?0g;&fR0tmM~M`J}B57sxEV!d~ys?bdglAu0KxAQR1DOdOX8WE{P) zu&)|wA5%n}|8|P6)EycQGiTZDZc6mI6|=hXXg7tc2Dv@Sl5=ItVE2s3y@uRLaApj~ zs$Ad!r#9WLopJpII$as7zuUJub~@Zp;|<|sP9o}K3??g&#(rF2uQTgn_vrqz3#T^W zN9aif9IdxHoSq!7Hiyy{7?fJZeaNEtvxo^;CHU2a>h$&3)g2Dt6g%eKkqD)W7Eo%guhNf&UJ@7X&8zOy zeG?hh7LcbY!@Cl<9{0D|G`rnx*uc|P1s3XLR3q1pr08s>wd9C+-b{?!uZ`*TBBHc; zsUsNb+7VF74=)ehZd&wRT%!LO-DdVK2sMi~&oSCNnZ1+cCw$wqu11|JI)T-13t4@% z(P6wss4cD0+P9!<7FA3#X$U%h8TGZ(tR?d1l5+Ts>6tA3LKBCD&P0druh0B_(WBr+ zGm${zqd)&^^pJaxcXvm*aN*3CIUjyvk@iU!b7Oz}EAKQxPcX6Q#T{Ut=%=@ z**Nb&S+QKwk#Q&WW~5r|ok~%w?k9@1Pl{js{o&I{>&Po4pmh7me?kikrBG>wsHFV) z-jDqn^26=<#U_EU!^6{6>13z-A!t}*Nr0MBOu<5e>J-ly`s;w`8!@ARS!1GKI)mi+ zT}y8biuidJLSBXTpSU9Tyntdy-dMq|C=Bi>#$`M_)4A&7j`!6owof}@1)i7GS50UP z;-Z)5YxamY<8{Sf$)=M!^I_|DmImeGS*#S56QVzSn0E9_J=9EuK}sMYJTg| zqy3e4n$*42*^;SKHgQeAI-azG0jQk3VbSAv$cn(|#$y27ns)Ro@iFyM_9F4vlJd(g zjK#~-smbM$6yx4-Q-5|_n&I#uSHjod!O5llLO{rws`=yn3~}$;nw^|IS1qzH@I{in zEN+!bzum47_dLq1%<3qR$}(7Hv$}C`@*kI~qVSL@G1_eTV*B5V1d}y1=-) zO2neQ;tB5}?eF}j-&~f4@_`;9>B+txt$1>4KiT%}4?b-JISx!1`QlGdT` zP5vrijhcJh5c5sjdg3B&g{>&ga?K0o#>UJ#+VNx~$%=KwVHRaaH$&Fez(bc-WG-x@ zJBQsq+J<+0m#YtXesGy0ZkQ#OpbaguS_k-pUzYjFOK;~f2HL8I`JBo1&}C?~*5O>{ z=}3CyMZ1pTT}(xhY-9YQz&SsB{pG`tuyW6o@2+ODBfe91%2gvxlctqUEp^MjerD50 zmTNLACr0ssgLH3l4!@=> z^l@QoL_*!=a=$+bP1{`Et9bVOAnOVDIgK`Z#V+{Q@yB-z+SIH$vNCTC+0xo1TQ9Iz zAFEVlrw^%omMnO}sgvvY=r=9XebQ2I=6m(Ci#-_q zdduekf+8hlhVC6mT%Va??@f^=DdVt{zoCSU*SO8{is2kn)irka;cynql6b81Czs(+ z^SM9W+(|t9EzV0qlU=XUW)aFvG_rj>DW<#dUtxL(L!eb~qig-$AM7Ef_aisEy#r3Y zVo&7jKbHC(f-if#qg)i zSx}utu10X}?a%xO8&|~G`Xk<@v=Y|%DUBUj*Ef!LC#!QX{$rPK-W#ah1#38I*b0TJ zmqmDBhrg93{#gD15?q_o9@7`S>>^rR6wGy%okZ34NpeeY1?8R(*P3Xr(OSs{L(isk zu2u8hrRaCm=z-TCS;opM^T|V&y=*<+3AA>3V?Rw7*ZW1!y9aNY_3UGE>cv3Ixb&;< zyXMo1!Qf_&pz$i)MF~Tqqc{KLaP?c;(W0akCWB*1jLa38lgzs9GhWASA8iF1y8Jm} zFRO%P5WI}8`xm3 zFAY|w_NgZ?k$6b(jv~(zEUB;?Enz4biD_+sFjS^|Fml0h6#!7ZI^j9#-WtW~X;|*y zewk6s3HlsBOb5`eJ-Y7beKybfsnjrEJjC_ayJ$Ko)uKc``8C)%nV{;=nY4NS?|s0K zNJ+EOKk)I-fu?y5lS`*4&$*E9)!gY+jT82#6nj|FzA}gH3k=6-uhm-l2?tq>K@FiFuZdTsyIx9|K*T4l-RowO7d;`j~vjJ@V# z6FD89y^(r&v>>)(_;YWoUt3_pce6_}nYlaHPFZZyeb{()sv(HGkVh-UTJoQwNzkj4 zo9WNJclmL#ve-|^xu-ym1ih$&VWJs{^}p4dAZ z^y}@mAqzllcHceqf7;?5Vtl~sBO3=lC1W3=ZEdrfv#W5yUvJ;@d9X5)Y#_!FQadc8 zDnfCApz{8!b7oGyD1+ahUsRr^7&?ie@6IkpEzqypn;NaV7l}8yRs?lycUtxt(7+obGollaaSv;0Uo?G;!E55x@U_V|~_B~D3M?)rF(0Hsp+BD#J zy8xgRt+{~c=N#j)8@OT!cdFzn57D=q>eh=)k5R>;i?#8NHu}b<)!Af_E_#Z;XTtQC zsVsOt6l_5TWViB{xjq!ukDdPD&-zB5CC2p`Sy^-9J$BD?Nr-1;GtGLZ-F|%bwF|!Y z_0P*PapjW=qer+nzt!GFK{O$oe+PTcP(y~C>znLvy~61E8`@D%(d&Da!hCp+ps}jQ z#QAb^ouR<2#?gRJ#edoIy6%_9U8z#ejR+hx5W8GvKc2@g>X%HU$aAx^OOf{M_S>J2 zUtKD!_rL^zfzRHIw@gw_&svuJ4z`?wD6W((eX|`F-0jDlh%E$5p$@i(O+6>7J+JJ0 zM=*-xqi;P}7M8sF(7|VY=7)rY19!#D2R@C5VMiDJEHFgG7!bejD`f1?{989DNOhx) z{MY?(SQgI)K7e46OF9~*axGJ#dWc&7X@!kGaxaD9^W5>H+8)d@zvY!D*cev0A_LK- z`Um%}s##f$#XHuJgAuL=hh;?@BcnIvE9@leT?-mYpt@db8x{k@Y9Y(Fjq{>y-E!+G zh(Phgm$&v25TqvyR$xz5&piDjD7CA#cb+hrBiyDw7?m zaPz?y9dVWKS-jiK7Z$pbHDv-0SIQrY9RB`NC=trL&z!VYci0W-oD4Xv<2K}Trmm30eb@?0B*eCz*!Pzgzxb$!7Q@<~-pMp7moQin7e10c3JG@M? zK!``~hKnDnC+#7(`_BxHwB2!*?HBvT_~;8vIu(Q8z*fu8M3y-#)$fhEO)2&?&&tJz<7KdpOLjH<@9+k&b#_lU*N@|!Dg2YNu+EnFpJ1o?MYRul+WLS z2d!nSb%Z=q4i}}~soa;YpIUuV8nQGuZZ)eOR*@5jQ9TG2=3Q&&GK;tEOzxU!r_~BM z`W_)k_?4}VSp{r+apw8&fn06{&sDq)SN!-l+eJn(>t}_xRCy`7$grO9L4Uy8SHBe2 zmH2-j{Gu-z3U{12qJHBs-S{?+(qZgtZmLi5iyw^z+6d-b;mnrCWdR#F9!(99qH|RslnFHm-7W?p0*00(Y-0 z1?t*TaQyBv(=yMjxsm(QP()>CIZ0|A2dIt$2by*lo(2?K4pwjB7N>%bvEoh#C zSag5Z8bFxFUk(QG91q5lnp4XybGEhLJ-+vpfJu{= zx!xX?g96&Hrd<&y$Iw~F-8DIL-p&5T;K+09wFQP&W@snoGqoP#a8#u_hjj_d1g(jY z-7PQXLwnG%Upn2}YMeZedk(vM?n>kEQ7fAx<4H z1K}O(4zdq^jZz10YfS>`3D#*xKWrqu#);vJUg2BH61$&_#rkdcLr?Vg7AkdpsUF@! z(HuyWM*G_nWU82c%E;|ak@L-l4`4Y(GrSt!oa2_K!EQNH2VWB+KBc>Ib{-3lSr$kA zAF_PEwhYjkd)%35L0f7IDEle42E)kst%Xc zcM2kD8bj#ylvFDX#-3647Nsz%l`XTkC>-T4Gj|br{c)Ze^~09?-i(hT!*|NEns*2F zKA3Nz2wM}Q8oah9Ij|l7$@R$nccaDLG)$W$r8~*b>OmyZKEpr=UlwuangpCxhGn4| z<(QI3&Tw23m8Orera`fu{yL)%B3%aej@ zSIWjS;__1@htI6{eEVks!n>B$^ML9(VlHpZo;DP;azgF?LbAsSqlBHc87ESsF?CmQ zYCGh5cZ_W}8Ko;IeX(YLOs2fwEnO#tIp@wpx^e?aY8saNZTF!QkeUAB9XPon$Z&rQ z;m>%Vc|;#NHe5=+zJxP1_mJT=i@Y+v%W|nwmS_{%zzka3?i2O^Cx2ogd!0$T+GR92*!D}@3CS8kPKG+aCe z?QgZo8#CCefLm#i6PAA2#4A~2vmWj`!;GoO4mLG8Y|*Vs2&f74i>ImDdSQu|q1Lzt zKJBnAX=RiuGCjQcXDH`VMf@H$b5D2EqS1WciT{ABN7_#6nhl?h7G{}$vHVZ4cAbjp zJ0n=@7g;qRF|w(^rF@RPg0e(wz2QCV2x+%pQ!fWTJ?Bd0ozTd&?5hC4M8gAn{`41DtxZ2SZIMZ(C8Bp3bJBx5i<{AwKOXSb=V*aj^<(bQjrboQY&-n? zc%NbZ8(hrwq|=oBx!fd>IMr|J=RFag^+FI+g7iJ4k<9Y`ctnqp{(*1Ir}~56Uj&Xp z1DYWtrMYt7z+8X#X1UXn;G^m-lA9UANq%c-8ydN|y&R&Cpn$0N%%?=^gGeWE+&m)q z=!DOxz1-i|*0$t)1;4P#9BSmxo{q$FtNV3yb1Z5`mdHUQ>1uQV^dD1arJ%)AByIGE z{}|tyrt>dmx?RBAq&aETa=k(}^kuz<{B(RHaj95q3L2f1RyrG=T)Kq)qT84F(6W}- zGzIsV@`}w9eD{2)QmbgNj7yD5>I;W&js>c(UHK!sDlku8$#fV|(zJ!0dXL_`=R0z| zu^3gY8z|DlzoqBDlwYo)73u7?Tn*ks zIymh|-^q3Tg`@+YklULP=gmEz;p{*1iz}j=`DY*8;}8%63^;z+X-#6${m;~W+D||i za$OlO_bhhVn1j>SZ z58H*V@wW<-w`aZ`OO%bcTVPaDG`=h-qYbrlG;BHO&K6ze$yfEH-&xsh6!u`Ldq_A; zSffz9`N+6B#W7N=qUdf8XW6iC40ONNo3*y+xFp$atN8U)1!rknr2CTO-if4myBtN( zrMks5p9~Y$X$`6c5~x%Y)e-UPOb{Lt-Rn~q688!*=hRfMbb$N>c3b!p6KUo)ZL9>_ zD`YokKHP|-iI?a%AFDssgMR_I?@%FQujW%G_%eL|$*t{Lr)fpc4uNuWwHu^HA9t1- zU0Qi2tEciYkni=CZETx`B(HQPiFXh#s&3(5@19?~`B6+7=e`ijbm8}Kp`(r05YCpG zRnCrUW*_coOHl6&1@`xVVg~dZ^6wMAPYRC18oib)uPE+-zi^%}9M}YzKwrpCUMRs7 zNS7yHH?Y$F3pKborTd{4!K({vM=rxl>b_R9FUl~|OxIm|0$lEVHuI?st!3#@xI%n% z#^UoRhNqDn0H{%ZGxoQ#c{a{5ljvS*ydxX>8@2iHqtm*V@i;uWxRH0I8?vpj5CJc| z;!3l-*lqB(m+ME$Rdy!TRurms`W=%7yUvSbH@Dr@2~pV1zQj494)AFfYhG*fMT>aj zg_*!PQ3U!eOpxs>kw)W187veM$Ea*qs->lHq?Ihu6{ zM6=Kta8QN%ZPER5Zoi_tG#zxZ)UNX-$~xesXuC~5Ct&dW3I=QIv#qTIp7&UOJ%})} zJX0{Ab>vV3Ye53+}_-!(q$eG)TcGTudFNzgM9vp(US8!_=?kj zvE=>&O|T;6&F$sBEcKQ()tLw;$)|j@DuOVlcNTff-zD6Pt^S4R$Kh;M(PLl|9})F`xUN5l($K+D~9 z!2~wOnLA@Qezw^;VnUXu>N-Pa9_B7Q8yTYQGO2V_Fn|AM_8g)*&s1<(o~81NtJKw< zz-4WDs}sqJC~>Z@wfmQXgDZIcyuIoljaD^pp3{_^Ft^AqtUvLNsa>nxOU4O!*DAiU%nu0YMj0P#zQuc=hcvaBc?)p3!*591j}6Ra#Rq2xZ-5tI?~88Z z9!#cG?Quy%PqRqaJ&7YUhghP%dGRL6Mibi1)vR1s$fjG)EWC#LoF4B^bV*JwZ$N@Y zYU>Qx1O&S5M@xR#j}$eiIfyzyde^tS314kuVxKRzfZtf_)_T)(q>DCWEzCI;Kn<93 z1~lNW7=ZgPQwqCq{~#W1pDOZJ^MGBM*i*CF2K6Dp*1YPsxU!`8!}<%Hnfe-k81YWp z=G9M@M~%&2sFr3rplx&Y*qgsolC<*6tJ7mTvgbhdH=Vn@jb7u~9l-cUZ(RMsUG2B@ zlN<_3!{_T1MCW%%P+B(&o^|Y5OIiEvXQi34Ti;mEA{(3e13?J!#Bi=YX%Z?N_VwYO zt@&l!wM1}^JcPMKRs9e_4W$Jux9AJbuSIm#7^)fd8H8NA>W@umjl5}#Ggi}zdB=4# z>=WlUbILR2-zb)>$xR@(=GzH#g}Y{II{^$1v%$IR>*tL#?0Ygw@8Ku0)9jH*D9M1GeJIcx=s{F`HYh5je z+@j2LWy~V04t`4?&{vbNY*r~rrs$W-Y^`K#tgzh@iw>h|j};JD0YOXfwErS;++QRP z%Ibwc@O{;GT5fM-DnCm|PZvdZ5~VV8fBF94p;%`%RkI*y*vIWpI2QGFL zztL$!{(+UeulL=3+brU`4NK13WRy}-XixO>f*gdtcX_lQ(jD@h zWNo{6fUy9FADBNr1?M3=3fWeYdz#;zJarx@^t#jF?ce%ck`hHA4r!jPA*fhZ2 z<}qpr`;uH|wD>i8?g@BuC1&S>td-xCzZ`5gw!2NES!wNU;%cUFcYA5L{MsI9h6`g~ zZOkOk8EM>h>`pLvEiyh{?pVIQDK}RoY&k@J@I2eg|$#MCRW~ z7l3qS$eDEU{9pc=^6#hs>B7~%iMzq3EoU6q=OpshwG_Z3HD>Kn2?uef8v z?R6VWKQO{=GoXI5i0`p#bR}DN5yP{;?0y#n_?4#{O3|Gv=00^&vwuO( zYHg%CGOZlumbvAEZy0~iM7trUYv6opbS~VGGezI0nP(dnN)fQ}X)WMlC4E}J&~@G0 zaLo%lKu+lzphyrIM2B;qNi`k?dS|<*4#7=ejvdwR6lQe_k4IC&`k}3i`jh{uDne$ZwH;K z6vUKDB(rm-N>o|j3V!eP?QH#5HOx{@-w{)xcS;3=udl*qb`@G&u zzjxZa{%YAQ$GB}>Sv33E(c*0!Kg@&86~0*qf(5Iq)ob0a(#-j7=0jgA)NA7 z^N$A^mCrIrq>YoF32s&XSj zJXkqDRC}&D^d)i%_@yG#L8|Iv^<{_?KoLV`2`jvByVfCRTDK#+oNr(3E`HM2*T=u! zz679MeCXsueh^LJ6r&=(x|MzmtP+x3%4-+usP=}Vg{HSSJEzhC&7dMB8j#L*93G;Q z7qI?=9tv5kyu20!&}g^&Dt^>&Z~8s3nZ~cM$XKIHK|vz0_Zk)b1%7ksMWp-=_#G5} zp)s0bI&{f|&O3Snu$Mv6w!;Ofu>;xR-`?{ekp-k~C+)%SQcX6blCVSUZ_R&F+*o#h z>wF22swKTP%+)o?4^@&RY*5msF!mUDcb{#NT%GD$IeMAZ+;qQ+DNtQ8H4Wc3Y#wYo zn5)C%BhUT$iqvr8Tn(tSHA)!}A4s>7vF`$5xd5a= zxO-|*eUz4J68+MzfMUjt}S$}}OQd*-~3 z{9ksK63O%PkRB)Fy=5KaAIK(5T#^AUG2bXYTQfQ2Q1Oo+Nf_K)pB=&^xCvR5L6!NB z6I?H}ayMugIL19wJ@&RJryK1g>|UjK{M{OH?;0v_qxfO`#(Q zm3+=jX%4gFJaci2Qn{J^!1dY-}w55dtVPQ+Px`Rt8F5C>V2T5qgC{xYi9UHp@~xjH(? zGx2Gg2h0QbODxOVcI*k{;Qv^1X)vStOxi+IIC}Xr?l(e34r;WuwnlGlkpEXh!MWLB zZstR|oE+^JfZu6gqQcL?6>vB{IabxywT$}0LaFI}jUzQj8{?oDGl^`1o+E~}t@dSw z#b$L=x4+SW$i#69gO7vS8~TL*JX(}LwWN<;;CfFm`?SKUkgV=SFqC5cd9z_Rz$PZ2Oc?e+RP!s?2T7eL-f2ChRuR4>*&Q_*JFeD`R$Z1LdS) zpZ}dVCB5pIjGtIIZMpNTa=AGUqJEEq9Db%>Cw{ofeQDu5HM6zf2bv&Nbx2X~8@KG% zyS(G)F49$K%k4LV`VCzbTLNmm4Q@tywW8^)hX;x1b9!$79yG3rzTtnoJ!}v93oUYo zi>S%437ncnb73+DX!zZogPWy4za(PZe-Gd`#!ik8Cw>op$^YshCI-03V7}q7+AGob z*x}6&z%ClWlhQnDJ<5<=`iW`aTa2d_fRswOe}4Eti^#$8Vh`6=WyUYIM~E(0t^SnIl_1=7ma~zL!3(%hw5jMvkxF}(6o`HB zt=sY}82p0g@<=bDDbUEZOI#3e8Y~~x;_|^cil&3D0OhNQlY=%S$K$%>EjKYXJ`!Wk zV0_MaS=IG`oMHzZNrCzye>8n%BH&~(V9XtUR@MA0AG^HCZ7agZ`{a@q?bk?h3GCZt zu5Y|ykxqmIQl%|Z>W z@o(wEQNOJeHJb~_jz5w^r4EmV>#FMBMO+WHMACAm2z>;<_yPFEgFmJ9C?7q`)vv7? zPxAm%M=tS?FT3t=Rg=1t8EAcve^D{NWuwoi%JTyt}@@`K$sV`l=K~`0|tam$wTlW-i1` zclHkIo8H{^0K&GX&+e-GTFqu;)^H}T3X1?R*}rj(kT_1O&2WKn+pKoN-3AA{ky5Lu z!_gl91s1P48a}oxhnVP8%BQy@u z9HIkb6NzmW%PJ*56ff;@l1&MOiMzZa3Hf1ggI03XRj&>&wqY7HkmbM5U%C3i?eRj6yhaOXOwGFl(=xln zSXirq*6 zNGZ5r490sZkSi%65O9Q4Ia9X1FrR;wKEOgi2@D}q4**{CMlh1&9%;-5cUgqCn86-u z-mcBz%?tMfimiIZyC63~5fp}UM8bY7LHF{xjl?+@kC_q>mT&;c<_wgz8c!(|jdV9#W`k1QS2}C@C_|K}2>e z%Rhb;9_xHMn+R!t(Jt@ML*Z{5P)QR#7Gr-IJ;4K{)fd`ghZdL;?h9xCC z>TIqPL5T7hK?rEP)X8#-$Bmb-qjeUOoVo(`J50xRKv<#V+C3Y;gNGd0L{`*}kyQcD zFltL;m`%Rex!zmy0+Rk=<6{_r3~iS$R5=YT`n}BPp+>5aY{m=xAWk{Uauff!(Eua} zX(lIca4nIfAR)A4_S~PFp7`k(S&ox+Ln?`!)wK}$;NzhTOcVmamD`+x#zI4rv;_w!kgn_ z&=x>eGD*qdFQb@a`Fj+)+GtYoHR93}Jce~1iXqK?pYMzB4*byq$y}JrCZLLcLFPo1 z)K#I&5fvPIRlv7WC$PVu`M$nG&+VtCZHy{F!q{8WuWg1ih-xJ0Tu5%dGymNHa0)wF zn3Q@bgiGUYOvtvH7l(wbf!66Ska``3^yp-36r%>))>Mh6a{@~Ut$V#0iNab*VmMB} z*gJZEf$OFkuS+ET70N9jUnQql-HO%Je%ZI!dB*jIqtFY;NMY?xQmAvVFMl+19f-9F z+>A7;U#(gyso`PZ$aN{0T`8Y*n2~}Q*|uA76xG)KOj!dUc11Z9hOHvK*7M+A(Ep9` zt&;H3?Vq~0b^5xF7mZhm`gFWtB@NlF-f-)JhY|;f1QK3gX-*xMmNFp)#1PkTAvNK; z?NU##R&0UMiV|%|4%V4mNjQq8LnbOb3%^ck5=qPe)2C(S@~)*!R6a`co~Uvk(!TdF zl0%eOuF;;fNd){dZey~|+9ZeKu3r-E5Aa}YE}AJWa3~RnkCseN$65}LHhT;aGKdZ7 zYrrU+E4cz$?7Zx%LUf{Ai1Qy&_3RF9cKt+XMcxoG55WZ#kdr~_5*9sH#taIVBnhvY z`uYf6@&lF@3b0 zIiVRTd)%(Li(5QIyV0>z-;S64==xx|-u3~Xju1eR`?q>Oyz1bkRJ0x`dfvQ!wHd9x zqo(hZaU=T2(JIco?^~N%uxPEz(r_WBcULM!!BYBQg^urWC+hoX-~gvUShF2?7;(dJ zshB>l%hEu58D9i>aT|G>F2}n4>}}?39#^gB%PD-W!M~IoDAW*hP`G2HJ5VLZUnTIh z&|XH1so;URfNy`uVGO@Q5tnD_0K~ADQN%~J&lP>J`YH2m93Xw@LvV!35ip% z--M0YYVWxI<*n!50!q0 zcl&Uy-Z9rO*pi>EdXzf!!yd?{6~tC5yR=?_o2kii_=DKhYP5}gyMkyTr(3P7H2@$j zOX$zsxUg`Q5RPn`vNjX#nbMyZp=f61vpNNb0APF*4wk;+{j{Hn3A8HnT#LxC&MWZ{ zOH;d`+u2{i+G9t8T~;T9Z4`^FhMP~cb&ipg#ka&Ir>v$?3a;dar^|I+_{ao;Zl#!f57j*h+D#?6KXOf$4U;34zo@2>`i<0v`J;VkRvNCXQ= zBs#Z(Rcoc>$Z%#BNr#l!XWS8o#2ic&=$VN%U4mWA7y8DqVTU^_wu?S9%~X6%U7MUy zrL@!71e#8444l?gF#Thv_{>&GhhRsUuNY*Mwqe}~`hvm!sRH?MlO}lv4s8Dt63#>>ti|W9w2k)NPSe89#N4Z5(%>5>j zy^U+Vm)f?O94Z>r-p8b6nTxkl3Nl3)5vtt3$SSk=$QC>(2K+>wh+veiE}0G`^U{9l z+nb@pZ3Xr4a3cSQ>zl!^`r7AB=lPbpwpE_`xPCY4wN2U*$IXF$J4PA^C}4FJcKh{^ zUZ8WEggk6)5QF$R16ak@#DIP{r&#r*q%J5f!O8IY!l7@>kok?8in&h3} zhdBgVI~ePgKntyb4;}l6dm@Q;2rzz&1frzIq>(qH4jCgA+{QeZr{7*$g!ZSjTdK)%&?DO^E#kiWAuN+h zaA@SoT#PmzXlZ$8#a|2>(%)jxd^)O5%aKAs&3xnW6nnW!1mk=GDSqU08J&}zCou!n zg-BdO+mPsC(}x9x+q)Vr2f#!U>%~%Y4WqDx~2h#`% z2AZ(DN*jo>1d$FFP81Ty-|+BFa$PO@F*iI6Yc8|(#9_1~%bw{{EBkSn8)qdWzYvJc z5#t1nU(T|*FEyMX>(OD{f#cblO!T3o0Au;2|9E6!F4QM>H-KtK(0vXjAxF9>rV##OIX$13XQ9W2(C3& zFW$~carF|TCIGkeiDaTQ@!Qs`RyKL#56pdu6Km8OXhklp0KusJMKH_P?)3zOk}{6} z-%g{y8w3!cNMHAP;E!*H_wkK z9k5avXjcW)1~c5UZy0xBz7P(Lp^nmhoLo0Z8U_M07$p!qcy1RfSzKE*$?(s<`$W3T z_M-JvbcIZ6M%pbY2yOr1uidpB;`cHADVaRyopFF%^dxOUpt`m3E|sHQrPkr|$?l`u z;7WY%%W{wL{dbE`II*wk3oaZ=a)QG zp>zaMFRWy!MdbTmSbBG{yJaMw>9^F~%FYvbD@n=L%SCX_Nkxx^2lo9;F%@G=MHVMZ zI+hi4G-s}a`3g_VkkkjFr?)mm6eRLP7d4o8iMgt%Y0cTM4`oH?9b1zQYa~0GI zl{+{~w#0irJ9bO?>Qqar(a*SS)U=HXgSWx{L_7!{7z2@7AfJ&^gt78Tp^5Ag?aln? z@bf1zafspm9PM&MJGua<8#QnZ2x)A)boSky*2`$OG#RgwZAOR5x>~A--eMfRK}vVC z(rGfWpP*68YPRq1(QSTza(lK*IB`w)YWMfc=Qt~gNDZhRr+ij#Vwklff-<<0?YLWp zJ)c-=RPXOM&a-yEfGgCZZ)$XV#L{r5|ErBowLh|NaE9z)!yu$4^CQj@Tm;j$EC25g zCFw(RDhN#ppP$nEATcK~tzwD?+W6QeXu_pw&0Aeufs6%PqP)7Cx3a>CgG`k07o1#& zzekEQzZh4Sbx?6S@Sf?e+tW5|_v{x+bmQOf0h6sB8nAn<<~L!I7J10Tqb-GxqGez_ z+0e^M`=9;?Nonn1^!Roz(aGW)$9+b|+zB9yg!?yz_o$!S^mSSzng@yPFgeydKh8Y_ z!33K*Fn?5$a@T%Lj5HT(5#m$8OI!mXbpyx|Ktdf7P_0BqLLotu8gKr+ro<0`m>J0Q zxpRK|U=7EkXmhZGrp*!im&*PI+8^TFF?~E~L>N>8PH2aIo(pTI@s$5@vzcId`l$$; zExd$%Bf6YY{$Ph)nWGmBfyE4x;_UG8?^{AqwTL$CQ({BdtB4#cVcGb2euEb-z|;)C z49DHLo!JRdv-YuQLLvb_D{)oIi20s9EQDe%R(ne49I9&(O0V6g^IQ61w~JlhG}=E+ z#NtB)rf;IS?NMJJsFB~kinS^Xu@6E$AtPH4vbWre_M*Cq}6p5mk3iZ7CMMTU{d^dyArs+71o6?{*U%Be2U#=HA^;4uJymB4Zz zobVX87tZ+!Kut4=+eiJWdw0~=1Oi5oh<_VH#V?X!(-y@kGI}(aCbk_TU$+n8{%Yz+~vSl)n}ROWX!hSbU5kFj!n<_*0mplYS|y4!^r3vXJLrpC7y?ODTlAlN|U%4>7wK4r4|b-)j*&V4O1yWS{%NYc%BL2 z^Qe@-_#fnk?wg(-fcBLCR8YEt&hC8zb?gRrLwCdlO5bML;qYC)&8~wQA-Kn)7+0Fz z+)jMYw~m9;1puFq2X!iE-WqvLVtUeJx^K$46HcogLzQaSRjwSC?UoVs%0I1G9S!x< zlwXv4eAw}!>S^72_n31_%gl;&`43wGj5W*f8f20=)%8jz)N&JfGjQ|A{?Hx65r0hW zu|{*HnO^}s=7W;m@kE_Fd5`0>rkT}5!7O}#M`ZP*C2c73$wswZoHe!P+Y)+*^Q5h+ zZSHF5+Y4&fg)F0@F!xrp(qvNCec_Xv?;Gvt9ZZWII9_mEf9q=EM^Its!E8fs?M!$$-I>*j3;< zZp}j5En>7Hly{UkQjnFW$CIZkB>YFlG|f!;2C4f8ObaCRRx@6W0eO|wtas%O6EE45 zw)jHLTY5H^JNV!%LY)o3h7&N3pag5eGFgw?yyZpX1I<31$3n>__$*fMo4!Oe)o8t} z00WD{7hQnjLqJXzW9t*u3j`EXmq^^4ubw5>NP6@8dW9ugM}-soM<&4J$EE6?mer>$9htkY)*+`Z&c*u}z3BVel)$lrM>SYz)+yhUj4U=+9o1;v zXc2iewp#~Ob@LE?mQ5pv>6_)5jz?{)#eN;;)m^g}BP+9fx|6WVkJolWUusKnB%+j+JEoC4Aqv2r%!MCTq!(*s~!AKkmQfho{2$o*YW|PvCTNN@eL#`#g2_Y6roa z7!~cvmrOLQx!ftPhns}M<%njHNvr;VPI(8+_V+~(G0b)-W3nX0tnaCW&EjD>_zd!F zIp^tQ2&u&nl!B@Uia&sdp-`}r*P)i80_L*U_DIsXT};1J*?qU4RelO_22z3c&Ylx) zjGS|-sB2bj0po9DX0REsf#rqbZb9{`2+a)vw1KizKRsja4a&o;uOcG5fXYh1nLl(0 zR(Ce@xX{D9<<2mKKCz(Ur|GV|HRwfs>tJWxaeO~IMClTTPVGrw@9Xb%fX?Sp>|6=h zdjN)l7Jd)q+a8AMiIIXbWjb(W9%B94-$A^fWA5 z;Hc6lz$B%00+tl6cA>|z>peh*!T$b@e#4|CwHaojCPe0H!7zwu;D7aIPLPJLb_aB{ z4SF0=qo6!ljUeV&DDCMS{`uJX3ZTl=kj~|sIhqo?d4~L0I|tLI(^GQVva~Xxlv<%w zxQx=~xvgpDV8=|^p_bQkLK!=;ELjujZn?=iWBNjjeEgsldJ{Nu35f~P@0lNbB||ZPxq$Zy{;PkVvbjh)`6^-zYa8$hF9QA)}30NcQVm1Tcj~5I5ry;n2(|rT|RaFL0 zpHqLqShf|ljBCB23d-D;`htlreUykfE-EulVT-Ixn#9-4%A5!nhBGoiyHoFDj~{IX z5r)$1;gssZqQr4to$Lxjhf1zFZ*-caZO~@EK@o24xSlVs2a!T+y?Dl!A*8{}UGop!uhcry^+Y9=?`A>(CJq%l4u6kDu0)dO~_ zy_M|-MwL10gir&@+pv*P+9qluWI5~>8H_Nkr?KC^mgZJI0hz4li*{i5+41-JDgIf% zKlaIi4L4PI5$xCMM8Hqu$Hc}iN7aSJR*mvj@KLccFRvNq)7i6a(Fo4~J&n=#7xSv2#>7FE z--b)6T(L>FZ5*TQjK}88bQ?dNl=rtoW;YXPQB%un4j1o*b%KJ)sWFpcJVTI$D)P0M zf*`qZ<^#VeC#qw>IREj^?XvG%N zE4^P;7y~dm+v(TMc2~hF>JwW(t2_%`Bq`(~NEdg@C-y@~b~z>tNDXL`8TEL}g|W7k z*XuCxt1b&snZliZ1dUqke14TW+~*~<8${9QK5*HkM}o+BsG7m#Fqyx2e60PFwP3eN z8Q)S4#iy3MyNzRMkM7K0&y?EC+F2JY;c$4?NJgPwq$8>D*bkd-w6uvvBxy z#&Gl32P}Vgk_W~h2rn*Xzcz*_(4(>Wclb2%_k*HsP4$A$^`9we=PDS(ADh%@@X4qUgyfaOg&W}*4h=F z48!=l><3X99OAvlOv|#sZKp2H>d)G8%o4#Br8!{lC?m(hnrnWcv_GT)(m9{c$^&wc z-&xNtDR~zpA)dgosY^tKXzG5VM^sqbvE664g!Qe_M*f&zP>jW09$0DsYcxFYMN*#Je?~?^Mg31 z!a5+!+Hf?dT5$drMRbEQd}m5N`vrbc0BMmhL{nC4LP|L#rBasGRxXWqPrr$WavL8pp#eOwq2c~Mi6w0FEkWVyyhj26RPGdbm1(?jXA)BWb^vccWM80s*Nf=9m6 zXqiADS)@#}w`{;#8oV%Rdu-y92Oh~3UEdY?OPT{Zs_A7NRVP5{F0$QYBa#iv>(sSc_Du_)~S71RotYqO=W6Ef)9R?DA&^ zV;>$hDGBbWv%ma$>mXN(YUZzKJ6C$Dim&i+)cESrT4Ez!{V%ME|H}z5+r|)7aJ)14 z3~$uVCk$c^;yh9JFG8ux$ox4;H|uXW;2Zq28D%PUAn8Z9eirvW&S^LPzVf`uquLO* zIIftOKT5~_PwIp}1OPu+o3yhbS+<$us!5ty$DaLcch=edh|jpAx>{=4RHqdT92Bdz zT{hYx=|Awu)gbMi=4w?1J(e`wY)abSeZxSrv8qpwUpjWC!v$fi>C*EDhP`Be#+NGF zZ%yvQj=U7#7zO4>o$d60tZTvXWZS9QO(6csHWYN?HqVZkvv3o;=%luT2M@j2j)6m@ z^=z{#!=IDMXOHUamp$4+|Cc9#tsuXA19Hu)TF?B>bleW%bxd#?KNrS4d;2idw&>xP z4~~GsQe?8X4ehh%f4L|=_A;IvEi%F+@FmL4=nfqeyvr&}=)q#lFBoWQTOX2_7XyjK zXtx+4yBU{`ro>;MW&4zuz@-wy8BI7;yA~BVpsI)PdQat<`=|T4nzP_HcN_06iIu;m#`G&rUv^$uzlG5fGDG)WEUB zeeks`aYAdSc%Xqf8~NPNGz&fPjAnGVFaWv1_SKUc$Zj+E7hMyN|2oB3w5Yd;6d4W& z{6^PeV^0QxmVO&O5}sT~iu3DN4sxU#rt#omy@MD5l{e^T98=?<=vNhqTf}yzpykB;?|LE5E2CgCUsm$cF zlLlPM45>4KRGxWqWd}-qNWYsDL)dRY=d?dq?zoR~`#``F2G(y1M6*;fW^8bG{&?mNwNt0Shb5xIG96R93!(&+0+kP0zc+8}j zrps~&B)z4&117h{QSn3hrhA`*FMXPIXL)l?Jg)*tLl{CX1LVl<^Cv)^f67o(!cUYm z%G1yb5hSiVPM+1A)mcm^7(3E|`7vqOHMfRdb) z%`2`=GV)>wH`FH_?=!D@TPTs&33O5(^>a)V-yTabjAv+cwKe=R*G*mJCB*4iHqI?C z9cX7UuJI%iz9VR?c9cXiFY#cY-7Lg(zsbMo;VYAPYopvb5|_ndjuBMzI9v6?%i7)U zaP-Ww*K*``7G`_5>BYX9r5}&x8OwsZ{Z4C8aka}x^5ytm0c_K zH@jBW^66hNEt@v~;S~vkp-ScAED$a|${xd(Zhg_P2@k#~N5-BpClB7+Bb8x7$VDhS zrE+>qKCgsgfF?m)+9QVO-6B4=BB059q1$|@IKX{F1MJJU5pLDtGa~ZrKnV!|4s*+&CRUbD zcle;CEjcr@gA~x>jn%qYb5oLH^-&)po&iGWZOL2RYEF|?eQ>yCBJjJTEi-}oVp3a; z?QG!eI1c#`%WGMMdILv6zzNDW%y$3vq2B070+Cih@dYrL(Oku?@BYb+A<~hIAReJ(68Y$ zjz$Xf9Q+c7cfSS#63@ETg+p1^_o&%pgtwopK}!8+oK$L~=lYXsh{OVIgrxue4CSN1 z047a41@~p%)LSXXc?4{xKeqE6J=w4nHJwZONv*ukWGOysa#Kgv(|J0|BeeUWQk5lQWPw)(P9Dd>aqDo}NuQ*mQK48t;BmA|)aK6t zUFI;xbEV1u7oaYEr0NA(ZV41xsSKj!uJOsnG}JG|O7d)JD5PQpv3Yr^Qp!>@&RnD)%*8*{HJ&*yNqclgqh|#I)wW~9(IOmn2npN+Q(t>?F z@s4U1t;BRXM0nv}-?#+evh?z=LMDvJI;e?v%2UkPsZTK~+j z$WV7Fruy}g|99hoah2#+fj!EI$F%g0iYv*WL6g0W*RP00^$PF2ih85rwm=RgI^JI4 zlQg7|c4~Di>F;sow?Tr^Zx+5b&x$xst+6?F%s6QPqlfl;4lhKCXptpL=gRCed5bT> zWM*T=yh*^{7N|Es6lwS2T< z&+JMK>pzhcbswwN@;A@{uV?|`MHtOwp@AUfD-_d~hQWIF$(nVNj8*!DOIIw^x?Vvt z)^lJ8Fl`KQR`#AwR6A{guie_Rf@^gx@v|}^;5;taA+hg9*-CB@+U?MV8EBPNq_Tw~leVOTwhr_`qah_kx(5ijIa zexZ2=ia4r3?AW*bUXg`Ae*w`)QI5LJOYn^f{ zFx6kab?5fZ%Rl&TRNT5>WM50!uDJ_u{B*`xnKX41=YDqvp$x}1$A95GgT^54E-Lv{ zX8fRiq^M_YATGg7O?z_nbC@0t_j?Zt0fQoH;SLoa_bn z0yFTAX2C{A=i2erF;*S;yi@5L`PJ~x0}3B}oXe>OwrC^+)t}9l&sauOGG}t8Me%{`F{}QN!jxo+=FhX6A4%beb z&-TwWexU;Q4&f@#fNhCAk^+3Nvgd=jNfB00rLcw&qKa5Rte>B zL8V6Tq*`Z{>VDTu7ZF~GzOVdTFIS>|#El6c-Kw{zjuF=(nRo19$x}sYvJ4R9k!dew zGEX}fd)Kn?>u|Gk>>emd+nmPFec74vNo_ez`Kf){@*c zo>*c*e<)Ar*m3XDSLX)r`BC(8@X<9X6*VL4@cplQ^rWx9!qW}*w=Q?^e(g>E&GWNs z^NHnEOaBYn0vS}|#icR}`1b3)By6gUtHCA9J-$t10~@y^N5W+`dBlU=-Nx+F5Yv)1 zZzz8-9t`tJ=FV)?S&M?qMTel2x zFKB#2?S{LeJh`!dZt8&=Rpehop6AYQG5ebYGMiEj1C@w!p_5sP2gbz5`|S-0G9WpXG*Th5MjU(c$FV%r4cxpx$XfqVoBR-!2UG#qX-sDu9%ZA zjHh=$5D3LrLF?nh5A4z!^NB9;p4SQ4>SPZVDb??@KB9=l1S)H=mLV0tfMKDAEvvBk ze{A10LiGn_~Bq+?s_H zR-LXVL+$YNgI1}Vb8CY#jn}G^G1tQazqR1{{0Wl!uLnwxZbkhX7ti${SfOQRV^%H?>sEQUA*ir&v8S^O`ZuQicwO;f(qv99kp}C_{)j2 z7D#KdqaJi)%L7W6I6ytn7GWLcB#DVuJY`wE^QiFF9v--;DjdG#8KY|8LCE*IP2!TZ zw}^BPAlMeH7lT#i0T{FIK=jYET-F&)!G+|qY2(#RnrTv1Z}$f3ZG*1g`f8?xiUy7e zUAxN4GglNN*8u$N370K_u!4}}uM-U#=R>$)2Zo01GVaA2>tjr2?B(_^Q!G|ovoU}6 z<*p0jc4uORpVD8mY+133k$Uwtih2?(OYv*4-`N59)EIQrjH)X_*s>R3!Khu0j2_Q$ zKogQ_Fw_5(Au=ZR2*Mesi8^fnaL$}=>T#BPH#evDdg33fKMsU7JyVkq2Ej3ljwaaB zGS<=4K^EU_zp`;fV%$im!1yIU&k~uT-RJqTf=d-EWB~MdZ-293=h<)GOjD%gU?Fg} zD0Nw(?)+ObfX&1rJ@}7^5Y-Re%f|Jd29eLZSUkwsx2TCD684e0!aXclAtaPSJweEE3!JFH_u&YlEnwL zqB`ZW&~ES~$s>MQ3Yab%<8#GsLWGK=T>>ds+M?)PQ_StS8$+YmcvKb$p7OnCC0C@| zBsJr6We7*)Q$#1dp8aW*oT^R>84#mhF-QV!<0BWKW*XDh57SaJ>OCxwc z@+)gPYYXXh_{!Oza9MriI2%L$wUeYxwUE7wXE zd6}3Sh4>V1Z>e!vV#QD5PzgMpy#Uc{xbi&AoC))ESvXAa+3lx2wWwSrN7j5F8HR?Y zk#I7`s#+!#t$NU+b39_dJ22F%Xe(Bbs&@%kt-L7vdTB)In0}Kzn|0YKUI1<(~;HZaVz1H=?|T&R=f4J{9p0$xhT=MyyOQL9O2?Re!U`=$0*HH3~r5qVR7ci!-4zwDH} znr06Ww5Z>U$qYc{plu8XN)B>EL2gi4pX+`OV!^_!2ViYE^L2Z$nt*z+dAgn4oHAwZ z*Tx+ZJHn}()IqeSf5hn*uge~M3=G}|{*x0ANJI!$1unCd3bb=lJ|Is=!_mwdx< z2Ik0HN_5H>S7wS$Ww}eEfA0SD=7}a;(WrHarPjKQ?$3L$+rb!QUpBIu-WxCrZW57T zb>fwbq__RNHLs=~yvQKiaOm|Zze|Ww=FsWi1eN-0@4j(rwzAs5oKbUG>`xyu0FH0% zHXp7}rM=D46#(17P+4bV$C1z6fGum`B$Xas_0~We%M-ma3Eo*J#oYYbdkjU|Nv;;BVH5zu2u-4$%T)uL*9J9RBmsg0j49SHWH zTQ$S&FJ0y>o(6@;sCjA|Z?4#%znJgJ5!Y!1;J;xrbjhw=p!9T+_TBB=y>0V&IR@4X z|Hz@8WuKQx@##hMviGt5OhYZLg!}XSbuL9MamW6BprW|Dbz5CPjGt**eJI&Q`3`Mg zO)3dfV+@s*i7JecW2rXS49GY=n`S)z_=Y%vMmDy~4`puRw_cfAr6j=*CQmcv*lxv^ zovHhuNiSfgmfJ7h1ivmsnFXyC^vUg^!}9$1MqIeUOwFTU9?VdKO>aS z99h*siDHv0L`Ad#)OD%uc-Rk2wz3%EKKb=4$o^Zbr;NcpA@*ZNfYCMAw+rw zwgB*Y8CDAWUbsv|^z)S{T#3W2i)A3-PsFuxFA@CiGasI)WAQ%jtZ-k(D2 z)TM_BEPbW;NS9r~zf|$1g^pQOd<+aK(s``;c3?xDyN&EQ3^s+OpJOgexqp_rr!r;q zS$p}_zGlNFsiP6oTx?rDLPGSevtTP=Hf8GsW}3`IxP`e@m)JS#oJsgr8SEthJ_U*=SC4KHkw$czsIcLcR zHRV~#o*lb!A4xy|_9lLn^$CsWvm0}2Luo#?KyRIhx_;TR9z3J2kT0m8BxM2Mn#W6} z4S*=1e5IvVW@V{Qd@v}rY?%R@S2p zX5=V$gBdVJSoVR%p(_IV85XadBqu??10sVCXuo1+UArBog8J8Ge9yVgMq3I+AP=ty znwdXJ&0sTC@{hOBB)~odYwZmfX;}io9+V5YPl6zOs5sJ&@U>DKPy>gq??EWq)03lm z?>9nX1rB%@;Mu4Lb_hfzK!=qL@~A%Yy&0zy6+jd(h+DvItexavv0R{t$2uKkW=A|j2$3v2eJX9?4z;AxB5diG5$*k5x9*bl> z&e{er7+3$rQi^&Xp$r>VRn;~^zgglKU=~_f*^d%q%hUiu-Ij~FXok;~w%^M}oZwP6 zF;jh9n3jAhA#-`E$M5emF9^3u6&NA7^9mpl&`AWqdeePxLLCDVqz9xiGAy%*H1;d< z`{@&YQC^)TWzVPAL)lp+WCR~ZOgZIHJ)OiHx$tsC$gOqjf)8EaQDTMk*p=YfLWA;X z95Pt2X|2dkJ_pPNGRRq+e*D|g4+Sx@g@7D+HL)nEyf=|n0WF9BtPO_Z{;%>17k|xyt}FR{MTm69$&Mak8g1^+4sb zPN&#H=S=#GOOwVikRzDh;V`wd8Hj;5faWK%Zr>GN6WHm0=)k&5Ge*ehZ{|i}qk+NY zVvo5P;&W&0wk~-K58qrYLcw)_tz(_FYe?eo0PeZt6IStq&iM1Mo`_+<&5pe00uR& zDSo5>1?QeG3A%xu?c9R{o5D1*@6ks+$IFS8lZ83o-8HdtM84K*r){Cv*3A_74%|Fi z5zm(;J{i>x&FFR7krmP8b#PtQigZwm34aR;i_!G-RYe+?nRl3<`KBT^-fAIwb(5jA zH^~PZ7{&R8;Ol^cU=?6MhRV5f12VXmDA?48`z6i^2S+s7_MR+pHP8mJ zlVwabzP%M7-wHMET^tCl5K2gRIn-y;W#!Apnx~jpZ$*zafe%I#z;o1`mg_g_dXz|p z&=?Qrp|p{^jnSwoy5x$x3CCZ^WZlP%++z0Vl<(_Jq>Tt1F_XslU)hvW?P2-8Le~4& zJ9$`gCB_ME(MdqbJ+q43X>iT;>tnXU3Jb z50mxcPcrz&B)x-6d_aG<*$-?IVTLnlIdH6aNX&`wIahPKmqmX48f%~(AF1ixkZ_|r z$@$am2jx4G-n-Z%i=Ht@zb6~}fpgxFd0%oyW*=;XGs&q#V~)Tyrjx{zUK)p(2_ayQ zjMi6bRl1ZHjdJPy?aXw~;Ga~@Iw~&JiE;Exj0(FZZBd~hDCNi7M2JT+@c*z=^qi`G zyv`schCXIz?r9Z6wX1fm{ecY=J#EML3J(16cMXI`tlZNKsRMSW(MxHd3cHgO5t{_Y z0_k)rm;{M_V!C8n0?4<8F8}h1)GhJ_LNG*xA~&_{7tzdf$2)Qf=d#DUTz+8!dDRqM z!EPE(woVMtx>po(G-d5CG@tJAbN8G8p*3euAGv+_PwTlWS9SM_8@6B5-g_3-LpH)^ z*tss@Is;)q@^3_mQP6`TO_cLjF68)$1UQ&G|9**WAcP_FlXF(J@4Hp}H$%RMx5OLD z{oI{tvz_~m%C`4xR;SE^($OF1j}Jet){F%X4?ym3sQ&K6-B&YQl*T?m;$uN`+o(0sy6kedpOqkH!CQ|98_B78Uz_+_sb+W;_28Ov}*0(t~^XHYaf+Z z4={*&{C<$`b?C}E*0?VqU6-mi^vwtd;afae|A}0BZWeUxz*M+dzbTXad$-WAH|9xD zH}oa4C$e38Cf{lRhp|Dkmj4%+0d%rpnejD|e<8W-MmNU!Qr%(CIH6WeN`lCw-fU=Ku)xI_K%7HCSIqOlPY zVRT5HQTf1jk_5DF51@T=ALSJaav8?1@2Q*und>y42%dKGYez|xsAm%_Ha7ks4t%}yRrKcQ+& z-h(0O*W^S;-$!KmGU-vg=(LC5%gg6VU+=={tE#K_s3M@%&cpZjTfTk)=en&c^-5n# z{P)4=@CQJ#W?=zpt7j52D^|r*BH!x&zLPyEV5_H5VKFmNeXr-4wzGw(NzP|Crq@h$ zO6Y{Qm#e>4xmemSwFew|jtZ*qhK`!k&wJ@$|3JK)Y~tJ1mnzsn4-jr@bsd*PN$+}M zrP;>y67k=o>gM`huaZ1xcBMFNw@u_)L#%;PF@tH){{pz+84lOT3tmA5qY;9<2Ah>l zTg~zocxb7`>HkiFk8Aw#%ERf^VPN$#8C{*IEBd5vu+ki3jLP#8xLDKdWVl+qHaT|u zUDBSBVJ1*~^hH0ZL(r~8mQ;{>{@yG6Ub241qMyv)TgNgJ)j;wP`_g{72GeN8Nay!F z`{ssIlliyV@qWMihGS(@*tslJy>M;UTnQp=O1i$`^G5 zG+-mgR=?M)FP<+ezXC;P9guVmE`@;vh$kV^iI##=qjgxrB+pR?AkhU8@FXW92eg!1 zz{@-@pDWG{W?l4v;iXFfol7+lOoph6ozb<5(>}RcTVEDdQ|USN;>Aa#C%U(O8EdTA zR^=2558ZOn{QYB+>?okWnm^Fml*p{Az_2kSvE@oaXM{!8!WM_&ejW#gcH&kR?W!z; z!foXNed=wgtnKxc~2E+ug0iqfJKXSe@6RKTaw-S$96K^0D>i@C)#p zc)CAO+^ zQstt&R!FkB&MS$uT~RI<0vtqodqu{`pr@8DGpsrVrNDwNsut`?PqSBNo16QTq--|j ze&B=wtpSx4dxKyLm~p1{w52WCO1P?HiL9I9nLZlCcG?<#Bnx=Bh;%(_2$YSkf>ymH zyP|9z2N1_Cec`#2K!oKK1Xm<>nR&dG)l7IacMJ}h^gpG`z%NAY3dmyfS;hehksQo0 zq5>g{3qj^A*5CoxkAHqW26)FVL9nIs#V#L9mXX8vH)μe{M&^;LL|(3Pm@=lGs+ zbkbj?ei{4}A&tN-8%?fNYsQt0B8I~cC$WV}->%?wly_x7BwTv4mC7s3l~})nb^afx zjR7&cpY|m}{`xc1QiiSGTSqRN#kXUVXmB6{(SD7N+k4Ga%l}(@zkXvWVba>NcsuCs z0aN_4i`v1m88XTr9-~n_|I@V>e&6SMs!FOuH!myeN@-LBwpeIDFGPH%slW6EPoC%& z6h^8X$FS}QMq#V7w8ZUTKp`$^_W1=5u{B~H5Nb6GchF_y5j~L81uJ#&=J;{;*+F&@ zXcW#@m}!z6-xv>o=)+@6?jp#FiF8wQ5PYQjg$JI;>T-JQ!##TfxC`~*UpXs19sSAn z(cl+F|G_pIQ3-UJ;T=2;aVaW+32&^nEUNZ@`1MgvA2&ga17v^Yg7OQ*XHxR1U!VPi z6X!_!0QKyk=_k)qQw66f?d_EeN`*G%^|F_RqPG0ALq&B>o|OU@7@%-)I;$-xP9jBUg0vg@rm9Z{`~2zlqw=!h zPCJ4vY$>_YpH?^-7FUSKqDHYzpsXcA`7((jgRev_0~+2CVh8ZHe%g)}(s=rXSwW;?cB}(-vLe8> zWcFh_TeFt!5mV{*!4bguE=tabzjYF7l^%5$PVfflHh7sD(8~)nzPiTV*9#-VMj?6K z9J4Bv|6%=@s_mqOn&QFY06^erGEq%eusM$U7wKP(FTq#b3zCy=J8KWn^RAgOylKH(NUyKB3^`t>1IRmjZ z#Z@r#2&<|2)f@q4v@D89yRfqAsk5Vk4K(kOz#)xSV_i|?BxBNH1s=ofwke;p*vsxbkjvpccdi8ML_r`m_Dh>`sktdsbQl>eO^$^ z_=s|=zA`F1ux%+6@?c@kjQ&RjpMYf>SJR#YP>-}f5c99iJfSy#_k6qo;mziQUFX3W zSE)Og;U?rP4e@e|>VDs$=Elh5+FyLOOQpxwjc))aaDSqZo76W-zqSYp_RLzYH^7C( z_wJ7jlLItvbplgw0MP+dx#066$RfTd>Dk7_UA>UhrY@7X`K}8qi@5fpikXh$ll<(X z$&?$8M`PheJ%`^5jYKB3)0)5HkrHVY_oB0!Mk}{Jh4LulO#&xLQrlmwz0{VO#l6Bi zX6f)vQnnt`Njqfjq4NFrSDQK{hpyLcTdF^av()W8rP>>fz;#PtD=&4-eCKFp+@sGX zz)Nv6H+_%M*qrP2MN>PLdPwC!sW=gfgk7p-$p~&Hb5!r{atu6{;>E6@@*d^moj-?_tfdOJv<1_}rp@n1--+&k z?%KYIlRE2Kd`Z3v9yr@A8-dqqzqy_x+ka^NN9mVkUXz;A;QfozU!j@2G_3zJE%a)t zGMpFP#B%Jc@wiVjsI82#x8i^Atw703MUlU(8CL_J(r3TVBdG59)SO|q%9J6!5-pb> z=HXm+u%Z*CAr^N{m7Z{eWMk)~DPU7xv_d;7`s6}x#^R1%F^}tbOJF&zWP#(T&!%>@ zb#&$!*Wk$ndN81i3;F`12&JRBD%X%wVe6n>`PAVIP!Niap+p&=9YjJozfW96DdO8K zb9#YM=i~Wg)@Bi~7-Y~y?4P3V=QnRda9dCu7On}e#{bIOb*)GM!-=JGBuvjl+^E0~ zy9dVIQ@bm}#@R9fMp8^W;>nVVXrne zgKP)PAPl`qbJ~R2dQz?BzL5d`%1k)C=2?u+#fI$@VLitx2Lz$>|qgeyy zUPrY1T1Kq$*2?sy1n4jTU_j9TPVG{D_xq<-1S?HSvd9#~)76)Nj9vfuo2iex}` z_3Bks%WfgRTpxh>C<2k#j~-2|8qg6vOWwUF9c#d$YqO3BmJ0w$y@e0F0D8d)AW?X8 zUc=cB)UQg6=r7EHZH;KD4}eF!G!AbXg*#DHbP#D;opxa)OX#BNmA@Eu86% z4lNr6rk0X0VTJ4p@%iacS`u&Vi!7PQ}*ISg?X zngk8n$^52{mSeR~|8ZgDOu4j$5WN2+vqJJDuNdG+sd+Je?;c1rLbFIXp_&-=wewAg z`2W^fMnAvhj9~gYMb*@XHzCL4sNpS}sBeA%kIYN+S|35>2NBmQ0Rf`hx)@9V59wIl zFR=+Ahd}_|WRrpDU~a7Y-T9P?I|1oj%$f)G-5RrCC(Xw2HVApW_hubPo0G=4;`%~D;8^qTN{P34_?sKA=BvMB||jrHO$WX7=5T!j4@f2vzn?M4erg({%E9-JT6 ze?WPD5IRBzcpbAYGU@b}|GMERxDRry-KGy?j7_sEG+$=*bv{7|T4ujpua5eor~yWN zKiYU$u6ISdUrES?dmuYlT)!j=#}ec<(9q__SqOu0nfAaTI~p$i25*;S+6*b5Aqe^1 zCZe0}_NSkhbAOMS=fBMt*$OM z>)OMsll(aUnn*5LDw50G_X3(TFPRU7vEAwrXv%0U<=iL5P`&FFxd^k-N)w%t8HQVP zNos0aV0qapE}@4pgIR*sv6E$%>P1oCc9 zWP>tpr%X=HxBV=j2&mKn&*1sNQE*-bfHM}hrM5p*gv?`cY$NQL0tkijh9-*=S(+bD z*C<5GjFDp>9r3Vmk8n}$jaQPwYqq`nR}ngb@wY?a>;2cp=mm`9tvlht0gj9=N&2uUs*gC`;5Sn^r-u4swGw_#j?2{wa^o8E3Bby46b4M2L3 zqYyQ*!4uFm(<6!=6eNIlAgPBp9%mE^04tV0F|Z*wTz}=x@-d9KF&XTLY(i2OhKD8I zHI{;dG{Ov?q4)2r$NS$-N0$oLK2S%(Wv)A*y7VqN(-|x^!lu~kEnGp~xCSR{Hlg;( zjDwgnvvG_?7La6$mOOP4eY(0*(m}2uChE}61F{s!sOQotRoOscIgOYDn{@|yGqjNv z7YN<02OE(oC=@AfhY~2lN?z_5mzV{gVAn+FZ${oG#n3=gOJ15&&|TI+h62?wAO)Cg z1iP$9O_P*UkzuQfQ z1rERjl;tYd2wM7#aSRN1ozE&!8XB7X3jpKL7@-IVDs|RFo7#5s&q0yG`$Nl^kt_+1 z50Qt^(1GnLLU8Fbl8R6v%?I2V8vvT=1;G}90%jS)t^4Y+e&hN+4e}|{sln66!jE#+ zB0ay&g#Yh0qw$tbvtAdY-}czEyUF5rlhTg%eoOHnHl)_^@@#EYi~zc z$wjbV)cpwe831s{lYp1aQg{gF9Y@iz>p26k;Q&~dxn~6hFLAX z_i!*&hq;xZ-&d@FNVjPSesNB4t^iWiq#>oEp^lVU1F(6MR8e?VB*7oe~ndDelI+6FXlf?C`uXe}fPpl#fhF>U$UB9ZgpVM^^~uZ{Lz z76WF7GF*z+U%*UqY)nArULv+QRB($1Wk>DvISF(XpGiZ_$qdj=hwXYoA|6*-V%(wv z)X?ferYz{9xkwafe&3g z4Poh^7`|234M3C6ToPN0ds&N6<%cqek`hK$Ep>l^hoCaB8?#}R#p<%$OGC>&=8Wsl z#Lql66%W{i9y{tw;1i##I<^U+k{0||$hJ9jV2hpQ$6rc1E21bzW`gB+eK+#u4cMO(Be!c z&|`v?O@_67L3ffgp%ncHWB~>M76H2?MFtdzk%+R&?ljCYmw%tH5MMA(TYrsrH93`9M+65Fn(`)qBQy?TA2opro4o~U(n)4X#j9Hd# z6<9V=&P_Gv&-LklC@Gdmx!?H0$Gy{oZ@gcv?fo(sIMWtwBI@V;8!~=&*#IeOca$3K zUQW0GQwX!d^&7LNj}@70=w2Rz>cUQ)V#{tWbWI;04J|Lx{iHRyFcvJuJCgx3K-Evg zA?E>GbZ9z*wG4B@f)#fHO4)q$1;kwvb9}cw(A<4DcKLc^^yn6J?;tI^698>}T6(LE zavb19Q^J9jWT5-GApevZc?R~1uPi?ZR#?Vt|AM~*suafIQpE2ToOGi0`xh)>5!s|i z0}}#(3?XNHThj2Wkk^%w^-pG49RRT@?S=#y&sk0P)z zHhLFJZb3Ds)hjzEdaL_F-&9vkP9wxgf$`NQ=EBO?(Cp7g-EIb7cnV*n2aepb<^hxdZ z{ai80Uu1oIko7o!y>=Wdv<)2KQX#44kYhCRm@IoW^j%=K&Q}Bx>Q;?UXgZ<;--u`x z??O5E14t~;fSvsZhCw0LfC7#*3*YEht73z0;$blP%_+7;N1|$KmP+bb9X5GRatw0jA2qlY8a&QwF={7Q8 zUzT4wP~$i6He@wJeTUbe26j=(_tAHky>8;QXD#6lxmcm2J@_dI3Yr2cd;mNRohlC8 z<5p+yPzR*kDG+t7@f>)5_*v(Hl-bGhyHZPNkRfY@6@}iynWRDdC~Pz*-7Q(dRqL<1XECJN_sZB^HKIfuF!u zS`C{sF>;*dKv3CEQuPS}Ps16@K6g$hzx_=%>^NeOBS|*1&+i!GWpF zR!xug2$(ugyA4QdElLA(%m6$ODG^5}J_G^1A1zB!+^?cdAk{QHj zIcr^mf%`+}F6SB;7rk%|Se4iqY;gclMEl-t`=ZKSzTk7l6&ivMCEWaH-`e=Vbt!|q zR(^e=a(W=>)XfVLiRLO%x;2CTxZ|M$i02CHyb?K-k#j%^>@SeyFjr2{6~VEGzQL0G z9IV;=z{(L6q6a9@8`su{U#~{Ey^FoW#k>6q;FQ_ms0Ki61a?Pys925YR846QU)CgI zzIqc}a}gIcUkb2I106_SSUEZID}-Q>A(&^mgD-+nUTCa9ZxZA;AAIz$QgxiDa##f~ zBF^<{z#E|G{0;ukh)~k7ccnQ8Fz|PU`$EnqDI(n2?9BB4!`pj>HI;thqY)U1qk@Pu zX)0p_Y0|q5P?4t6y9`nSgcf>;g#l?nMS2%0QU&QHL0U#2p(hwhfC!Ni5JC?m{~hM{ zf1Y!3ZqCiwS3K<=7s99q%vi0}b!F$6_J(o>5^xuT4Ts)Q~ z$}TzjQQ?y66OGj~Qx&kf$T}HyGxG7k1-#@=&CJ-))IV|_cFACJ`?+6dN`!Rf()rjW zBIjHJ?nhtdV#^TNz)vmG4;aO{xW%Lhwc?Q}wFd}ov*=ZAN(f_s8DeBKGDV`%51~!l zc4{Im>)YZ*R~hT`n_d3QtQS4^T2^-K8vnBMU4K7WBsLSgRc&mW8Mrd}4z<1oJ8@+M zI7NRLAY6gdDKygniQz;T2H0z4KgVUI6DKBkK_=o5Rf7KDDLJJRnaB2ZO|}Jq<7T0zA)-MF)l>PdUqqz9zK)u562)I=%;zXo3mTl^R&^AL2)nFA-m ztH}Pyj_Ae?0*zjLwE(ST*5{=cNaDSa6G_I}>F2i^5cGycKvL9CX9p$lmT+NV$I^(A zSP%$Q@CN_m)$-_Jt1uAMB%Hh)^r>%}w#oql zb*6i~{F*29!g}N^(8zNF5C#3qw11dO?cX zfVm@fKT&hP!1kNS^3AfpkAoarMW9DD9s*|`1D(I<7Uir5d<1s!`_+a|`VqC&;8c|b zj|I!by`3tCBUd^Fo1F;)9uk#3dMK3j(2E5g=&xS}iZO}g!q zV`IrWxhhcaN?XQZS=FP|wjet$;0ec%qRan156oN8iy|6!YbzRCh{NeNS*yL82Pcy@ zQVKg;EbTCBZDw&CZF<8Rs{Ys_=SNNz ziQOjb>|hphjMe;G60GKL4c`&IqQ7q4vX;}r3&KQ$Q=urg=gph(+nr)9E;Q6!y^i4@ zQv9`rnX)ojrC7|2a575=V%0m zu;&hI_JGgx|4tgfvXE$67eT@7X2+8^x<8!_8K=QLRPuf~f!G>)8ydKwlF zq>t+$vb9^3q#g1?F38n9=k^2UZG3WW$1UF!v2`ap=aZ7SjlH?*3R;CZUJj|{oy$sotoxQV#K;(6y>?c&SgS9>a^Wss)F zvFq#2F4sAi$H=EHm*zXSg6AZ{tENy{h<@q%7>JKP&!~mN>-qzAJ*i(3p&5!@Hi61^E_hno0eMtDM$+LU%tL% z$Iwt>6IvD_I)Q3j-V({N4|&eRHBj>OQL0yMkl46Gf8p6zr0vSy3`}NhQ}?aR^2pD=R%snQ-91}cd&|Vnpxzm z7JKsa7{T!g6Bb8xmRZ}wtvaz&HnQe2x+9Y7YNRN^-?Eco*#dm|A+<7z6ba7ZZkI$z z!1uY7Rz#EC8~hAU9)(h@x-)-BLsBl@SvgbD>D;PnFLJf#jn_xJP+RdCA1uG>$&6#F zj~UPY1ep{kJ%98lOds@uJT%F;91oGoC~LFi&C&}Nx(IoCgH#jZfRep~&y>wbXV5l~ zFK^YKo5-PWW$)44e%PIGnHWJx-dQ<`U+blJGk7kfz6^@|Qtbr;3*!Z5Jv_s9Xj88{ z*7;6OYy|p>@tng1`euQj=vek&y+Z}ohE;f*&On$Vk6dcLuh0!lqi-Bpug-T8K`w%a z^lvoZD+x1Px17}Zu#Bs+{k?PxIKMj_^$Gl)`8&xv`g$6N}I?bU*%)==O?H##fC4-q8ZpR%p0b8>VNBCPb z@61w$wYsVQP&LjDrqOWWch#Za-QB4fCr5-%E?fqI+@AgjY&n&2H!>ZPVl?37`jkiK za4DZ3+n^b7Z8G7v%AFaoq5k9ge~`}i3Zd?uC#UuVhrCGH(6Yr`8X4&k8)UqeWu0DQx zN&l_L+~gcBoH`eJZo)3)Ig|E#cgBC(T{LDE)S->}J%Bp4b{|q@m28wX@(x(@8G+lB zcAgZjt3;#Jmu{XGC(OB_yrR^v5^gyb3iwKPi7F?}4^1OdFTI|Udb|{?QR~5_TIic^ zrSEQlg9@6PP7%tgNUc1TE|DYE4z=4dHu0$*xvz<8qvjV<{bL;4tT~u<&U$65`SFSt zd2?=mu)I%C(ifq$;Y)4|DsvYr^7a^+g4aM~p_oaPn?VQ$oy`^W;<_YW26;x|&Ud0R zuF&nf2sOi5bY>P0A*Jz;p24 zc2SUX>2B_~cJ1tn{gWakWBY!(sayf5>efA%)ntPr%Ao(@{MDazFC`-5<(h11Dw4i^ z=ER_L6Z(9cUS+QHgNk@g)V7zX+P?|Z^QYzdZYh?AIdd7K1LX!T-WgKGVFj>{gkx{1 z^q^P0lD*8Qo66qZJrzmt&5<`McM2!?Pjz~WQ|!f5*w)5nam%X=c+M(RC}OHcC?peH zOt_IPKPOV(8hY(kVqLC80IF!b*p|J;wEB5KztpR>G#gFsd0OM@KMz0qXF30f)yxg@ z?N^Ox8YoH)<;<~g4B)k!Wi8B)C7#EN%7qDsa>sbrrlWHWK%hs4JnAUAMvK3}FR-M< zn|Pm_uNHz=dJO4)CAP5-ryFIvtlJwZv@crR(;u{LvunF_CP7XWDT>je{Orx`__*cY zf1PbqwFfI;ffr@`x*K+$>%K%I^3PSUt#hjsr1T}EyV~N0XCEd(tsC}6a$%!!J)~bv zj6O}L{7f8ru9>C%NMl`4qSCbgV&z1E1s0nh7=m3;ZjA7I`W7SaJXzQ0y0FZfAJiI> zC-ODi+qrVRHHE_Kv}d-{CgT<*pH_0q5(uE~c`Qo(_2T-k18&m0oM5AqUTJQwH_}oW ziyGp>zL~tnc4@0El+{{~`9D6V+l?^wwH@cCszc4y@$eO10zcF1fdgYs`b3@(Y#JqD zSOnJW+aBk6io~C7nN%j;7I!yT@WWDwTWNi2x{jfrAW9fZ+Vk2^qICLF#@eu{^=k}S z-bjLWet`BlM_+I;y+P;UrOaZD6K(!s1)MQO3CeOVCo-J; z?)8O&K-UO=R}w4U^8-xf# zAr|7d1Mos8|LF&bD0Mssy||)`F1U=C9I=^frVkH2#*vfZ`cN*QoF{gXy{}wA88Kgk zKzS!BRy_^M1RA&_TW6Eo;EU6_LB;wGOAnH$8UC)Hz8YtwR5E8#mwdEUm$t4U z)d<^Qr|9*2K)>JK>pzYl#?x%4e%Ny?bJCiQ+K)8=nm2gwk_j?Tjc7*N97ow=Z9{(+ z=4>uD1bvc<_^PF~|9H*Rt4zxjr~O&U0!E^LW$hF|AVn2cbi>;;ry`}3qdRUbsu=Bx zfXOusu1c+H7ZzNO#+u%4bqh2ItpgYDY@T!4dWD~?SFb8gZl0tA)Oy%CoJfyG2H?XszrP3VzL@M%B0(%?sYeUhPTPi+u9D{%6$=KjqVD zm7@w=WmO>QjY%-0-(9Us)Z1bdT zw0a$S81tPJnq>^JnmzvTWF6NHgPITS))7?5kPhR+t_N`a5uB znE0$!mBEJ|(mtb|$Pd=n^^dm^71Q5^cWxkCxnTY4`{JvHlw5oU`)|vty>As$?#56y zR$Veau;$hGlWi|m*u;#?rqJ<$;^jh==aEx9ky z_%QtUWb+?9GO6_BiDa{WZtrEBlRss$6H`{_aXXB$YeLA6rK;N3{);4cW!e7-l54G2AhhV}6&CP7E%tr(U2jfSgn-!W(f5Ytx%$mlpUeUY}%3oDK+n^ndO z=QpfoJ-ByEWLapy10wX)^q?QIUcIqbj>|`chqZGuTvN=)w`$yC&C z7h`gNe<(FH>Qsn<>salg3>t~YJIkY;e-kDG&3j__C*@N&wtJ^v>y~IYN@{d90Ua+_ zyLPYzU(81z-e!AIVCbR%Qk>~hQy8uaq8elK(ib{*KikXhktv48+}_mdcYZxwUIvy_ z+v=G>sPW}iyFhD5$~KR;{udGIR|{EhS)__|Rl04PJAg zE35uv8fZSI0Gy$7_(KN^fE30$nGrh<4mqG1Hy&m7M{gDQ`-Ned4r|kd04FDoH0HJz zI#~zx1+Kr(AlvOZOW?)2t3Bj9kt^^;3+{rp6`v+k~N{)%Cl9`v8K{IavwRjZx@ z4Zj^NjF*TZaxh$WN+$&zWRS?LQGxGMMPDOfBhCrZA8qXtL~qj?au&r@^&;^*jB*X= zldvpVt!n6CAWx{irReQV3E8c4PPCUCpm?;dzx8mvT}sW1nEK71cn)&msSIO>#tR6$ zC>#(MBKe&Z+Bst6d}`a(I!3@UnY#TASwS3W^O!Vet(OZLsjwpSZc-n&#@hJKVeVdE z)LqKbUZl?c2z}V1K_?vo9=I*D*Wp=Y27@y7mp{Qc?&0Ayo~y7!Z>KQ5r3G7ILrud%UmR#Axp4P zmJyAz4K>I=DZl%xLODz0bsQqw=g(u-Ip>zD415uh__*g5ujU`NEM0QTQIN*oh+r${ z%DDT~L`U!#jJbX(ymER0=gR%_uyn#>ZLzZ7#TdZ{t6hv-OyFQkZAHD>znJLoVTpw@WTf^B?Bi0`>4 zvyQ2r2b?bNPgY1p+m~`EP~Up`9Uuk;N~2~FgeATOR8#!&8*&Yb)+Gt&o=z@QD?tan{U|_%R74+iQ%29pTE3 zgm$H#!;XFNVw*Udr>_W7Y}0k(R>>$6j@kk<9)d)rVU~^1wqAFTgU4s}>GZwzuvki) z@+~C%Zpa~d?~UYQNkC<`jgTS$btN?DydO}D9_bb&*xQ}RbLLv#C9E>tQT6MZ$%IU@ zL1gFMy6Q@l?5k;4gi?U+<4Anfil_)aka`|<8RkOI7YG#dKogzy+ch)t7Zg9t;zU7$ zIvvl`LNa+{*Bh2 zy!pOu)1eE(8yK*Uu=hNC!M&S$+Fi2?w%3r_RyfRPW@6datM?Pw*A8%)jXixid*K7U zmgVsV&1ltJ<898xx=UHyYs^gtE_F&+#P9I^;LB=OtKN*zZI{T3rn$iF z_?OPQQk0S+QOVZBZ)6+)WKR)t6fE&4ypw>Uk36jsZQ+KV6rJ_wC?C7%*-h5C#DAn>-|9f0*nSnTC%zl_ zyuUvWQEe$`RE8SM9z4V3ftt=?aD-XLe%0MN)wRY-mp?Y5NjC0it;gN42>)AS{^$5` zO?-Q_XzO>Fx@#CDf)iWMP{Bm8qPPJR{{8l*brlb&=;Lw<3^*MqY$U}(+U^)x8t;K5 z2*yXNB%Hm%6qf`Lk%z4nTr=0=+LgSAwD+PC%X_~${mWM--XzDFU+7msl9JK%pFz3# z!~0-s=S$`iZtdlwXw|k4L?3XqHroGRe+}q^d7|F( zO7)xixr>zswj(7*D1ag0657>SKj>qpSc*1M$ok2_hIt3vhlIoI&VErnUMK(ZLfaPa zn;*F-%BCg!`To$D z|8TMaab)<(K@+&=*-%!B$W0dNx&e7Ocnb9B4}j57dZC4}gfge})W|E2e| ze!SUW+t4g>6}fIa5yTa){l4Temv2RBLxLJ%#7LcOJxGTBTI5BlQ(!F~*>p;vLp&yJ zzyZF>9^!3!C`ynZx}{?FeR#Afb>(Qg6BuVP9;E&?(TN^dJlr|6u^XMV1gk2j(TES6 z0dF;hZ~qVoJ+=U&rWjMrk!^ZRgs+t+OYgETstT-#29TG4DFHE$ze{@6vATmI zX2Xb0*BwglLdiK~EU~0ZnsBSqULp7zPfChe%SE;kL5Yq*h|VtZ!1AdeV&gSaKa|w% zTBI((%CY@`WooE)-)ua{3 z^OrtwqP3}QZm<}HV(V^apZ=s@p<_Q-8_z$`2Sh0&FR6VH6?`d#6*htdrJNLQB?&?- z1*!QGY4NudUX8n*fV1!{jo25p>YJj5R`Rw{m8OLv-@JtW*1x7cn(?g=e22X2GUj)^ zV-eg{@W;*^BX4)Ts^a@Z&mm#U;vpuZb4a<_zOzoailQD-0fsoEhH6?)4V9Js1M#AY zE0nef!*f1*G>0C5HwG+yo<$(%&UaCp8JM~Ah?^FPI+-?6sdE_^+WPlP<%vE`>L#+C zI|30G0dfVn9Q~{!IrF}oHR#I__+^rpuc3d(IZ)&jUQuYw-$369eZ-fQPgmIYsrJyV zdHzn_89V-ctg>y1+Vo6C(g%_Z5nCH?LtJo77*y_&l}cF7UgK7Vite`Bo>cROR%8U# z<(C#g+W-?}C2}kbMqi)3H}9(oO>-VC2p1!(qnQdp6mmClWF7WHTgy5XqtrT29E+~& zMppItf3B=W?H=q);*NIaj;)_PkGnl)nnXtPMCzr9@U_R0f}eTA?4&!sgM(<+D2L!{ zoFy(74m-9TxIG1K{~eiZntxD7M`WP(r>=6Jv~ZSQ5WGrMEJi)cMU=06ZIs-M=O3)L zE0?vknHAw{8IsXh5dOBtEmjrT!4s9@RUmOD@oBt1fu=j&H@sSoO-oga)RZ#f97*S+0NZz_cK&aCT3U_wR+R2irT- zRnZP;lx%kJ9sT{EW65WjCzlbLC8cWhqMzqPSi=Xp68l9c?}6@b!Ee4#NA$j&cUq+D z&{EkU3ej%4qA$tOB)W|nSVni(vhDd%u!4#4B1HV`5UIsSVaJ@O{H9utuO>RGN@f&b zVUevA>5#izUXV^~2@cLIX5Qp1<`GGq1(LM9fe<52WqSR-br`nvW&Mx!;1yw~t=3C| zVh@_UM2bUmd=apDSU)A_CFroDm4C~k4DW&*tUr~hZzHSGvFIi%CSBS0@K>lVAFSLU zn|-5Hd3IiD?{8-2A9Hiwo5QYdxpCl!(#4el<;4!jPK0$$(#8~2*JO}?}3=SFMh z&_&L-e>8{GDu1639C{qqC`ov^1=qIcuqz4^KL{U~kancrL3!)N)2RFqOI+TfE8 zVMzI0#6A%7sIpv#5w;Ra7Ej#WjVYFSv@_)&WL>xMYP7#H#K)NJ3P!Zdwa|GAqwVXp zPDNW@Sw^H_#$8)O~-)vhgn^?q{8RxPB` zRR(~)_&W6cvWRb$y-lUXyi2}H2imrj{EM1TO75kWa6r56Q#3}EM5+5-2L(hm$^4^! znQ;Ta1i8X{O8!=}qL6Dk>$Y&Gh(=cQzsCGjwyFYp^a-5a*XwgMOO7JzN7t*3{Ax>( zrL)x>z#+nlsTuAvM|U5wu%V(4VPX+7N3RCyds`oY1i*&+IE%47y4=!OTWcel#=$R+ zBqv6H3JcV^MWYEk0liSqhP@Zhb_8;M9z%d9^^}8fD;yww8yhxIo{SV}G(&7^I_X6W zjPV&L2_&h^hkmW~rB1_ZoS0NkRZsiXu$#2k$!vt{e|)xhy<#zIE)alh7dL-Q-{6w7 zVj*!bb^0f~?bN|a)xk^lD-SuaeMYjY@74-z@#b@bP3uUWec7y$C-RSJHRfHDP zhRyLAtHI~bIE`;Xm(Qqns>W3wBcP?H`)4iA0guQ?W;C?oGiUDKOZGot{o1|YG-xmn zoB_2~QqbCXY#jQfui44xuSEDh3j#e3_1(G6hmFtNZ2(BH`r2nNNh-j? zUHbw8IiCX#yWZa1EVS4zI^ZcWZgqDyU!VB5T-lBdl+j&8)mj7qdG}6pXwmCx%g~1q z_2R~E_3GjQe1%K7_m5WDZ^1;912n+V%lU>!{{d3H!;=ET#SBC~2ensU#ea=gE|5Gi z`|rKJ>Bgz8$(OYYlR|0%IT}-I-~{4EsY;6+N-Q}4YK2`6DD47kWLTWWQKw81OXP5K zSm7aQ$3vH^;4IfLFP~S^%sWpA)g=2+Xr@9E#6ZH&gzbXHGWi z@cNx^Y#_PyIoH7ug~!Bd^oE9A9e`hKV7yxP^(HC9bFSl!Yn7Cik?Oy@U*Qk@gR(jS zUBMFZDoz2{5SqJjKwc(7!0n?)VvN@?W6}E!p3@Woi0T~{kB<0l?Z3J?rz~*{)PAqo z`$znNKc%^NyI5F-D%_~^YJ4jdi_5LDMf;n~v<8OgH~W3xd`55E3;G675=A4FM^YUN zqKlS16!=sP@*t$u3gDkwmWZ-<<^k?i$t|}rqEDc<(6Adu47qSjuKT(;03!}cXYNpR zZh|wM!n?#P=eXhPL^jZim|FL+LiL@jC)S)(4u>#~&toauI{bDOS2_z_Ym3n#%#}+s z-p1sMY-Df-nX<}S?Ib|gaU=pc<7D|%k)14SZ)55NST9}|^7izBue;%R_~P95OOkE4Uwy?XU9at9C zbEidzK7KiPEeHa|7lRXWBkEM5n|$L2?(T&gh*;bGXx;qqu+6;5uENdO?l@un>q<%n z^`q9#W!<-JZn)Z)?faFKDEnnKq-bB?2TbDOv+wDKy0`3)Pp!kAYNlf1UEHIDWxEFO zJoXfLoLX17v)B+<_ziNxUkhU7t~6iK)oaLVub|80NI7QKoOO?HpNY;wAd7f(f(*37 z;_?t7zSz<=_Pi#Ry<7S;clL_?Jhua#6Q9IBGtU3*d#g~Vj(yverF+c*zBW01b^{h{ z9}2HFEDTDpW!(g)yIJH|ea;AXgpiZp$?vE}pP`E{`pL-KbGW(ubM=uD&!k?XNY!ib zJBc$$f-Xk|&xf9juDFHTpKGQrzl9UWBnf_8)riQ>p^wd{qWWO($x6az-8|b7GyPx$ zXO)0mtbFASv0ROUjh;TRJ()g+W2eW-OOaYwTfS)-PO^d|rq%n$w@pA#BDN9MxuH_e z_0%Jkj1m4Uu_@XYd^hIMIjh%J$|=IOvr$eIUV{+>Nd9ZcawRV1x z{Dgb$!9-4G)|b77IM?~N^T4D%Pn3OCh2ER`5lx&yb}F^Nx5`EGy7v<6UhyDZ=81hg z9X*hltWu?PiF3$2MSF>fKo|v77n=Lcdbr7fl&%xE7F12T95=B6QM7MiU;c}<6B^-c zz~v&VaS^(+32BRNV?3JxL=5M_pP`erL4(Q-b~mLGO%L0?YI;BA(|3VY7;Ckv49x3W z41B-%o3jiL5d*s6J0G?iR%#z8DVc@dVf$QRdWg z)Q5}an2(+E>HqTls3U4lCOOwg_Jh(ig%LwZs}YQLrv{0N!R$Z1uf56zzrNIEWff_X zP@A0KfrScuGv}}vTWVYKoh(_6BpX^4+D7#c&8gj+X|9$$cbX?6C2_2|pp-rq#gHc_ zNQd+*NYVjsly5$~uC<0|$Zfx8IOC=42a;2WTCW!#AMfBfBVMuIyBxIR&v|?`Zp}>I zyD;evgmGZr{#5T7PAJUQ}#kU~rkdakaeN0ds@DPBi>QXRBm2nT0|0>Y4H?TQK z2aTSsCl?QRW4{@B-RC!puZ~!$x8u|wZ+L7f%5*@M-c!x_;8iiO$gLs~Fg?mtzb+m} zFlyzRTQ}?s;1ygld{5hNIn#xj?auJ3-?X@&Ixz;xYq^aL9+PfP#0R^$IfQ9RO83Ss z3qq}b*he1722i{OT>92aR*`|zJf5;zg*C0W#|HuDr<(E~yQsSKu~+8K8?f8QORKAt z7VfgSit<8rEzZJ=rM!^4D`0s)1Hz@oH*#CZVy;oxq+c7JxbBgdPBHgx85B{(E?!^TQ#6JFGxQf5hz z?3uMs&B;C1#irGQe6VF%9jEaN3+&yNEW#WrY!)bQ59?c8a*k4c9-oaV|I@s84%new4(N8<{dOyv7@=PEE`; z_t4aWno}kf6a7vvY;4(8Ubj88+=I64SBECkN9+6SQC~K$#D`kXS)iXQnMd(!!`+)dxjmd*(c~G$VZ)o7sSd~-@^ot-G-H46_qX{RF;&d z9t?5roQnL5=ZwB8Uvh|igCB$Ey#tK2+_cmXS`ILir;RS%lik@165c)~RNl2d#}ARw zciAu_O=%74X>99gS%`_>&kYTRqGUOjyX#Gn~_@|og)Zs6%x=e0xhAfUQ+bL_6}O@ zIi4pW2(o_Y((U3EV!z%NL}8#dUn+3Jb4%^Ns#h8a-AN@Vu`S#jci0(aaI{zJ{!f)i z+>OiGL`PeRHVqVt*DWTB-?BWIR4L{CTd2cybz_2^)cc7_m!?bn0Alx`C*dyNsMcan0$nehiV-%Hf#*vjxZ_od zx^_}I?eDWg|NRcFS!7Q0`p3~R!=(|eJ#mEaEJ*9^tJAhmV+mn4r z=pFzm;U_)O-Wbd|$cSCz+oO+E4v`$sR<-$|cNylW>c8%t5y}4ZypU{tS+-3Z`E(sL zz^KG5(D@qrYLDvOe>RhGymW32((D4~EgEaj2ynZjE5EoUKAzp}z9@Mp>E$LhtNT=p zba*g((HmAWbpLl~Xk@_zHymEl(QwnewgWk%R!UPo_wrFcbWd2S;#pZ@Tlj4_U%Nab zg%COdgbRL$02!m!CSvv-n2Z-4wnidK-fOQM3RXS$fmPNj~Xo|gD>tVpO zS?XkacGdc}1!;xh;Y)JUA$ee*T#Yomm)hIA*b+_LawNad8VtJpFZmQ-qj^>S`Q;NB zX(V61uctU&cP4!P{kF(b&wl^B_I$8BIt{XADbMsP_cEi_Ik(n4R+U6p*y2+LGidCKtRy(0^o_f;yDC1?yq_5Vq_~0B6NHyYrKHcL|UieN>U@;VkOu4R;`Fr_? z3KO*(uX27LTUN%Ps;AwA@+z0PRSK}10Knawh#}?ynZ_L7>nfYqwgM(De?5*An)R8q z&H&ac>sV63vIK3odeu3%v3?h7u)iSyeE`Ci2qEfY@)X#H zY>#O^5~hq_Au3vwMsXumjkc_D5;4G3U5I@tIsY1lMNgB(J{KNRzv*k(_e1P??yQXi z6RWBk?_d7aS~xfrau=wD7W-}AHj3qdzgn{5##ikB6H!y-+iQ!9@zFepI5t?$$`t_A zSLZIJO%9XxLq2(V5ZJ}~RLArt0&B?$TaEgq)6U(kQ75DN>OMo*XTfn&5NnL}_-h*o z!GH2QiXQ|L5j{dQKxrR8K5Ox;I~*0qVV5=a2Suc7$rE5Lhz6yMeLX$%*}O+>>FB)F*$jM2o*h=8%IPO*uQJXsP;Hs8G5 zwEB{6oU0{$*bwJhG=GwK+>73Ra%vHerH|B4aq`Nca>AEBD?+DjkHU#|Usejo;JuIg zyPIkw!*RNxgSpDiyE7?_#A?GgGitBMekAY*vU1S=bnWVAb~QJ~HlWZ<6s`{}PM8SW z2(*w%&w)zu*B1sMTzA?9+joOEK1(l3Q*waB2tZfn3iR@eD*=5A@YkhD(X^pxd-2h; z8j=MB0D-e>D3^%FrsfQ#yk1~o6xv(Ar%Ts&iYLqmdd)^HWZVZP*vJ&dnwCi<*5JQh z@&u5_nMTv79@Dl~HzNavU{t9slAj2~62^A<%7bZc1@IqXGKQ?8F$%2qVcvy|*c{hR zjHL+B%W?`_wnwVAxc*FGB>+I%cSC;;i5CJ%Y(-{LviJXch_%z>jWr(!LdC-`be&-4 z@@tef<%3(2oz;XI(Zyk>9zY^$u{iNuowt>0L7~hzCkm8HOjHV8his=zL=b{MIYYp}LgHq}(KH{9*z zzW1vBQl2-qp%FUugg%mxxqaF3Ky%|)&u03vPBH***xul*^&lx> zgUNu%x1vyFf;yQSd#35En0~k4q=l4ZcAp0YTmy(h4IrsOKmS}JZWG(b=lO{mtQ4Js z8X=J54H>J+9MY|m~U9|^25+S)910i`$ zKLf7Jn4zma-DGEl4naiRVEZMNT3cLL{HPC-*G*2%G5svqXmip_P#~9%h8mTb+rL#8 zkY>HouGJ}U@w&R4@88cV7Iqsr+Pi9i?Q6U5Pu5b1?IUhj_gCY7NM}lTRz~)zI4D8} z>v{B8rqF$2_4NtWn=WQF_|9ZD2+6m8yw@}4tQ{7pekQ+W9~L|#lOrR0<< z<-Kg#h*B_61iwly3uWihdq0%mX6dDnpPSg3yU)`z;0zL!wDu7Mrt zUb}B&zolmtB>?>x%WSZ-o=lNF0|!0gC;1)&W&8`o8_PtQJM(&CVF1z>9_9#P>-mvn ztd(4rfoR@a<880LH-0KfqWP43VD%#tuhc7`yb#N;LV!gAAAA$9e88L~Ax5pS?eebO zuj(hCpG#9+KOgty9AL2KuL2sq7xMeS86KFLUyBL^gyb=ixMx~@mLBYah-!2H)}H^Z zb>LU59JCE<;!37c`G&dt-}!_e59FGECJ@h{%h_k z0F7V`9ti2e6IEfS{S}OP{t3YD4Om7FP-V*Daucul}a{{mjpBWuJpLFc%*s4)>+`&abkRU?m@H^=$fX8;%0@k;^1Q7Z!>_dUn1faNm zfN2DnT-{on`4B56mSZs=AV)!sz?Dac-?Na%K*HFo=fI5@=Z`W7PtAV6n)^_IvAtC5 zOfk9*I0;buf9ZPv<=F`^$_$2G*5E<=vm|dIl@GXo>NemFKrbRpu?Lvkwu@SOqfdak zqb2ONbAqk{_d!ubz}f|+Z9MZ@&|xIiH38CvG+>MoCyr{Umk->XxnK8}WBLBDDc?n! z1%u+XD8)Ri-%M%9$|O9u2N)`rS84yn3Ll*>I!3LVqEfD?1t67wkXmV#n^_^p7WR8qSjSTIvi-2W*D;Bb;-}Fii?_&P_s4tXROJ5J1F`G72 z=VcSR@m+fHow(PHj`;+jj(BGv?@cG|?m~z^v2sGOBopnR z4t5xDA~FSOTi5FBI?sR1-@?Lz5e|H@smU*7><_7e|P>Gb{_I{yF< zhdeG&*ASTZn8)cBQ%v@xp;nT%z*vFOZm~vU)L`&bQbBOVRM5Z_%Mrg=PzH?M(RJ(f z|0~2^{^JyoU)0d{EfL@$fa@|s{tr;e$O8rI|M&sEqXA~!8I)sr|D)^lf7^)vU;p`~ zo&$7iB!e{cUT1Yt+ItGV=ysj5-is3ux;TjHbX1G7e6G~Ark0L*ELo?y?dlYu z@>cv|F6$w+Fp`7Uf?XIm>UiXTGUIZVA8pE$rY?C$IBf%yy$Ko z_K@4#YxRpnZ7w?8WV9Q+m6~hddaf~3RR+i!YKm}-OYTdDcSf8ZRR+m z*6VD?TJYEQXk3Q2T&gQ#EVJtSUNPUXO=vyk!hX@SBWly<25YDk-+@0!xFG}kRJ7`$ zCYM(MCFdt&belwUaw2Lp%(niHHb5?ljEX&&huVJ4jHrPQ!U<@2Py0A41M13Sjs=%N zTN~X2M~l98+qov~^JYt|-j%aw%_zg4v83{YV16fvS09BXXOB*TcTW`u)c*XUR_Y>v z^?p#rKf9T8slm1>PpP2->2>$%VBr51vuj9tmSOBZh#LU@2&HKhnP+NxxGNhL?WHE_ zjQ=`i!jJtd;#{)V)hM6q^N?q}hY%Jv+*muyFy+Bve96kAO_7yToFin}9kKc~H^LY5 zR-p@Bnd}}DW)$;ZI?E3zxuLt7z5WRUlz}YeGxsq5A_EW+oyt9hZhsrQY^_oB%kk~> zXF2oe*Vbds$XdVy06rD)AfUm&E)B-Nm1paom9!^}J)PJ4S8xo+V^n23p0SfSAfXw* z6(X@4zy54$uB0(xpv>lh{vY1u*G2N(o<2>j4E86zl`aNU$$ewr3cOT4EbRqQR}3t& zG^L}f91fy;8NOnLZegMgU{6fJ*(OhlT~rrOeNXeS@9G{Z<(al9Rc)nh@hY9Ow5QA_ zBbD$a6nEE3vE^(U_l`)m-YW%B=mnY=RM~V$MznFx)HDK#5x! z;*-Se?admgOROo}TS$#l)0xVya(ugByMH2Hhak5`%7HP`Gnn!|M053HoSQOW#srY?5ru9r1sax*tFx)1uc zS`F5-;EZY@S+Q~2rEkmCUf8xtgtFUxb(=!(tt+Gj1NIfT&i@5?JU2g=E#EruUlM0T zP5y5xfZy*l8>t=6_bxN4XlJ4mW*7gJ^Z0gx?|^#Ku)X@7iKV=o^-@xy;*;i|KS_6v zy(_miahE;>@&IlrAO(`Ue6!LXAxchB)A(i`|0Uk^JJh1_%y(6j2t+u1} z`tg*t#J%!uf{rZPzl;}+{@b-8Z}71m3ri0v8#M6cf}71>OVyh_yEWaI*{@$uSvWJ3 zB)@HbU345fjHYHhuM^X@uh2bwxrrIbln*8`)Ut0XNH&i&mT>eXU(l8eB2P*L!@n$> zys@c&i(2eRkES~07~6JQ3Wvd5!|d-i`@i{l<0yvS&w6I}bBX^K#N!p^F)WO# zn&{Oc1(B;-ji-E3LdWEJCTV@GC$>EoU97Z&-}SbbV1$*Aky*oafi+rZdA#-71)ZM3 zbbZT7nfvS{tL(eX-R=Dumk@uACAZghR?H?9ErwqZkM?$vaO%J#{gJ4ODfax(Itz?t zx8?^+o2FA<61xeK6!0%#{MJl^MHLUKTSvRmD2B1Eo%EwPliR6psP zMcgXBn+2(8tO`w-3+<}D6pVy~0-qgewGR+l(!X#3m`TnAE71)g`F*G?0AKt`0jtrNcl*ZG|0{|~qwL<#;J!`SK_3l z0-UjPmD!w6fCwPktWxhEC`ZK@rg^cthu;)=q4op8Frp}IMv$j{8gTZlhE!kFW*6%A zgzmVQ6D%^vbd26##dXff>vFt(TZtBDcfy;dE^XhHnrE7ox*Uy7mtkW{$dq}W-RLkx z#D#3QA0`hXNcrODwA`LC(789v%Ngs~q?PErWLf&xj_KwlbO{ zS7skYv(myg35jkz{S<&tS4#|%fzNY^wM5RBEdc`ztV7vvDbVKlbF}oDpPIjpsqNeO3DXb zEZs|y>(MME5BiIJqbcd;iQW!=DxXTL_&@DjX;hO}8n#sw#~y5~yCj?el@dhUSY$CM zVrVOygAfRtf*P{ znEQpcGe76_oSym9??-aamoGW@ySeYZ&-;AO^O6l(H4zmFlgkQ@_=h2is7i-OUqfV$ zXUQvr98+ha#6hggHg!&a)|qCHykD4>h_V_t7p4>)8td5kfphfZH(g3omhF)D#$H^Hiy^N5G!QNMP+@D3=@XKY8NTB#o z^Y9Z(ae}l~(b9hf`eCQSVja)~P24>wM}m&jTRKhDBUxPaD3z3$iB*rX{^h{r?t2}< zO47FSco&TF-zVtM;cMtYj4;e6uC~ZrI*5j$?VoV;YYTw}FrzS)o z5sOUxHf8r_UC>}*>|?O~>>MEw>Yz)XLLye#H2v&~ zh1aTI-Hy*k^;0@fEGw4Tr&68QvBuPbG01rSZt*Dp%K3lNKoNk2q3oXxit&w`i~nZl z51=sJcsbG|Q+9SwohNB8>>+>FSh77#ckeN)(2}+ZsbNBHD&K7x=wRe2Nxm;K+vNM? z`66imetD5^y!FJKwHP)-hwMd;_nfCD_`~&A@_QI-$15unAD(W4(*~($X9&8XGcko$ zcH>J$b9sz%t28-~+q7)!5GnTZ8w7!xV6<|!e7K|$f|)wg)&U;HDyuEkwl55g3m^w$ zK-i_+S<`5TJ4oF_>+vFRL2%sz`uQQqR@sMj*h-UR2#WS)@L}I)#Zd5}kzl{Qfs@DXEG3jzscvDWRs@~f_(MfX z*mP4uQfd9Qv7EkWC&ZAt*Ay*0;ppp^qR4gzY;qL*WQ=zldLeMCF_D1v=a%Ay9|f>S zC3Jd{MZU$erOMXE2ABBsSuYm%%MC(gE)*pp2cRHhysS0GP?QXy>`%2tMwx_Zgnr#I zu7&({x)Z4{hVQ)EsCQS#c zHb^UW^b(Tbbx{C-%_+aUDf3+4F8`^3C|vN;BX@_=z`BfLuH1LoM>In;YX3p@4~J%a z6U-B2%fc|@_`Aq~bLCcswE8yptLCTVs=xm1#;z@~zq=gYAry7Oy4)Y(vH%hj&)=>S z>@1Z~!~=vC*KPf@a<`?o2qib-7Y4(rgn=eTqRH_qan*)a67c}OiQ20CP zyk~mnpg&ZdX-yLbT6-9$)44F@wPa45@MBAd7AgK_N?=hOGt&p7HIMAL|8fB)J9Bli z07>pZ)oh98Cw7Uh9-&^ojBUHXzrDTBd`I=zgiOHksu`sB2fUe4FGY_jJufRQt>8nk z>$+Z!zSx^ zS`d&E$Hnf5Nb(3?qd3KX8k+wgru=YVKb`cN)18?bcSgD&k=fs_6((KlcuVb!w3iXC zr#NZeid&c2etmg2N9K0sAC&F*MB@r_JJN}Y1q{Nbtjw9S{}?nbLR*N1cYFA)I_Xyv zCbs5Eko|+HB(~OCd>2zU<`kqzP!87`4*Mwjh=<7IMykz??CAC-X+BYY$OcJkBOB*Z zLO9-eSdN>SYI35Ym_qGj+ovfWpmRSQ1xpFy{lqI;_u(KV@iw1nUX(G%x&lfqOzRgh zZ;ngis+AQ`oTLsxLUB-^t3lT67+!h$5<#9D&SBU*8_>VN?04jAGG*QfbMwqLYmun` z#ZE6rLsoIy-{0xuJ;rgnVuQ^P*kg^+?+Ykc?h141<9a#>)_;BeA7Jz6$v;QwHdxC{Q_`;9Y@Lo{JDHgI4sF0Ci zDJU0)+`*=Lp!{6!>C2FniXbe(phEMrA(ZfMq~_3kO-a3tm%g=)SL9`m4sLyU>Gx)B zy&wMDzm1=3-BbMcpE7~(IqrXd(n^_?!`A;#E3K3VA44h%A4dP{PbKN4obJE7(z?g4 z{25gKpGMJ14_+m{{%`-v6pwlz{kx~=!Uy8F)54y`rk$mm7o2;KCP6L_58qV zK~hul8^1Y|UVod(9gX+Y?m@ z-6-eX>2E_in$lEpk&*Y`_m=QZl$pR`Q%&wRA^~&XJ?KWgFE&ea@5RyyS-w!C&uKR6 zh_FT1JI)O9Hq26}hq^QStdMhJTjcqyWoC%r?ZYmeM6PF?;i5y?;t4YRH=bMLedztb zo0vRd*Chqtz(e0*{c2M=k?VcU3yG9Lea;e06#1rH!Y#{Nr>!rrTY!0U6+{ zHs!H9{R@4*oPK-5ZJ~IiCP=PaN>Ry?BcCR-u5?# zDIpFdm#TOi6}PTsZ!L|9pCG;Fl*i^MdkLCB^sOcA=K69&Vj!Kr0p&#B5lNcR-C>a+ zo%V4@=4>I!;ap>4;9{Y^4q}TF!xD^FdqMpV{H-Ss?_1r z0w$Z(Zu_bqX)H*a;T*a+U@=Jbu@ZHQ{=Gcky_%lz&9Ty+qP82QIC`qM(jD7!Cz!j5 z?A5W(agJgPb=R?10DT?ZR%NygF@u=MOoMYmo_yd)fm~>YU*KUMr^V>TM3speeO*F= z?DgaK!#4T;*4ts(Hb35eWy=p#hWilk$N7{4EI4d_M-@>eg~Cu=M$4r6Tsx4$+N zqQ*sikaJ+&JvQC#-f z%aiu)UeRCfjvE3l?tN*SUteC2Jnr*qn6u&}o+nQ!&9=FVLj>vW0H$evm@8Hbd}fPQ zV)jpAi39!}S7sTtet>wKU}FNcR2In#$nkYQ`r*DlOgEjI-#uEa`UwTU`jZWOq^<@# z+H*s`u9NH|z3pD>c>w)*BtIN2^nMs>HBflUb-q`h3BSG=c(?r4qeajk)Wi2^ZD_>(p%Q!tEhxB8bR)eSUv4M7=ba`(i)@Ug z6=Vet0I^%X(Nn9^5vY{dU7=rFh+5%y>>TpC_U!3gsmaH&VmUCjkm6rY#5oB@mA3kJ z4|xqUNxy$5k_Y?X`Yk`=vT{j!85oc4;&?N85f~tDa+Ed6KRe#tW?hKW^P_d(sR!}aiqN*W&JNY;8T+u)ldAE}ty_UM zH5l7RU2xliFm}^(C0$Jg20#--!Z;gzYAHghp}yvY3ngtU9k~U=nZlg%up_~zrWng5 zZSWPA_VA#@jde*%li+PZIghado;kUn-Ej<`P)RmKZ?XI1eZ6Lg&X?;mt3|8_Gsxp` zWnvrc-tEszTQ5Dm*2;65YNGV)eMCGBV$I9mAE)yQi4Iy`qidu3{G!7dusv$SXg{wm zb{u~{Dc2*WwuoiSrRHp$n^W8?zfRZu^h>vJIgX|PShF2QG6sBSHbLo_X52&^X~%{c zVykN>0D7M2auh2wN~M#z(tG4~f8w32xq#WgBVzcFOFXyR%PMl5SwAI*zAtl4$)p$0 zIZmAD>*V!lSMG+*Tq`3Sy=H}zjl~h(VqQNHVR`}lc8!h^7k}Cb`7Iv?iUX#`DSL6Y z8rDDLHBI!ak}M6Uy@8POjVxjo79B9_ER3gJ3EFOu9QFvOI809vbNZh#kws>n5v0Ix zDhp+wHr_kKU_m8CAonr~zMpI4x{U$mF>**rIn@0M7A^JeC|BxwC0Vl&RV2sYGUmb= zzfe!7+_PIw)_&zh#SEGg>xjgZCDwCT^k0H!lb_5%<*|;B=BOezlh4$x?S6IxyT`dx zrxo^69Jwh^&bM6L)^y!S;5z5*g-3wtVFE1aY>5?k=sS2$CSA=_{DFXhun-m}zV^WO z33IubQ@ipSh_A`a-**b6NyFKv6DRJr2DaNmcwYJVgHzszlFI<4@uM;Df?pkOr-(T5 zbT0Qx!OA}a!^}w3JnCqWiSj7#15%Y9cr^R_wibIC8DgOj_RVCRXVrxJc(S(!=Sk-4 zTT37^>xJb{1qaNd*fI>+F)zHQ8`nLQ=4Fpwp_VG?v(>CF)FMl{|UuK;?(iExReo0ffCcX05f=u(utEcmrD}er3y4|ErvYnZ7Os@&(mpp*J@-sskj=Pq4+^oX? zMO^4BRZ(=p8|fYzh7aZGd%WLrG^lTrOGSLP?wam~dg>(AX4KzyNb48Vhywhee{7WW z+88hfg0i(Cg~0^yvKdzRJTwPm2Uq+Tc82dp6a}p>GWstOF;(O=YI#@NjRVzSr;ei6wLv0v2^j0t%JWLKf>M|AvEYdEWx>!L&EgrF&FLj-A;IXa}i#xhb(cx zry@Qc|AK4w0h-u<1?AYJeq*|2+Fi0aE)*QSOFvKR#x!bS=4ARP$&f0h2~D!=w~sYi za+WVz+5+#wJXRR5yEr4Z#j(wXG&@dE*k_IIW336X+y;jY;CXAKL_+d&Z{fkJ#oUsv)?Ma-zqp zu0-DJv;ZEBA&KhUK=r+Dsk7epo3!kRZ8$dBtg!0c(%*OoPV_M5mz~x(#zr^1W_FF3 z-;&-htbcY{Y!}qn|2HNuJYYBY`bPA|rCm7#G@uc>11o(PorE;K8(ddcmtmCtpSA=! zpn=M_U|9fXtcd>yfv|JC=)e5-_VwTWlt09#^P9T3c1Rt+la8$zeXpkS#lO)?yXihF zlHJa!06;%jBKE%rao9!MNL?@g>{_7jbN>sUtY8CvM&)U_?pKGG|L!1V_JLW5!|C3f zdcu8abVsx{$aV@^O#H7eANJ?<0Rg62z}1P~-;*+E_H@)~a5B27F1&{~UGHcg`mYm? zhS~_({sfH`$ld*?H9Ww7tlCWeuhx^tQ3XXD$bieQP1hIOLqO<+e)w-(H2}n<&Kj~; z@BaC=%2?WcOkMmu64<>%#?C*iiqRPp#=b)d+W%Ta%{w6OepRkJ^!Kfrf4}{8?az$! zDJif4D-yTyu(GPEu&~YW@uz=%rNxsg=_O{9;~Ek8c=3T+;?+(q!r^5~NNavXQo~jeIPIU1aXGExwg6geuHHO_7@I8X?=P= z?!zd8VZ0oPA%{QWIVLdf3iC|UJ8Hq2T7nYORu>F`<4wMpYQ z*}IXltA4Ur-9aK8$sEx54M&3-Zv?=H!c3B$$Dceift{Jp@jL7rk(zUcUk&DZ&D*{I zvtxI&H}Wa+CRrM@vk};mX{od0F->6kw_!F^)#IT_dh_1*fqB(iI1|6F1Rx>X{cj%I zzXM)947@%sl~_&D*>TpbG~85Lrz656l~^IGMaW!c0E>F^zU6rw{U2jajhu~~+?%I~ zKhTE*yFXxu{oDcXDuEb$f6^Yf{-{68ubt$k>g>k4rLK<01bZa9Pso4$?@hnR09}qK zF;5jd#3rY*OclrycXr|&1;!VpmF2-Sr4p9chI;YJKrXZ?T;o5%lSRD8woDJ^gDsrVo5rH)a=Y!1;w30(WC5K~m~0~&J6n$g%E zAof#la$DW;X%y-uJXA?!5;PCC-bU z?1)!1mV^Yp(I9SS!2^kszpvUoLojdTW&11#V5fT?x56AhurcyU{p#XG9oW}e zs>kBu{&;({O=ARZ0248@YBiKu1Ayx>sFmtADQeOpL6kH1T~gDrzV zz%ZG9QeO-|%$D~KDypee(0Y)TA!Qs(ix8O(z`Y$%NCn(sCbD=g$E%M$T_Iu7J>vCU z2zeuUYXV5s20-S{>*?Kom%9ztLs_D~A06*DAcsd+F*ZMttRu3;5yrU8QDz}L)XisB zT~?u_{)2yZXTJ8H0#D;?-=p9SOEJ6>eyyESf0y{`1Chyd&q#RFJAM{B+(|jFg%OY?{x`J-a>1_gNNm^8@G7qY%=s zMO>>1k9&V~AC+}V!vO9;Y!|q@T|;-L3un1~8a#L3TrwC2F7UCCLqNX!KTfCNK74y| zmjunvJ+buXXI6waUzWCmNXt56xr4-lsLEHAtzkiZgcThU>D!5cvw9gIRfE#FD*(9* z_N>4@j~(+CDPAq)358w^yN|3DJ2Z+-6y*1FRe~hl*VKGteNk2HiMUeo)~Bili!iiU zn`?i(jq$g)sg5_<;?BdM-hR;{_ygX6jwZewB}OYbDV9M}z7u>|gcKXhP1)eW?J!=9 zvWnN`erUi&JU~`-p5asX%5Np7e7($xq2V8XCsp?Mwc4R?Tq)Nsj{e&LP`QdWzXMn% z5<@R`OZMm;5;Z&msz;XTOq|c~tg2&Qxc`*Ho$~EDHF)zTi%M<9C&RKfWhnB_IkEU` zqS=&zQz-3_|H9O5xET#UZC9vly1S zbnxX-d;E;(qh{g$nlV;D4{q(4hUCz9vczph|Kkzeg=YmIvcQ8j+?mIFr7u3P{?=m! zVdQ)K*8ej-YhdjH=ge_=R#&xR(pgHWDK!!unA89R#XumM&o8vZZ_W-x-);Qd@F_~z z?_G|U9464oCkq6LN=c18+uMz>A(5)fh6sRp~)WhosGg6J|TcBInitgAw;mH z{lfCY0D}24RqGHQguMSkLlViV%W|A}#Bb*Qa8lYvS#7{D~195dimxyhVI#@wsiCWJy(>jY0NM0-Au88kS3JLT9U`>ZrB zX>#ysa5H@t`U$?`gF5eTEY_zy4x*M5ZN5 z)SvtwLP&f^E?I0{#{HHNWpHjW+pgj6WsTRqnH)_1&L5D4>fpT&-DE?Jsa|=FJ#hBM z*|b+qj>9`e1_m-{&Mck-hF^w3Gta#WmQ8vgF#URSwD^-nw7)U4dd-OI$$0T%bRU z(OywjM*K>c6)do5h6&Q;ZN3|7htnrTsA~yuu9Z~peQ-F4fq0-S&W2SzhcMK8DT%ii z+AJ}_Bw!lET>KE5zLQHqyk@wcZ;Hy2yz4RPqEhRdGz+}sV_tCIQ59YaL=WNskv}u_ z-zqEJQDQu|4Slitrjk_~^pMO8p#HKs^Ss;NOee$CR>%zv<*3_&8%br(Kw0rmjmaVv zzYH8KcI-;C@ImP){003(AfltPAoHL516QCYy-t7d0$0Rc&6sly$uhi2bRUBHSvWq3 zg9i_Z%&4I&*w2J|$cw6j6f(>@xH>8K@&jndLW|&vj9xxFH9b47nCJV?-ZLWJHc1u| znDalI0jMoUeE!# z?af%s15%WjG>8Ww{4Zq{&J>=O4v+mNn{H!-5JAABhpM^}@6CIC=f}HNo zG$fo4RX$}{fxU>7A*-RN$yo33YW6dwmvAeR&JU$w*~s*N~Gyt8%p<(P!W3cJZ>3#b2;It5;Zt!c-z2G~l|$gXqTf zo@VBbeXNnMlKn9b)2GLEGdajDC2rWZXPmgQ&D~vgoI$U(n7?HqgwVlQs~)dqg~^KB zF^`Pn_@79l`FSv^LR5De8B#RpkJR?*3Ci};^? zmVfN7$g@dPJ?WNi;Lgl?eG$TwmcVE;^+PiQp&Mp^rEy`3e4ofZfoBi#f~L^kNf04( zHYDS!_FIw9a+angm3j4ndZ}JDgH#_&FrKC0Y#_HrK=EMFsNKzx%=Oo)$G~$*b8J5# zQ10mzwf)(8-$vQPS}jj%Dld|DC!~>+HHG_6^otj;%YjyMc$EBuB)3@3URDZMzHFza zJ!-PZY6pP*!`@=l4T8L9tt4pVUGP)5?{gDHh_9>B*z5bO;$yHERS#LTA9+Sy6NeQm zL*OWN_57MRBw|m>6|$h4}^x-s|COIe5nHLTD(z^?|>)KN2*Nr5&c%KYcZ+n%(J7 z;hMvC8tox&71X;CSe=+4hE-l`b%GS(L|dt(@V$P>g3Wy{wcXV`4`H+pr)X zcSmBL%2QhfcX7GGK-}$*9HRoFAKQ7IX>gEy_~o@3V!)|SwUCCpBm@8On(X$rcsa1M z@*>u&`j@8I{ki1xP)jlV_3*_{ku#-~oXg5Q0aus*r0+b5V==$ze%)f$cnpN5ly@kf zlUw?iYte9Mo(;Bt!HePPoTZicj{ZGS^pS9AAI)!mwF1;G z4QVGsWnWPK9npY=M$HOuS}@x<^xZJBGZel$UtQ4pC9KcFdO-4=I0TpSDTlM!x(z^M zJiqfX)K5k6;&k6RG${)tB-lhnIK0Ce!h=a?(mUJwz@*|S{!*znncSZ=1E(kn5}~P6 z*={^YF+{OZ*wjZbBn;!-GP(i2b!`c_`N;R_Tv9bMhg?sD})r zZgVn0{Jo>jENEArJV`9nX3XO|c)bvPo*QacqD3+}#$bl+#}#y4>5pz$)!bq5mYK-a z6+=ST37UNJmwFXoa{oZknifY2FEgkE#59`xot~q2G?49qg&@<~Cz`hVHBlEJ%yPVd zN~?0hP!a`Ja!&FO5=biJHX6$^8}PE*Ak8tBYovS=MhK51!~%SF4x**(i5wER>6udO z)E>oWV1q!y6$JnWE0%vnK8(MBuui=n7g-gA=e_YUa@xXdl}ESxj2?TBcpJPGeiMvC zCeVaCI#>zf9#}Q-G2<1>avGVm>i+FCJ&6tV&T&+lbAv_agEb^G4`>5p}Nv8Tj2J~3ShAEZFc^{_ zSJ6o0rBjuI2Zi(mJ9fU8lHAORsk7r@>Rd;$>MoXnnuoJ!SBk63}oFe2=$1`IC`Vm?-!Qb z_+nadTk*>{bmc!hO=E&`_a4qad#G>_&j~)9UlKRJqda#{PE0LG;qe2KQ;eypm%PAI zy4@w(NBx#oI{;+Noi@{Qb4E*%Hh~wJ!?q{!(04wwPA@Q7g3Q%WW(mYP`nx8J^yn%E z@`9;=v{xlD)uQs`YpcK34Xu3NbJL|iAY+qGC6gsDy+mU7D$v0qeOsoKr+)#f!y&^d z8|D@BZoXigUu@!qVm<>$i^Uv2vj7mFJyA8u`xVW@lSV${DktE%1GaBc*C(GPh5_>* zjbA-A^A~=BRDneyoyaXax(gjxU*tpBBmbAu#}370?-hzNYwha{XLnMY&dBJu#8c%A zEX#+mIy~)IiHQElV?TUM0Sy%#sisLp%_qW=O`5!nMGzNN|NIzIzq%M zsy5rkETlpW0{-0>B>S2Lxunni$d`|`S6iHnIKxl1Rp^m0{P$?Qzc~K!9q0$WjBnfm zAF1VtRW)S7e(u%u6VH6i71LyVrU<}~nrH8@^~rZT(=BvY!a>}7_W?&RLCgofcR^hQ ztuXGSs42AS6TA90bGm4e0w-UF-sPrVVavfvi?yJF1hJWVa5CIjQg!}#2Arf%Qf?3f z7Mb{yskuqM=8y`oxb{T*7RNz&Ci1|zYbjL>6ylJP4?aw81c(#o1}uFvPclila{JSJ zoB(rG)!^8B;>aCRF&?*rbjQbD$#pHX=92wUmx7Hm(jU`I<-Uy>eSc+%xjjmpzB9@}aY9roEo!00FQfg44;FaDa z?!61yycrrxSMIh({yZuvX4^B!)43AAq8wMcR%becRfWW2(ofR0hph7)h86Kkq2#B7 zmqE}ZF*V;qvz*WWs7O@WSb_LC?ICoBztobW7y5ws#*=7zjww7i_`5^F(q7}hl|X@2 z$Z9hq*#MoGvMEMF@n?rO9Z*BBmx#_{mDERKe9h0W&GE8jdol7_J42Kull{YJ1AWR{ zPb82H+R5yJ*QU&!g)zm+LQSUaET_BtgjPuqQ36*X*W{>rL%mKg2OEUCj0kpu7rm5} zNjg$0_YSEz+j~dOt*w`sI%L*f z@xG&%P>woAO{O1Jfx^<)&l-w+OfE%6BGFCZU@9zvCEQ6kFAbqPD^jRle9UT_;b9@Q z;l-u7mwp<3Mz)&EK41i}bxz^5x`qceq^jaFiA{IJi5O|*1k`~3d#f`#+Q&6RgC=|7 zkkE0`DKu}M!JT~I&>*@}Z2O4n*N;FlZ&3^}ty=`pJ!|l@LTvr{Q!`UuH7QnWAi{+v z`<5EE_s6!ZHEu&O?j*R8P+AtF{ebySS4*wTyI2K(&M=Y_tH|+}OMAEKSF?9QF84$$ z#9b#E49E`XamN{RN8rL$sEBrG2wA|LWbEZuNC=TDo-H2Bv47)t{M!1X#VfhI0ScD_%~Iq& z*+g@$sEp^bLp_Qn-V|qFvvo5-F?cV&H=03k_wlD1qt?mb*(jMD^FH;am|tFiy^--` zO=J^j;&U+UxSr4`MIS1TMgq1FfGei?JbYSQAfcNcu%<&$xHuMCIp(@(`YO zc%8G}0_A7!PGK>XA`qc=9;#amnrUi`55@b0j;~k9F61GznscROZj^|G5|;BrKBI>l z64_Av!`e-aS`N@zzYu-);wxwtsIr{(QSd{M%vSrk+ z-?EoejBN8IWPhErXUduhhZkNl#lfB4Ls5<}wR$?+fyynStPU(9W-J_zb8tcUxl6J# zz7wSzX7XDOrde*uq?njHn1!DuQ3;&A-$C@kc2=z259n37INYV+;5akXvnOhuwV`T* zo$oa{*nBZle85fIWp_ewop3iSJi})xEUlnJ5@wK~a1cApPVpyRGfgK=jrKjcG?{SH zS7?qw=TB!b9)RTr11k&(a(KlNAX}X9VfCb0{?@2duYjc%!MbW^SKsj1pfggK(Lt>b zC)SbPGNML}G_yURwb6!45Qt^^CEuAIJE?!*CxB5FeLLPuunDCAvK?JZbXw`BRvFxM z^F9zKhk)*$=VUdXb-%uupZc(E{!6u`aWe7AR^aDDZ2ivHF#~d@`kr)*A=rct z=QcJvRe*sw?8>hnJg>+=8EG5{zR)FE1;|NcoVT*E%H^s>FrjWqV~TkT4TsB;Y!31~ z`+XOs<#|xo6&bS_pXqDzQk~;s0{n)+ak_QGs zY+Cnd<#cgM^|1p6l5{yLp~~ra_cB>Rc-A8wvqF*jqoZM%$>r{ihQtzv?SM|@pPHIN+a4(SO#<1T@IT!}GJ}C;)+=eCm|ix9 zU9(J%J=KiQY?Y>)(FGmaw!!agx4}c-Vd~@{GNBiQi^IG7MacF-wJ_{T-VpvaDwCS{ zd*yE~n8_U*`Dfc)a`55ciKtZGa`KH11)gW#E2qa?4wk96aaPC~ycq5k{&J7zL)zm` zx_N9qa6gMfjT?sRz!jJiCWJqpI*7j84Egp5^HcNk6CY5vIO>{WM5*HZ%yfgm4v!B? z5eC!KP{xja4$r`q|!EuB*@!bg|pa)GgVo za`;r$0+uubTz#vhhGvc*nfOqlww+()>A7U*!OBdOcsRvG8~RV8u*pN`ya(PGQE>GN z$@PSA=Y30T4SRuZ)W{OydM}j+;>P5rbmtI953S@4wO4N}_>gN`L0PjWt986R6Qjf^ zRSz}O+=cK@IC-MA6UkzH0zIs@~lqHhAX$=@k2OZNL4JE zG+jzfkiQ>{m2rWA92Hr-T&YOoe^>V=2XVwE#_|><0-@p>xG>t#TouP7oux_kyM9jl zPCP}fE-67C&CR?@MoK0sF}wzul2eclq=OwtRx+E6Ds50sV9zq2aZ`Qkz$ExRk%SA*f{K`+z1$NT$H^PL!g%^8e|UAWUO!J+W?}^; zLgi>(D__LEWNWUpXzRQ2A!S=~A^I?MLmS0xv~>q{#<=z!Dw6$u<4xj#w!aUKD@~Z9 zK+mi?8Mz1-XOrfZE07uadcX2{)iaFdS5c9&EvJThNn!N@qczNI-iB5Cd`8agwht?%)-H+0LXvN zB2!7uQj6L4s$$y+1vW)LW6X{OC?A2!o7 zHAMcwfocl@uA+)>4AZ3)TT2m&w(D%+;P1^#5~o~C{QkjpMO)c>z z=HcR&xdfkM25<}I`XxK=h53^0#Q&tlJNFw8xnEL4A$c&-(Mo{0=1zMHd$HM$^}0fd zHiykqD1&51ButqvO%LxYk5>ogL#>bso@GTXnt^Nqh{YJ)ut)m6NPwBF5E!u*S}<_H zFc(2=z3J^gwD3$M30+P1n)k7JcgPQXl-YTqw8x(p*!1d-Tpo~omjGy6hK#}H7+s}VPw4L#HW#und&8IscrExj;W5vmO2>j#H^Fnl9AlK6MJN| zNY$tp-^^NE4dfeV;4IHv@*!C(^49yTo|c@_7kU$+)}Ci8%- z$N&5O%Zj=t=cx={ZnB#fxTmCGD%Q^hbl8nwte4GS8*2XEjDHw@(Jbq!=l+HFGHool5Jg#JL&HX^$6T})HX`; zpN5s@Ua&quL}GKo6%V@KX1RU<`)gBUI);<}h*bs*zBZ?F_H}RM_Rn~(WC1~P6j+8J zT>qM2h1gttiQ_b)Q5iNuO=eHG|Kb8~b)0R>V^Xw?@L0M?QU9m-7_5(NSP6@U<_%v8 zbBMI#b1{Oh!fYF_aQ9#@iqq~M!ZMx4WLMt0;P1hy;+qvFa2XDgjU&h>(gBCrf3jhg z#NBaIx$Zk8xAd8ubx6fd#Gt|>hX!H4_hwo_?4b_t!>e=yDC@~H3}gC)8P* z;P3#@g4WrNo%#iM*OY6Oq`C`9m-;l3sXenGTz_^UEeI)Km#EBP=f(_9ojIMn_^}0zt7)W&cC9b}4 z2v~f>1PImnzrXu=!rACF_hdVUv|KxW;OTu=ZFFXos4c4zHK>)<%}wTxsvwVfb>kw8 z+??FNYV#pAeWhaP*=)x&u{~EsVWUkRHp;U-3}5R5%{GH4OSkApj^hSTa; zU9o)A!d9nENE?rdb*KOyrKMb$W12{W{$&v+7zz=MNqP@@#PZ~U{KCt7>|U*y>BS3X z1E|!T;C;((BXH{_h%B0rXkr(~^rY{s?ZiIE0D`dZe*TZVCF#+vb8Je%^OeAE3zHL1 zE4YTLKOMX$>kFwz=NH#4t&?v}A&r8;g>0s}V9# zF8LOtAO4M`X(atW$g73edy$mzmGA!AP@-2O`A+P$)Q)3nfxs5jCPjUzlSkB(c9kpHI` zgn$*e;LfmfYi_H-kfsQlu-kwyCNbuOE#L>)1Jq^L zIaKGdY489vFia1r$vl`RJ#%3+Fc_#$3;9&k7eRgIqCljm*2QURtt$gQUeX~(20>pX zD*qhQlywXT=Y_rlf9Kk)Zll?wn+%?fM4C~>2Fm%pSW+Eoc-D=$kD#ruU`Qte_b=%Q zkq=-TF`PL)xQ$uPDXGleamAv{4MEI)bFI-uJB|1+L}DdY?w@VsP|D8#@7ySVl3h`m zH^~96k>}y$H_1#;>)w}d)=n3G0|Y4f4|vRGQc4;er4oI$(sva01(S%P+MPb}bHfoxW$?Fnx{XL2bwDmIuTkl^AjDI8- zR~^8A`H*BotAkB4P8$EE0V6QI*Gd4rZOd~F-X1SoW(KI%K6e+75YqNcVVjHp3aT(U zd;e@tAo%!ofJV-s-jd?tvVWK9s2BoSOw0d6?Nb$lgdHXY0G)>A`f%=T4%r$IX$PeI zYXZ^Ob09jL(!V5cfF#HfkaLMa>}DHl^nbkn1|%{6**V~gq*nkkObG&j4-{jM@Wia1RE{HryPf13fkF(9r=-UOo1DX3Mu(jq4^%gUI-rpOw0ywn3vl{)?@Ujk^Q z0KHeT0D!g#gn+7=dO2lvK(JT=7CZt}Ug!TkS>Ta%1CXNwSpHwT3<%7ofi(ltv}P&` za_fJu;xGw-4?OrB4^`xfUYF~NJya1o2g;3x0F9U_z(ZN5@#q)I$ph-ubUsmodI$Wi zlgI!M#3}lqgLDnh42ln=@eUmr(u=G#Je8hYZo>+mZG35D|g4x^?T zT|eo!cz@7u`HT@!KwM?m)b4|^-@)MR7MaeN2?TI! z6&7*ne6K4rYEGk*K`R~|A!ILqNt)CG)mo+`$o5y2es2#*c&t=PfGORR7V0>9`!n(N zL+!doE@Nm4mvLbjmtK8rnMqY8ouqXv)g>V&Bt1duu)1jheWZq3>I#*Z)+%v0w-~DF z$+dj+N~)$6Yp+4!jh#;83qkjl4_yv30R{GYwJwX#Za$={wYFN06e|9Gwd$SbzLx+3 zw(CWy@aLj#E2<@%yXt@%I?wFU>nBqw{3gSUMzEHM2RMf<<^rvip1(6ol#^!$A&VVf zUu5TRAmMOgmA4`6TR?Hyp&#_34VO9Nd7P8j*Y&8gAd2PW0|*(0GB)jpEf+BAnM6-|6*b_?ZIUGTuuKi$KaoQz0e3WtytE`m~M97-RBvqRb@6nsgA5W+Lem1 zlLML`a8cGIoxw))=v8mdt8<>Q`hic@y&2-W!d8A7#Pm=Gu`I1I-ApI{$jq(uUn0J{ zKN6buzT1E8vJSvuDtPrqS*Ot!-(4r!caRdHDZXqCvoJQO8)|{==m^_S#bZ)7BFm2E zg_o_r3M1ljot>IL)k>gJga@zN(8*M7eFrgF1b`^0?Nz1W)c?I!Ap{u1+=pO}&QX_K*e)oCl~{VTEa;OybpC2FL^syPqXHD?j4gi2eD$BW*=k24C^ zY0RkG;sI-5;(wA__Npw&2gs0LeJ&CNnP(f-T31xe zt}u$nzQJB{{G|)#?g7)Z8JTfj&F=sZmvKuNos@T?QTPYusYaBMJ0Nnwa)&MtEq54; zawBhqg*fJ%grxFuh@0t))L-U5*D!M8Yg~DvQ>fr73{*bBAFWALtnL1U{ZN+tQepS2 zGClLhl5lN6WXS;Y#QZ@YuPx=T060l+%?nKS#VG7`N%x^Uq_nAz%t<^x6G3f#yi$Z%w!` z_T9Cbqii`s!P^=S5JU*T+sP5v$;-5y(h5k~Mk;|-576zmj+xh@s(h@oUNV={{>Ae| zv61?tpaS)K{WLH&eL7#j_hi?NDrKW;f6tIoUi+7tgRe%Ax*+($=(hT~o7HWZg?bY3P?2ogW_!8tnU(n^#0gKQX;?y0am^i{){RJ`d38^8lxH zcjM^&a;W(2O|Sjrpk^_Rn(3?WXLNlV8MX53FR$tczSG_PLR-#46?*lUqOPOFOWM?E zqb5PlxT6Gmvh`ckt~fz3MhtDLyP5Gyp5pG+)y&LI&6}Yp&OFnAn=3DPG)HQBPrN8l zhn!C*<`n#m@nl~-^)5Y8gez6NWY_Gk@89x8Ri13%+{Gn1CgscloV5>rXfsd++nwXx zpJ|P1mff0wVct_H`0{Gi!F{S&^iXZwdAhC)@F0gkrEkV|W%DqPLH%Gt|LZeM!^Rg{ z!iy)wo_+BqH=?$aW0;}y5e*HZHaP)rMsb%(WoKs?-yM9kUQ0^&8Xgqr3|K1_Q})q9 zVq~_q)uRkR$2;gu;M4zE>yuiaw$%{k)r&HguDc=Y?I*~L!=60L>2J+F&IHPlzaS2e zW{yV4eRD91SpvgEnXl|ic+o!@J-yAw8Z?=KMH9=s%G2JobniY#duLj>2yXMuY zuoXPIzBIVv(N21yf(Jcgl~`z%-x(f$%=d>oCj!m1fmhnnG{Pn$?tjrshe1~>x2${qy(RVF~DMkoEgF7WY$S(3+m(|ShMN%p=VOjcV^ zK%gpcd+xW282=L+K#i}^akz1~+MgMDpP|e@8Hae&%<2MOnipWKjT=<1S0xt7=mw#? z+EQuWt_U;%j_i%;x5*W8@hM{1VNCw@R|8}%g9B9I9U>RYeli(z*Jav(IfE^lF_ZI> z7Vow^xa`g9tFa1oaeKOTWGN1D9|5g-oS#+r-1vu~$)gaGwzH$gB+RoXIcWz$_#ymT03Bi&T6uU=@Lp4`zjRV z@y{G)OI#O?4Ymd+T&T0!`2nFw$-7OV3qb|>JX}McL8GHouGbf~!dCs3@Gl(U!1MS$ z?&z2Piy3`6lZ@%#_A{L(W2p-e!^xt5B1)w50Q58+hjhyZ9^T3JggBtaJ|Q#HLTjhl^$ap2Nhwv>;;?n}0tnh~fknW@Cg<2ECDK}-BWu}56Q(qq zLaXtjwQ2rY%ly@KgnoCH@Q?{d@+-3y2yWZpJ|Z6wBHaT7zlL1%$*Ny;^gooHw{gPN zRd}}{)BEbl`2+iJ$wUXAe9(QRne5pC!>-1WsU~2T?FweSpYmTCda5n3MH43mke72g z@28te5NxEE>a3cyg6557E{5#2XLI=!+BDwchi#OhL=~0LL*gIhtJ$C5XMcW2!31?T zx%qZJh;X0r?G6BKC!*fdA)<&gk(vScv&&hIftj@k`0Tr_$u?&SWeb3qT>B|cs%pZX zBrGQe#W(Tn=Qv+wJ53w3UlXxhH@+Fz-pOsa+|{T|N99Z=o?lsVx_q@V2>50JZ8>m_ zCv2`Amg=b69QR6~vD6n73;6wBr_eG>zu7u2|Fy4F?h_9o079Jr<_?p;R|crwDOtiV z%~~}GnAs(HCO?mZ^Q(u^OC|5gxvqBdKbSHbsQtgxYU=L&z@I6!$~{d=yjNhOhiQ_C z1b-Xj0iQaiT`@2h&l_Haf9pJ0CV0D$IraI2)WQ^$^0C+A#s1JzoNT59x2(AH*Bicp z7FCV2xk00=;vY&16!BB*GiMenG83!mmxEWW6lL?8v0|$O>B2x+(huZ2xA1_giwxo6 zGP4u`2kw~;JT?FPqb~|v1uNU!X|A7CjYdDI3_T_8g?_@Ys?HPUvoZr z$(Mqd6_ulNj7!m4nWXuX5f~sxv?45%&EI(Zeyn83C?L6)^5SF{(9PcnWluEgke>DS zDP&!U0Dk)Smh*AlYfnW2SF`Lop1}Qvy7l=7%^v_A8ybnlQdj7_kwez z%^$|ju#C6zXC4lWPqrUQy-YGqxt_>zb9x!lX!eW7+W39uq%XnYBT0Xv^&O3-2f_|# zgBb&rcBW2tv%R-R-WGCR@?wN-sGM+Kyxgt&+@lEQm;DubUgftmUXB&H@nb}H`PZr+ zMWcB4wsXa-U7FLi=`tGk38Ii?&0|rOB}VR4u#G&Uv^bIZ0`DpL`d3#Zv)m{yttn+*#eMi z+&ovG$sDyx5Oi!N!owk^g5sHLyvD5c5WTWFWAP7l+mBwC!h)9`oL1!d>bgkxqwCp&V^RqU zg&=pQiz18p*{O!M8(C4dV%rkd)_k3066e0{I&-L>yX{3`cET#g+~%r`Y-_ zng@e;wAgj#(h}=|YoJjOmu&WX@wZkPQevBCI!5bV2i{vQE303LG0(c%Ubc5aG3e%1 zIsQ;vy$9~6vQOo|UJ)Pq2lXx&twro5RJj0@9Uefcl1 zzZ%_k;FshkWYkdXLbO;S;vPCaGMd=sSn2kt>|-41&9h#Z zQR4kj)qBsWXEjx0-)eGq#N}dv9tMpQTcVvyw{PF9ImB_(YpCA{O|CdT$W&O-8WK#| z@4h579I|Cp;^B(7;63(wEUSMQzB{bGKf4!Q_KSMFkI6_13S~Jlkgda##;%8*uD081$@yX(;o-FNp1j|uY6D%?Ct+N&1=;;@G5rsy&t%i% z+E6voqfv(~;Wa_zA3gFS%AB-^OK6G7Xl44?8uKEX^V6du%PE>G25arI@jfPm_Gx%XI}o#`rtvaVcDX{psu_b;b0E@DG+ z=BbmA65eUC%~4PI@kQ33M&Y_oEi4w4MYI<5EgE&^uev4_w}A$B+`9A?vLFnQpY2$+r=nr-GjL_Dev0rGVJE8pPtF) z_8TWxveyl!+FnX3zhAlb#h?Zs9bZ(bHL@bua&Yy@HF8w3C`OEo)jaERL|TdrXxzOz zGncl0Zit09o^HkSq}??L8F`xAZEr}f_;*6SN4M<3h;li@l?c=mtZ%wgou<*oEh9YH9vbjL4(ide6&us>`jWw-r(n4mwW zb@Fk5=Sg2)&YO&otvS!?m#xsD^CUC#AYgPvQB|^bDfEkPzqYY6ib;QB7GM>*s`A(U zIfgcLhs$@d4GMLf{DqEJ#PNozEKHp}9N=j~p=V*ByAxuAE0>2aXz;<^h5AbV^>D8L z?&nx&k5!1flYkZ9fMDs05AW=yij=)LwgmucxE8pa1mG|dW+T8VV^wPNRR{sNtjs=z z3dC)GCb}Sv@yuL152uQbcxZ?m-3pES{d+^jS1dl;#n%hy@va=N*`t+vgpYml853{J zpV(iG)U#lx@wxk(POR$P(TfbCysm#u3)8mM@^l|jy?CFv`Eoj5wJ*}sQck#TWFx@u z9W33Yf4AoTR)|~pC;(sV63$^gcSdmkLmE6%W0^#G(#2}rkq69AHV*NsamJsDzeTNGL>=A4eKoh8|K4mn8B z;MMa_JiI&>dh;_t^qTxqP}gU?45u+QRvo5wJ8FGG<{2feU94RsA;zPxca>&|0f<_nfROww+xl521RHaXdmHqZ zl->2`FX4#CCiCOhmp5!p8|us(w46g~r#d31(+$>swz9RsbqR$*E`d@_vR z7_Qxgq{HSm`&t~wPJltq!!N%LXyytwRAt5jJg~Kwj9Zd|)hfHtHtuvEaJ&0P09cRQ z;wPJTci+pMS#QZW69*l;vg_CP-)#e>LjRUj*}RFk>Wb@{}+9F`5 zs&R$9@#A@w=T;b7V|z=di4@17)dO!{yQSp;e=+8&Czayoc9-P1*XPm+sE09UX4BSY zWt$24sB?=)4T0apc%D#6z+Bq^gUbu7XQl&_0D!xHy&TZ=0;~bpbOBuc{o~!S(R8tz za-tmRjWlF4a9|~ZeXoj(rFBis8$R?s7#m#!hv-IR_f*4q6lP07meXtdlx($9ivu~V zZ-1&!C@zkD^L|JCxQ#DmS5dlB=LhQz~Bb&h}oU5{vh6s!TB{?RYtMo?6;? zJQY_NBD68uIk=!p8-$kDCDV{9C6s@Vo32RuG<6nkDeooI)FtiJ-KZ03kJ}ez;!S#n z18foJcuwfpWLS9{&0q9JZywipS}L?A9PYX%^o={u)9@LlP2GgxMBm9#iN&AYh&tbg z#FbRf#Ig^U(|(u839buVb#(lK(0T(`jJs9H7C*31iAV>SwHJW=1%3>%HfnrRd-xdeopt$BeI1s--ElG zFCU&J+M-_thM}$5Jo~xcKPT4W-eC#pnX~kk3>44=od_OvA2KdIEF=ofU)5}DdWws? zTOSE(%mpyRcp8hss{5Vqh57dgsv+FKvFh`(swpLuL|W(~f*54JL5Yi@{4+xbX^qn^~%pikvLSBfuU{?PThP`4Rg zchkIeFH_2KK~c*y66r-Jz~^b!B`YZ1O#6~x$-`a4ykH|Np7SsDMZeud%ia3rT(z~C zRf5onCc-xI))L+C`s{_+Wj%+r(Mwf(n%JdT6Hgl}R^dn#X`}e{h68HH`o_Y@*Z&v0UXy#sxl4J@>6h zrD1~+MLD8gop*Ev@-)LL0^5=9#z0Xu8esu<4Y)Xn($b65soX0rqv>%43-8@+>}NN& zl}j8{eodun$4>U^a;#a~cf`M&aNJ8l6iF-JdFP$o2ib-sJxtuaz0W!V0&6V$norrM z^;GZ1@Zc_mPZPw0j=W1|rG>7QrV9t!`-K5bU{`}TN2KD9~~nmnpKur zBC)6Gtz!VV;^@?|zn|sDlHzy$un07cCXrM*4@XhD_tm_;AC^yjD7D-t)A>Fj zuSs)o)l`k%Rrr=c$8XtAj>YfTbLIQK*RQ7dv-PPA3G}^h@35egMF5MzW12evx1->9 zbTvuqro9~7&YPtltb>(PY@@asZb~Es1trQ0VWgmmHU8dLyzKfutTJN;4#b`#Yauzt z?oX>Zz=lq`dyR}Q@=(#>?TNE$(6|W;EaV+}>DmJF^|GeBa?)*HO=De~vZjgqE5@tp z_ac*X4~`Rl4e zb35{m8rs_W&PAc;j%oEd^g71ztO6ADlNfA{e$3|m@ddjLwLz)-(XsiRZL!f)QCKwo zTTk?e>c?&=S#ABmo*7b4@ZC*8_eF6_H$1-7*dbN7n3Y`_d46oOcMXJFOqImuizD>IuC&XpVA}wsF37 zE%Cpk8Bhtm8RquJ^Ne&`=MmS(QL1 z0iP)Vh0YuDUhEQBkmdn=C@%me=^ZRj9wsAJqgH7I%#5?XzCn>F9sEOg#Z48cB}kT(EL*+`9Nj%5)8y* zL{iKb9|$tfa0kTvI*0c%ht2Pij-F&UR4)cfdCAb)g(n^{Br;Yf3ym>iPvoYFf?M%1 z(^b~qDe(`mam9wbiN%R`*P_PB+oVLuBlhlf5p8ck2oG{V>0K&1xTQLv8zZAOXRzyK zNZ2`mX1KbHo$_wQUlt+2`TAxk6t>2Gkm-+C;k9)A6x}V8H2mE=l$>$14-ZWXA}QML zHf$Y+%JL(f<|$rigaj<@?`_>$8Ft}rF|e~DZpnY@&GIPV;7#l1UG)@FIWu2p&sN8h zmoD8lLb2w3=z!dbYdD}HOZJNHf}7L#Ztl6LYP~I<3}o{uSVi`6AwTo!9?JtR(k;KUj+;NvjpQq4npJ6u6Geyef9zbpIXz*$ z^q#X1BUPxp5c7c9QLoX{pyU+i;O7piZVfzXiN!CUQUv@qxglkyeh^7{dc6?#<@h<2 zOvlUrt^2v{<IDB7HM8 zv5U?L8(!s7_F{O-X^m$s8M~SQCpG49c3L^ss}}7jCe6VHchTSU&F_p;X)N70&|K@6 zAMfUe#z|W#T7L60`dLg&r_sf`(b$MsT{DZ*#4d{$VnM6@AAds$iw?^{Gnx2q{C^Ds z{Z9#7Ntn%2w(<#EHz-9o|1NvTuFMjTw7x1}=3RJkk^A-Iy6oeKpf5fF)D2+EBPKgV zWJi!YOOAi8*$llte)(A9ZqLkoMB6{-3_gi$V-&E=`(EiZ#v|ckvGX$pqptGLarC;? z{Iu=j8|4z;*eWwT-WW^7jK?<$^%Bs@Nn^up)Gk_MjT(ZFCPV zyI~r~h;FUh%vxN-;2FmcsBP3D0aGl@F_`&ga2&KavWTIjxhEf5i(Ai>*+Ts71 zECWKCNkAealDhvFlp^q&9Sx5B41E?D5kK9ZMcn`Z_E3gT4Nx66=Q$a0>F*IWx);;T ze(+_#fEFZRLIc{_Wore$(V>@1WFei`K=#|{2ieRp1_VS za~^_%C9vFL)8k&7gX4|WeX;Z4iF6oH#cXkc6RaX-33g*E%vN0f=n{n(zKj=Zx}J9; z*ToY)FV}tVD*E0forHw6k{F11TrpDYw}P0>8?REHWT~k=dF*fqN%e03__D)szrgTW zaD$mCDUHVmVk%6rYt?D0VpN>biN7`mAf9z?Ss!-IYp<`F7gwW=@E9H$Ddy{epCu?t zbUnQ1^u+-bYL@VJYV*Za0LnA>CT?(6e5Q1}D1kMD;)$KDE|^O&WxU?sxMYreWB%~o z_2^;wEyCN{JxcQv+iemyLfbbcs+4@d)q z_T`c5d>w}Q+6qa#VZv3LA6Ak=zGFMbv&V>3JxSXyRc8qt5_;P?J9>151{vg)J)fJS zzw{8cdGYS&ti!<#c=VE2_0MCD;z+se%uMr8=m~At&(?DR_tc1^@`cp;b7C%zI`nu4 zN}%ylGk0uc{-o!G1_vlC{W4pbu*>gaq|9O(BUkgw&;LBjNU`u#zI3(Ie1FOYUMJ`t z{&+>i1)fY@xml`K{-PPgH}}jFAVMhLhx@VEo5=3#rc-+MOd+SN%fmvIM}QWBE5C7 znW5h56D*5CW0&api$g4f&}i!SUx=9>i?1#1CR+zi3yC#me_ZYb=7%c9{)o%1JcT4W zSp4%P>#Dc6rhB-CN{wP0i=Wkt|F#iX0ete_u zuA3Lp^j#zheBJ7{_ASgXBKZP0r;{a33l%oq2kioc+U2!ett1a#b~A!a_^N~Xz&2op z5KyShBzT%VXKlw^yEIZFx zH#pWY5V7Yjtz%ZRW6cL+B>wk;v$BcJOVg}-9#N0%vu7qcr@}iM5|HKsX^Ok_ZpLbg z6S+{6%_km_e{g~wckhh>qeOf8``0_(10pn6Z zvNau`BH5R*l6#0Hr{dnwVmPyK?ePG`M6XIOwKKGY=A#VY`XdmYmb19%Wo#@coJVtX zb?(9(VAr%Mgg2C?04s@9pS89FYX&dta?jB;}{Jiqlg z(W$t&lD*@dqoVm3S>Tez4Jh`R*aCi`tz)L&C96jq)Wvat0Z#c4<7v0uefUzN{ed%Y z%dL8!w{rdS)2jerJG!`IwKIJA$YNfj-oEysmk7u_B0vfn+?_5-;lEn2|L+W(Mc<vH?Mv8HE z5X7V4$dfO~Csr@1OMr^c-Oc=5uPTFK5^X^=GE&m}#rEtKGNqM*Z88BM85i*AOY@;G z+IpS~LY_8ZnEf&5xVyFsMVJ=AX#Hi*SriHPF(88x`UUCwMXxM@jfnTpnajN}y zh4-w9KVD1 zUFG56`rq`94~uE41ag(cpVz2CN0nbF9XARHD-Rt1rpWL<=u0vtB|=jKV&Ba@b0s`Y zxQeQNK}$^-#3zEk7AS!UJ@^p8N7)zqC4-iNcTDQF#TXBPTu4zY^-P`t&3ujOT3--- z!RG&B|I-N{o{zbg+# zTB88%>gE6j7Eg&j{P-j!EZb0$B6CUU?@K~U2}m^xxIt$M(MxHpMFojGyes*8Cgl)_ zCX>Uz^&IN)%|I4(jJ@_Z*5kvb7Qfe);N?R{j3vuOKb6*)dRjmC--(pgjDJ#W7ntUAiirFOO20ZC7%m^NHZxD`XW4O7rE1kd@;vBDPv-8gwJm z&WWDr9(uH0b_eZ`>|JSL|2CRM#DY82$MB2GU2m$ZZP8ZlNrBGaFttmHaL>YouL@E? zGwJ%4X6|-RSJn+a<#=#RNU>dd({VX)znG)c`&V@b8wyxaG)e17rx*=()o8Cf*H50ZgJSU;ZIaom(XXN=C4!OV4)Y}+nkBoFJzx5fryN|Ll2{c z6Zr0p^t6KNd~L#Px>hL1^xePJ$G&*ruZ;)jCRfR)DNQgz(XOJ~PFyUO1M2KNxMp+*Y0PnlAaA~r< zG~TQz$QN&yGXRmQnel}^JJ#<*7nDEw5reQuP|Yd|-*_(D{Ph6B@GbA{uRgI9hNHG; z`0`OQI(xDUDJjL*!JCB2mitck#A&}P)}cB&NvUg%_=E{sX(g?F7ME%O`CK6pep@Ol zLPNOIS{FQku|^<9s846=Sni2S>UFNMAYw)4LVa~p<8h;tV&>)SoKnkA2%dMM);*R> z53A}O(lno6t6)sk@8<7vO!~I|wPQ)9q5HM3S1@$F;T+&BZf@fKLJEPp3;iR9uzeQB ze=BTDCZ73N>G4wLtJ)uca{V1~tW+&4?UZqCaPE@1ted>c{+UJtulLm_c_Pjmv?UZJJ9&SY3=4HcWlN7D; z<3ZQ(HPTpElhkPXea~l&<@<9i;JAeSuGWT_V&wX2Ofu6C9}os#fR2Qt29UrW3No;K zlfM>n>U}jCp(Z0zbL6m7G@@WFRY#MV1{3Th0lUu3ZQjf4oL(~4WhCpa+aiBz%$5VJ z53asG=ni%UePbE$_3mFK57veXWK5hnTy2d=Q*h_7^~^OFeO%!k7}B~$vm$0 z6t1b@pg!fDw2U!a%o&2N`Ja)8O|-n1Z*!b}Z1-GmK|bZTTYBkEZP^*x+U`yzqc!&> zXE9KRnbD7>hvdgEuSNqlqIz~h8ZW1wc42&QDl8Fdzb#@KpymcEM@_+q5vwm)%mYy)$eOrpg{C z$G*#U_@5kYuQn$y;e(Qclqm{S|%3*{O? z$4rbaNUg!*4ZEsWzvzr7&tvJU9Py9-MyPZMI~6oF^zn_{yMr*EM{o(Ht(XtM5s30= zynQu^%;Qzj-nV_`=iQybqEAEYE}|1l{-0tfOG9%T3P(8h)O=}ki`ayvGO}S4fV(_< z4OONp9AT{(dp6BI&a?J55n}bUFj|MaI5un2rZdYb`SD9t(tfGLAGuL@5#o?f1d-Q~ z_jD#0D)Ut`2_*zoz5Dv}`rmZAR2WFbtxJcQ3VX2s#aw)-EEApUCyj0*CYH$MYWd}o zhiQO7Bgt&`f{@7~D!vIk#mrv9&iaG$k}s@lEpghJ0aFgjPP?UQiM5u`3L2(p@KDC| zR~7LKk7Zi@{WmNjxC8t4;{fVh{b)5kZ_C)%NC6bVP2TD4dXinb7R3T~mWpmhgDbC* z6#F6_J*^{1?}e4A<$$yY6R$Xv71a4g9_5*&cHWx*4^%kgFS$zZb#}Wz<7g>e-Gyyn{GMvh z)--6vu~%mY#xYAa}O)303^ z&%)guC*zM*y}d%PFkld-HjuLh`2W_Ntx7vA*Efbq_0_q$-qT3!Piln zokQQfJaJKaRna~4a|;QJI{sU{PDg?gS+^&Y-Em%8t*ed$cBq0zbA=#wymG+EYQ&o- zk|R0~ca$A2*X#uJC3m#$2T9R&PJNn?Ty`RntU4F0+NNF)!?NJ&s9r36_M?wkX6%>*y$1Ys~4t)Yx;hXPPpWuS{ z@85>zMTKQ$ZKV2R25-Cx#SHcV5xv2POQkpdW=HVm%MS@i>Q4!~7#99_nIQCUB7?h1 z9Om9UQ4#(DWg4R^~^N}60;`UbH~doZR3%6eK{Axg>k z)RrDzsTuvZ3K=IT(2-#`?tV8ks%3+f0L%xUjCJqau7J7WIV;#xjgs)JwoZ?szluS(yg6x6x5J#lZ& ziEw{VPy4%ks*Vi5RKT@ee9&E`^g)gs+82-O#D(N$goZ^TEu+QD^&%Nyu6J)e%aAVT z+k4B#mf>}hcH-0Q!$LuC)H;qte-rNApp)!#FuR=qJASTYhIU{L!(lUK@(&caeMS{M zLQcllT&<@A7Cc?3+lpx`iTud<6lS`lGNv}5E!|4>2yyKmS#v`Z3S1zZ5gkaodNJo) znIM_*e%KD>7yp}0feXs5T4@tI3naCp2x{)u0vjKva9j;h9*0>JD8~fw(92Yt@3Wze zOvO=!=c8_;}>rG}o&s`Nj%@

$thr|Q+9VR=-zz>AU4G9q+=#B*T+kUb zpGgfvI9XOs2)C;t5m*qd=5>p3AC@cc!dC2L-9P7-yEIt`HGoYku$-qfsl0HFfA!Ropt(r%S|(_?UNjyZ%!rBv?z4suN9^F0_Z@Q! ziku*~`bg3-4ymx0&C<6+w;z<7DAo^sHy2X7HMXdj$yB@!1bN z8pMX2$0~OP<(7BPYNDTg4?4th-tk-2GGB#l9WwMHrRucOtXaTm(SApcz>+?!Ou1OR zS{R6&ogZ8$uW!zI5fwk2@7#WhI?XjX0{U2+%*P(SiS6CKAHOQdT#X-5K`E~lNwK)gLjb0*iG~W*3( zFw~X32;N1e-o}UGy;P^ID?eU5m~&@gR8<`c*mPgpf6>R==lJ(k4R=Fmps<*x5&I(B zMT+Gs7Cc^2(q^$tiaAmc`Y1O6Q`eQGHvc8_vgMf3t|9|yp!SupUi$BLGI~$=o$Q7) z&CV_Jv@jS8l+T)Zy8irRed@u&>%=pfKGG;V3Jns0#8Ka~OmwAyoNPXA$#xSSvfc8% zaD;$2Q@Q(H_2lb(yrr}T*UuiJ=4~G!1AeneP@e*U6PkzG;)!irosNZ$5x4J=VaE=e zBhiM%juSaD?3U*J%?Yd^b7VCKUgRIPw(Wd;1LWp*5`z>KMpFibyW-P0w_k+@8)3ad z;e$`%nc5%s5fO}2a(%I3_-e%(&AM+csGc^gGmedTzz_~^(Mv8Jk=&1BeMdz<#rvbu zp{hO9Wd{=~r#ImT>qjeQTWU>QDb$nj>JcVa9eL$MJE=vt$hp#a-Dp95tnzwi`ua-1 zNeJp=@dikKADB#hSfqS;f#o!YUVb%E9uFWJ#&?7e5kW|hIstFP6Z#0u4VEuS_Z(yT z(0)7-&~u-F$S+hZqNFl(Z5@MZ_L!L60kOItVSHD5w1PbPr?84Ltr-HJH!qOXY_HyX z)+2Fz6k(%G934M8LP794wpK#6k{MI`-YXN>qQRhFmg|nQP_=ab05fuIgr_MngZjOQ zWvThwPe_>dSs4VWlR&x-;l{-V<_VFBY|C-hIh&{TWI8H+F^fBFiC0<>H$OC(el@@I zrWKZ3fD)0^AGOJGom~+agTOCiX2llhSxc*qBL)2jkj(s81{SBpE8mk*4H)6f*sv!k-sg|XN4s#F%Xh1cH26)SX zOZslga`NlObI%Umz_Ft9-A6*NYFnjtqh}A3sNlD?n=9LPIQ-p~wbJj|{N8ZNf1ep% z*SX7T6Ip7h{cUW~=@wB%S$|egBGm#B-s+E`cxdtp2>hV?)^h?IhVtf&(t^ka*ELL; z=11*f{10%vZmPuzxAmfUKe2S$?bvH*{mSIXn2|?p>_{RJj%fUCMN6)wi}B-L#e@d_ zh!3~!El$!D6ciqH`>i@xb??Kj(iQUhn^yyd9%BZnec$VWD9dsQBb9o_<}BoEVFr?=)UDh{bDJ_p-uphnZ>d?JYu z2bL$YQen{Hd*d5<>|I>fox`9#x&3PXta4r0!BJMQ?v3+Tc3fYf1x(8GmAPaqkC^Qy+2*Cv8~W^c+YGa@F#I%ubQW z0;1?6g+wF)8|t*a&1=bi=rCa4_h+@qSchz+{d0QJ_&$FBjwTi7u-1HJy1asrO6@pU znS+ZWVOH=W*3y(ha%p_M&L<43dS&@9qE5Y$R2}ghvHxD%;RaS=FZviW64h)!wPFRmC_x#(@lFXM?}n zgl-P5KLZy$TLJgW6s4gVb@A%|NnD7-;9;6=6E#vTHqnh6`fZ^aO_9kQ@30XP=<3r% z5wi$Nd4mL9xC*XJ%+{=A%=pnq=YS-vvP*Fv)jf8-JJZ6q>cz~P_QD5~fKNcQ#+-&M zyj>*^9n7D#n|Bd{0)x)JiIT%1hDhc^u3YX~TJ}egk@DOsip3Njg9hTuSST2LhUS7; zls7jjq;rwsKL|mTEq}N{8ifZB(WYeg$nb7+ysz3VD1nkZ)mClJ?AV@u)e!HLGZqi_ z$KD2l%uxy$3tOTw7Jz298sFDWD`)w>tOw~v?et(i02scHJRefeThqF|hr*cV!_PW* zZgcMZwVrgbZ@Cz5SI9uvqN7o@dz7M>ZNIy%TDEuRdiQFiyMTC61qu4w$-IO=i7L{*u>bH3h52L@_N&rf^_SSXJt5y?$-X{{%z^#@u z?JZYGzwxU0JDbtN6WI)39+#u=_Ugvg=NXg{+Q~)CMJUPoSS^6cDVnjvC3v5&Tx= z=$_M=6IsnwJ70J%O1J)2ZS?T=X)X6ii8+<~DUL;@8Rz9(!w5sW(L-;w~A;J}GwB=gl!6b=d#6DI`X223%YX88Fg-_)Y2=3rr$j_%JefZ5fb zTi*g^eE+|8OF-U5v(-)p@cE2%q zSxiPTd|0X6@f0Ic(HN!+nR(J7edW&j!-QpeBP}c!D>4HbQ99|X?2!hwzKBtK+4FJ> z(BCnGA%QQWC@d%yc>3e7QY867SfmmQQKHX{?w&JMZ)#21uT+}NHC;e`F7r6xSL&N) z1pDe|NP;G09($)O5@FVJg^jNSE+wl?pYpxJq?d?AvYLZFGKx}MoGh@j%(0lNTAf5> z=iIQSYUM5e>PDFCHuzhTs13vQKnNUD*SvDK3-u_H z!w^<{NKc}@C^tDDx%-JI7o}COHoHco<3WNb`Hn@NE^0S#FcbYYsiBRVqq!zxTZyVi zwNF{9@5zf-UrxMgCzCf#TA`NEQV;t5jxW!3((`$Uw<2z2f z{9p}#u`B2RO@415xPxNdh*XD&q|E5u!*tz%^JTrF5hXq*kMv;~ob{!#^MN}p{(|+q zB(+iXx7|Y#DLuz1J2(`&$bR+FE&e235!E^DH+hSQTZ=)I9+AV&DW2yB?{X-IOM9WG zm+r4=I@OhmXoN|pa^#Ws*Q24AJ7^ywJrPlLm|6H7Y~|O6;U@PkDpTIqXJ#Q2GX*0> z8M_)5D67=I6RfC@yOZVt7CUtY2g_#XzXNHVi&>G3PND|uI4qCmRI6Fn@pXwtLiSU% zj_*N-`tjmXqinxBZ&S9Lh^}QsQY$@zt{+n$`x{I$qH6RF&~CKvx7tmRbGmv~kzfeR zyjG~#?aTGvxVk9C$a)+Bhsm59gxu~vgK9|eigH211n*~AkM0q}_J_z)(EE4FYYI*~ z`Rb0;AmKCk@0nj1N7wg7b?l-uv<6usLr?qsR^CWto}CWx%A&pwHs*JyVMN7B>fn!E^iOC2@v+)j=-Q82lyE zN-oA7u5{yyd$Db3i}5OTD_T%6rY{zOrSasX@o>HJr+f8kLT_9w7g z15y63YFo^H<2EIFdM3w&1Z9E(An&_crkJW;SQ8&33 zJE+JIWzi{Dj)cK*hk!kE-H<<7x4 z^}v1EWG)_y{&-jGCC1mr!ltOKk#$cvB=&0J(0xqkvt_(tmb}nXH^HH<*Se^wcFVN& zkUhq`ly9lCNNPh3Q97m-eof}sCS3TRsqb5&Vd9h4Ls|3U%ig{D3kf< zR(;Tz(K)8pIB}Ha{LzkviIsCCu3j`{DtZRZge(8h(~_e7s7K+a5aAbNQ?<*z15wkZ zyJ%xSoMg$M6icd=`gK3oG8;6oUQ3WiCni799v0ac zYNwrtn;mQVWJkEJQc9p5Q@MAI%V7Y8qE;qg03QC%hr602)| ztBo+Cj*Q-NtvHT|drC<8qL#_&F^lq|Et7=Z`{s}h>17=z=ZeyxcmG3?(D;`{kAV_s z*pNiqU6Y$H=t8FnvBA{B_ZnF(vZ@n6e?-h6bGRKJ=eoW=DwtJ6UY7F6r`T{HB(M`8 zDe!ra)daoqL~492cFNJf26?Z=fBgmCkxuFp%Lh6qEXhAeFiw}r!kJI#_>rSYNKBcG z{xW?~l8sq(T_$A4Jzht;3Lfhzts&L&?}zTkD*Rbr2;wcJTE111S`y}8qIyAgn#mqx zTT!1OU3crb4e$s}Z6qu~=w7R$+JyVBV@VmZ$s-~8AW2Jz{p|j_SY-9pt8(w``~^Rv zH(I~BvM2gYWaAn|*g6(!)o8ErkaL&POz!Y^$bS zhWd06CF&&zaf-|+GRgtDUJF143p(t8EZ%X{06VDoit|5B4N1i}&tvMW9=F>%4eztvL0Q5R||3ktY0B%jRl*=!k`>^<i2YLE)}qh+YMayd_8U{aE?AGy9pZ>fIRTMT!}Xj|O^KZ+T3X z#In%OO6NolF_zzQn~kH?{|fG(Gj*+&dKI>LNHIUsf83n^n1A%{$JfQ=HJ)md3$YLn zHu*mGn5)=^?6$>sKvQEj@Momk)Az>fwEZ7E!&Z}NXTXN3*H9+@=k6^42LQetZjH}D zHAjQ5aX6bv(Pq~WDFl}ah_u0Tzzw8pV0R9PFo2i%A(z6#{qJyS0O$M>NyIt*UcR}? zB4|%oH?{D5G@BC%C2^}iR;R>mUnoQulR&^^5=R6S=RE3TvAsl>%Vz!H^B1WeWB8Gw zgX?%o$SmC6GS19#Gg(-i;l|HV=x5O%B&cfJkC8%4+tJd-pEh-hoLM`uBb42{*qmj;YmAF(by24vbGQvB?XaXa;rhX zF~jz*85!=*hk{@CT#f=~HfnDo|MUN^8N8@@k=}(jBXqTp?R2zY`v02{ST!QS{}U}n z|LmTD0cdIxrNi@Ap!qsmgDJr7jEjHx?H^=#_AuD_BGUd}Ls2N-$vE}h2Y7ZX7K_oh z*(v>6*>EUbX0zWu|2oC)iqRX7?vsWMnlKuxroHf{8Dq97yWNf;MtRkqOzM#v`x3#l_VL*>ro0K^v=VyG*Nar5h631&-U9=c3$jpoMoSoq%(4q zd(G^7;+Hz_OTW4WzR~H3`qALI`8uW+!uI@@p@LI16Q+yflgYNe*SK?sxnIT{7ZTz* zLWFrvC56Eg7&D8Khlx#x9#OUWE7s0a~3r@zxW}j;6bNc_zFUG)r8&R zh77M5X#|yM9_Nez^6HT|Z;^v+5=3kj-Z&O;!2d0}b1aW$X+lyA2= zREFTY>NPIbR?T{WpZ*}Xs8NypTc_nTK%(x$GMDhbo46K=1g`x5fDsxt1Q(qk8WJvy z)!Wy)Icu+v0)E@Lzu|1Q*%NDaSc;#$vCru)|L|9P@xD~OVl`mSc|7meSDa^Z@!R|-YE>h|dxQ4G3x)u7#W@pt=x#%` z9Oza<-SA@HjffaX!@@TT)lbbjk{C@b7vTafM^{LOrN5qDWOkHnpF2(xyM+%{oRB@# z9_zOqk7x$9b?ubQr1H$}7i*!!`7bZH4v*O~Zl3Ep7-kuy8)l#3>^xiZ?y4^Pfeq_n zm07myQEp-UtWbewXp2?nc6UeX&2b?#Fw?t=0;84E<<^IhW|sN1LTF65GI$_-V#C!~ z@DZTvTDg1`>^TP7z*41Mhz@qu&fjb3l(stv`k|Q9tXs zLxM|DeEjpTh^w!%z`pp;i*Wz!cia{FfzH0yt?smjT~?T4)g&$S?SjzpC}tkGqJBbR8la9ffWJT-%zL$>n(M z&p8M3$+W;wfv{%lY}i?!!jqywKC6=Hc%DkZk9;N8`HPqj*5sZeh1@iX1wV4Xj|{7p zQhOfQI_{Ue{TceA7K_;7>~dbQ5U|uNn3NY1SR7E)Rxeq4QZMl8%t&zmo}kt1DZ)%VVylu&hc)a71UmIOL_AFxo|y-V|cdii7R6j z&DX);%&v(l*?o0M~%vkNdQph@; zr`5NtHz>_7DUx#;D#4-ukF2+jt19faM)xKK6jT&MQYmQ!0ck`KkZv~JNOx=+6{Wjd zx?|G~(%qfX-JSR0yytxPeE0qhzqQwTV%8XA&UK}{%H%*fWr<$5!GCPq?2*^Sdq*`s z{#d|&%$CrRdgP*;jz;fa{#`04v8fjU8kB*=N>B0@)UizDWGWLqADJ~P2pC)e9R44W z?qnsw9fGFs=b1<}ZnG8zv^iUGb<>&%nG@ZGwU8?bSgugZuM_t(Bzs+2(ZLZmi~|Ia zEh3yMg$EEjRR*F@Fj=ewPcbov@`NwAma3#KwWp|i|JHva;@pKe&tp*O7uv~|h_93y zW;N_D%mq=YIrouwc!lOLSfE8hnRf$0)_bm~6h#(kN+z9?_=>kw=892foQW|A3*Eltp!IQQ$6GcC8+3=Q$xcO86SONOkj&Mlbn+1=1G3|pY5D=mHZVq1e-=rpy zoQKsk+Ax%;8vKdil5|$gRTn;0xVN_1k-h$@3;9G)zRRWZ-DXZBKi~qcakBQtNLER< zPZJ>tA<*Ez*W0~XYzfXW(HZ{MCZU_5cQB!Qwb1)y0is3Ak5xS-kB*s%h%rzZ3}suxfX4PVMnUWq*0& z^$|@^$v65w=f`nmt0p|tLd4cdhfUQBZ+qZV)l~0EUC=yt$1U^gO!K6W3$a4n(%r># zj9P7v8OY)VUhH|&`YUpO2I&6m|w zuM?gpwhFZI9A1IXEUvV9w8aRm;qm^~I-V6pM!Y1=e+lhufcM#}(}BCbr-F=7i9kmV zkql@5(H%h<4=hZy-D1L`N&v| zcn#>k>1v9U;bzqYv1o=>>NZqA)WZ%<=>UmLA||U}-Nt~JQ|F)V)}s@g&0;ve^280k zK~q-31;^Fo82$~aC|eJlK;vK3rJ7yb?D6;eu5l?i-b+-|YF{Kx9S z3NQc_LtLafouce`;d zP|MV9?;BTNPu8{T-;$j3$!N^un8J#PhJUGwV8=5nzogMr5|~o1OgMs8&f+d4tx(n; zeql4Z#OB9Rou}-sjBtNC#iz<~1(}MT5aq$<#TlMx&Gh*45T^{Y5AP3%^wVVQr_?f4 znd8O&o)6wh^O~0L);u^fs!SH9W-9K_syorSRPsPY(XXy#G#yV%mfwbVYhr|~65n3k zy-XUj(Z)s>z{@1vkhCoI=&__uJb%2RrDRzVKCqB|dDwE|g4IdI1Co&f<)S2$!4$8D zbmXgNzTwZ1=AdR`cJEPrUY&qCzO5#mafS(}EqnClY_i7&$eeju`DhBgbw3}Ha*VI& zY9ML#V<2-8qVguqSu3dZXX2uYc}S)MRqYg5@y+CjT^`E^lJ39tvM9TGyG(RyI+oM8 zVqK4uZFK^{iecA4%N57NM~FYhG4i#T;4bRa>;>uZpBEm8nn&b(f6!2(i>% z^&Qu;?Wg>52j?hJf4*je|H(TZFv+_MYbj-XeTWiy-q#LKPc?Floij|#`5 zW*0bXZM&x^Hbm%mwFc8``$rk$f3xQxq3Ttq;Y+4*^oWcLQvZUKCNa66Bcs?2Dvy1h zcs#?5&+rfsM6HxNsf?o&bSln1zFNDDtsy)b79tf_|JCmCQT<}zZSH3HT&o*W%Aovh zhhGDf2Zq@5x$x*+UwO&omYi_+Ml=fll|BB{ea=Khf?VIsiHsvY9bJ?~pqc|-^6BgV zikz6UY~Haej$CeGu%wi{AU1$EBgW-J`;5)^%$bFj>KEr+AD-m2PE{FeybKyx>s zmUYcifz5=&T0cF6R<`E*YaWv*lvB!&g=qTl`AX?~`VLis0S=^}@U_{K9^pJ3E}IE4 zUI0bYalD5hPITCfe@VyC9nxaXDyl2y%ZZn7O=_*May4;2?Q!A0GU$xKMVRVB{fbG&Hz!yFe3eUUbZ-OfoZMUB$d~o3t}H zfxa@VbiHsdtbZS+Chz0XhG*_vv$#`g>xP%VVTnY(WJ$=uLU^uq;wHbPS$d2-mN6h2 zy6oyWUqIq8Z9LdXFl_PKX@m=gj2IYD%S&?k$EH}VsxbryY|QYH->(ocx0DqLlTg^S zKb*p5=uZbL;#c~>S8i@|rM-sSqw7WVM)CMV4^E@Hv|uHh6CA=oh>T{eBgcn31#)6H z^?Cd3wPIp-0;sA5fr z9Z5{5H5u&MKNNYVZet$E6OePAzy3FVYklIGkG4gyp!QKu)vMMoXZTz}u1}RWV~u`+ z#%;h=sb|Qx3WB)ONWDHQ1eW)A?extBs1~(LfR=~y;Z%|=;XkB4A?wGpo9BWCvO0iw zyFqdrx#r`&GZ1sB zk{WrG(sr9Ud%rlfppe5-Xc2$cjW*;s1VwC=XLa;NU57-|4J2+AzRhU;y$`q^5ogTM z_13>-!XKfC6C+Rl23&2jWt@AX{xFS@<1xxt1w842 z)WASIb+4Cac*n-8&qcd_Pkk^jTk{CYeEu2f5;fM>p^aM1ieA3AuIFpVmru8Mh?gN5CC zJkO?n%7O~{JENM1`W5FyrbMLb%euU(?j-Wyn>9G}oxO(29^7H%T95VPdS-tgL9DJO+(f z8KP&Sms>iS;IKravw)274}Gp=9VU=RRB^NZ6=sIzfjA&x?u&m>(LzjhV2 zoa7G5%43bfQA&{lJ`8lsNgIC$v|$S2^KHze<`rB`^$et1tYuA>CWr z)$?yQylevko4-A{Tu>}kn#*tn-LcztWxNF26cB`AVuV(hy(kF`A_MWba(XpD?a@O>u?Xdq>{#T+<^7n z5q{AZ!=mEsJz|l#gk)IA8@%IXi6=%u(|)xn!5aM9o+aM?QF7Nli0zYo7g8$*`^?5A zi@4&C3FTJd={nx_C*=4~nSsnv0Gy$kw?9-03heM=G!5lZCxpDV4p_*-l|3Lrkf6oa z##eOPZTNi6eh{OsmW8RHWi5Y22FNR~Pg9H53|jZHfNLC$%8q5zt?e7kr&er|Osh2f zgx;^v>XW&nPA2jW&(cPry$e5{Hlb&&)DLR6fmKv}j9%f_5Lq&TGZHsf`{vAVfc{CY z)aUsPvH{rQK2~o|OI1s^>vCaPIT1xORpqGqKZGW*I){Hlgf19CEkS;uu}%qTiRIU3 zQ%&XUmTFGFY84D3n@`1dFp_WYnDcwT8uV2}ounkz^0@G93k$1JbtFl#u-W^3k5;z& z`U=a95IzvbmK9h2&42o||NJdAM7UBIB5Y=6A`lf$H)XRD+UGsS-TwuuZy%Dk6J=b0S zP;YXW*I{13rK6lncDRrJm}>lG zML3B>+t|grHfinktY|Q4{t8Hj1ed#wl`* zC=ICq&-{x};AkBbnz738M&OHZJ>*WL zUzwuT@tyTuD)E5O$Kcnf5MJ+THng;gqM-9B&#iyLbZP64+$u>3YH_W=o!UHBaaf%f zxly;}_jn|!@Gsm64J+e682@JN#dGtAw6PD|9vvkfG1@BcL>#G*S@K_%t27_Z-=OXc zYvau%lVkqq7JLB(b)Fj0ld-sj2b)ohvp@7XmRR`=1WF^+! z4_2*qH1Unv&!#BGmig`9CoTW-abo4*THnNi#)JKKl4nMUnxI0TOaT)O<__KJ1+Z1im4K)o5+CA z5^BVM)}_TaWjGRWRmQI}?F*N~FlrFGn=}@86ZvZ74ZT5og^i!kL z8fFq&gr3GWz@e8BFq&^TmJyiOS?p7@+@Gus;cP4?{1Tv&%aY2PpS6~{dK|T9maqNE z-<0r*@_ez{hoAYaU?c8{^FnO$YW%nRoleY!&mMZyc+sz`urS!k{H2=Q(6Z{~N;yjp zg;q+rdR-c-obon^7^Ek1C&J!>vc-SNZRAst^`pTwA21Ba@PF~nm;9cSl5I<2Z?oGp z2N|{3uUyI4QaKYH>dF4f*xH`y_t|UtiW5>H)uLoW6yqI~^l^*lW{~7+5%IWzs3k-5 zW*%zA>^15h64S~@`s@iwa=n@(ix5vOzT@kA)qEpCSE;b6b?acmzxtY{@kwGM45CN^j&zg;T#Da&Efitl+V0XBv>~@A=7o+;O?G6>KRC(l%8Yf&fU%rCER8FEEgc3@loR{U`Z88JI=Bx^NG6z)EIg-h%?Xmq>0n^=~nYP^;L-7 zE<$^&$8`2GGAr>S@45q^!dI&uzW@cI?Fq_p&s0x^`286C1a~cC<=_2Se8)MxOm>x- z83W8#rQJ8_M|n>o-mpjg`BMB_j;hDa)5ISCJh!IX&B_K&x@S*N@c?SKthJ;`eJti0FFE%C!~^MoN=q(1J~C&^>x^QU-u4#EPH8 z28C4CN}75SD^t=-ta`(XQjTjqgD(Sdc#aO&g`04Zf)@)tQE`l`xS5NF8m6~C4)iXy z0NgBX8PMu%`jQ;oUo;_$6Cw3j>4z?t5wPf=U3zB_MMcsdsYu9S`?IVdbB1d3KNAS; zDuL>2aJ;b~qr=N^zqwn@^P2}L9%mfJ#%A<+6;{m|2^URws?v`U)^_h`m* z&jJ4QbInm+?bZ)8$H3(aD?`E#YQ^_u-OqnPKPCHq?3Nh(s|`hXA^{Z^4TKw_RaobE zwfW)WJ~65r9|C}cE9>&y@2xJ75M3x$pb+zf!`%ucR`1`Awu&moCq||SY zs7wrVg+V=E1Xk)pxNrE#D~SdSn``Xmp3wv+%h5O24@koW(tBYxK5QBXkg0`qdap`R zWSQm3GA%9pcD*LLqr;-jtEwPOKV8`#+kf|A2o&1UCfJJ0;_#xlH9GX*ac3`80Gd9= z;_94CDo+j0_E{dP{-I+*Bl@0((=U8HOZ%BWO61AjOy@&BXv*hXis+>mO#^c~U6dAmk!rf@j-&9q*u;-c?|`#yyYxyTk_HyeuVv4vdQ z!|7y@$15X$L>b+Z(o2@-nTMI^ zn-b9oeoH50;T3UG+#^3CX?%Kn3&MrxqgaD4ioKnYOW^fgjAYy23p4$ZnaH@cTkRTo zaDfh2l0wGM-i=HE&(pqha!T%(0QYE-M*7L?8}%1-3MY@;&AL1g%VWrh)x_&?Qd~0s zd;SKk?os1bR0#>^C(&r{iBWNc@w90kLtzbXKDpL`d6w{VaC=zLEcovNAsKvm_AP$u zKrzwPJB5Psm-_hc*mVZ>O_B5VZ|EzC55@uZP)HhI#gY)3+1Zit`n+teHN z+T3oI8!2Vz4ca9%DB=@3pY6e;iqIdQq3d7r1*%r*$reMWsJW`po_gMT8SL{ZPKQdj z`@;OI9DAwd=z)`t0Py^uPaq;CVgXedGo$sS|B~4NA)djXS0#|87?Po`*zD)Tul#wx zYaI}pB7I-d?~faSwazg_6(U4!@yHFx^E(cje_Qf+;tD)g5wSOXgQMpqDD&dR=$T@y z3#-of#5YqOw?KDgz`h9z!lfhYu{AZ7gvbWnYzb#nddOp&+8-4gPT>6!QV2FKd=64L z!#TsfXGu~li0(8mhm0Dyo#1O`yPc70cL^g^1_=4+9 zwT3*4Xr^74j|&Tidx7ir+`8t0IcU`O`~@fV6aQ&|I0QQMj_>{Kt@trj21TKhus@2ea{Lk7I`BD=e6RI#N3P z_H*s-d2Oytb!HvwCy64*J6$NuL!d%4ykfY*ycXL}_ls~h>mf4CPGB*1?Rr^6$%+ttNVJiP< z0M&YC>P4yeP|$e}ddzoT~j~IVJ(LW6VxnYrmIkVwk$W7z!{=bF>9Wj zG7zH2LulZ(yzhNKY9ccJD77mSN2%O!rg@l@&wB0Uvt~hzh8L)lS1&5bEk*gpt5j}n z8nuDBW|_6Ds+INL>x^TFhB%wix?H0U{P3n-op$t4RTtFWq-~ZUpT}q@O5GlYF7_pk z2=z-}+}`wwV(bQX3uQ!ri1^>)ebAt-fbVhz7u2!d@OP?-}PVKBoUJE*3fW#(WIoG&ttxOKle1b*RhB= zpRtgp^vLSd*yfJw)bho2k+*rENtK8~S3)rIS#{=IwnU(uRfU?%6;hWT`rl_$C2_=R zCtw%MUWSMsHX}wFDBZA;75NU|`O>xxPEv+OWpmhg#=yJNd1VVBlVe^6YF?>)Px^_W z^TTGJ@{f!>dyGn4e*9f6q*Z(x;tR)~L|SMGE3dx~w%YB3`r(nf3W-g={D}zeNk`+6 z<_4yKFCeUmku*V?e-wMRpi}r;(l0t;#jcMs%q98zM%%vN zHL2erS*Y1>vCUfkh$CN`cI0!$+cub1-cDev<$GF*zJl8s!JcNx1DHx;bUW%lOi|{w z>>`id=dOaHc%(>nt=EJmrj;c@1yrBJytj*yoz|l2O)TpggWW39l%WoE6c>})7B zUyppffWvn{X={a09)F=goYb*JP)*L1(_+CqJG~U6^LWD)JNdrP2ucjBBLAu#tcztY zX1w8A<`nyuLO1{4y!0t49o&Ytb$Mdi=H&lWfU3Cj!I)kUtkpD2_%N{jzal{($iA=w zihe-Cs0Zmpx@F_7l46dW@a|mg1V|`qGy&{jKxlN>A{4w)Uch~>Y#X&7g32Dg8FXib-fc9vk{w&1^rk{`_AX@8a{D_D z!`@kN-+0L3%{yO&Rf&?}c)&no$)_Da$oVYUIel3ichJ}6Pk8Ma@yqwJ3|sVjR~X-R zcwc|xG`7(Nm0`SJDp92eX_g!&LtU^H;oXr=%jidW>{SM}eElvRk2feC1eBOEhxk=6 z!ur>+3uPfPoJp9)f;A|Q{ImVNcPH01wBpL>cF%?p<~9luo;(~}{#Hk2JxIGnrJ$fi znb^k8`1T7Y)^$ywT8|-aa}40X^Berwwgl=Vqf}&j-sLPN{Cak$6A5dBMfc?(zPxv( zyMhnwM_w|l5YZGesql~_UJ;Tpmrjb)BR^sj06gZZz%HV1mDZG>m3!l zuYY9w^xI^@Lw$HsVGs{7o7})*6|p^yQi;k-T%F;14%MT4icdxbAF8FdntPVQ{U}@b zs({8qzCOYERZ&)5nd+GP4u;r36nUj;^(8UsD?FbZf5)7W;z;$DE_|tB=E^3ydKIzr z`r{S46!yJ)g%e6LB~U(^f0^RHaWd*pxm5<=zVTGLt-E2+Dv2v{!)RZhBl^SGARGB8 zISqwGIJghiATOD4noIlYJ%95=7B|1O0Ra}w4uv|fd*MFhHK(zcC!DM4%|a;dr)Dc< z;M?zc8jcr-m_%nv-w#$#JYt*M9H<&{`uI1OO?*6INeU??VqZ9giCr+=gKPP3iY%)^ zf>Y{etxybNPg!KdtkKlf@^Gc-y6R2nIT4TSOUElD!d&(6j#%>Jd{7|1!n7Y+KhTQr z&B4Nd-ANK}7oh0OC~&~h9cOODN&9~+a>$#{3e`5Nz5i*a(jG{dXV@1P+hr{UsMMrD z3NS9`Gy*0{{b4ce&HisC0JL+hGm#(jm2-LT%2O-`sw9cCMJj0rT;?w;Z7>Rj0swgQ zC^`+os;aSME<9O>hN3vCSUy@0d3d!R7r&zn`GqD`r=H==doG%3WYZ;Ry;+mC+x_VA z@p!jj7$0vptQ9@ZL&l>0yJUmSz<`ErTU#7{NMB@SlOhyB&jLE8m}s*-1r+2A{*Bz@ zSEV55eL)%Ac2*ZzyZWsphi=daSwsRKOZS#|W&e^NC7d1&>{~PlL*(f{*vl|E^4y7gYPItb`)b%$V=A__LGG3F#+Wh5)XMbXq-0n=-PU*S zx)i#)Pdxo#akqV4m|a>F^%)uhW_uso zBO@HSu3{QU9)$=ihvS2GG&NKFILgZRUyB)LJ!5VONEiLz&gyL%J)J<2(lVV)O{*CT zm9~WESC@+;Kk2jyCMUhkxZ-71yvnR}1psTS0OTpbeT$q2XygKjZY6PG58yo8KV#`! z-OcM0SSN^=3L)p36S*T{w7)UQxGkoSJdZ2WTmSzOvgdyqz$iyRqY8g05CK^MP2mO@ zHKhQiyUP=DSbPA9_;{fPxk%oD`OBHKD|!^bu2+{#D3_!9A@Tk6yEn!Y`(;K?^4h)k z!slF6-lh(2@8A-teN9}(>PoYiZDCy_d7r%~MY!XXaEucl+>p~Md^ zkr!h^=<)kqDA89XUt^Aj37?ltNvSZ5C<9|!0@NvaV8q7Mi^Pv_AXCGuc^7!gYpfc) zrT?s91+yoGA`M7m=UMb04?KRWg**@{3LmFN`RjvH9K~rv!)`#82y;)92;Yr-5T1Mx zs{Z1&7OyQY*Uzij6t~dR%T;3ExJsP`Z%?=k<@3LDCYHEM6;&X5M1M)wxqPdWvu8-|J-f_ps+D-Ief`w-@e&3ZbLp3kfx1 zX;S;QL#Cb@E6|XCZw(P?6U9NFf5$FL>*~yhp3ypbu#II&Tgp>B{|V8@!aqfqD2k#9 z#t~`2U6|UYW2=C>s$G$5&-sg2aOq9I*hKCbf3~sN=s6yGf!979c66KQ&SBO17#g<` zY`J$K&@OWCQZ-&>CKHtKLF;N_HnO%Ynt|(Y@#BgR!Xr6l9sz2H{A_(Ob8 zhxVH%ChLcm$U##>@9p2DS!k%uo&(hE- zXAw@Z&QM$71#@29U3)`LRxQ@u-pUoiXk;X9i zQ)d+Z(%MAJCS&qrl9E+G>s@envCN0UVh(+(NtR<(0$m!E^N;$M?^Q6gd)i?@nLD-! z4EyoVCxHUDvFmGuMOFf(J@<|n(`2^$4Gf$><<7qoI*&5$WM zO7V{i)|z1*!6CuQ9LY+tEJljIu}MC}({_!deEOjf%VO}~ZeYLRTIyp%jz?oduF#a9 zEYXpglRyA@Wan4aK@^ERJq*fy8*vG%~XjfUMaY@))k^$5n+H*<0& zHs!Vk`0r0dC1;27V_6j&xc7q)DgjSS2|V>M7yl{m(eCGPd@ZD)tC>CBz+)1R*!XrL zY8fNhWaeY?b^ZluwiX{E-#IIr_)YD{^|B`hit*$E~0{*xcC(Umn~| zS1*ZJ#!Gm=bh#t;Z7A6XQ@lbl%qqecCEwJNB?$EXovt(oL;-mM0eT3FG@2e}z5U*_MCW=m^5iMJO%pLZ^NCW@w0=nrO9J0q9F z%(|Oh^@^~4_g)v^K+xb!u_HpR@skxsWKr&iLt42zZiblBgHoQ90&zU^0_@RkpLqX-bwz3Y#CHW~|yXaUEFQCGAX#UDK(jSqj&a)`%)f z8cObMJ6d7N$79meKN+pc0?h_R<{T~IV|zJ&o{f*nCcn>NW+~Mj;ifgM*x6d~#Wj?t zgvYU)#{n%|^?zbqQL_)a7tDwFl$h^5pI0O}#XyTwO08b;nnYa2E+>FpCI|94-*>*k z{&%=vUH`~L2F02r@7FMFc%P6$eU z>{lZ3T!6X`8TmrT_}9QlhIn)gSoqd22GeB(txLn*e^%3*PL3-FZlqVM(~8YRDPHKr zgs_?7vBA%_C@f!orbV`@OgxHhm!xU zvD*xrk`POJ7&mb1JrJ%5g`OG2u6S=WK zw=P|d2(L9jb7uus|L*SE(tbk2TcIO~oPsuQY8{9={Q?TtKUxkwKhLHKr;gRdHL?Tp z#z+>$ubjF*9?Hkn^lI1~4!5CuAQQ*{m|P-@@Y_!tc|{F!o5LCMxLU4r-vy(>QImzl8B7 zN;$Gh3o&m_o%xMJ##}*fm*qZzzj6vdQ{}SOOK>{xvi}gU?bjU#eNE>3p<1?PCE&pt zg^P~+{w4Yz3i{Va6VM0D8vldYB&+UeU&&B9&!SpIJ^}L!4{ISRKKM~O4U6>`uw_M{ zbKdvdyKk)9OPw}hi{Z(4SwKILmaC9s=J)ygGpvTI(`{d(_Or<{AaC`R){@KB8~B&2 zGW7P)3(eGu*RQk*-v*hxW>Mi^PzYnWfn#+Tkd;RO8Vf>6H zoOJ}-lVdXr$}q&LoIgGjyYQR&&D#A2MNV6NWnOeig{+-~Q5z)Y@hPCC$+f8ZhgR-> z>s;#JPFTV@nQttw2-tqlnjeMFqqWuEXDKxP8&PO*g^%HDsW_r~eH%#PH5p2+;;$wK zWe~A&s>7ud_NRfw++QBTt+yMhbW^PzAJrCI7g-~@%)gxOlrHxT5HgiHpPD>0IlS8Tab0t}YTFItsXI*)+GmXP(eMS>K7!T0}v0K4-ftrzF z8>i0UlSc}Rh0jg;3XjFkz6GqN_s~mN<1(8ZAswbw8urC!##dL6`LYl zrE5B3P~06(EnQc$Uj6(%wfSiFpKjX@4HF<^91rQTsbl84TdeNcYL0uN_(u?R%o_{8 zw=kCMgX5lhB#?@-ff}WL5+3y4M$!7MKXX?*%vCfZ67y}Fw^WO(P zAbvMlZVxn6sv87wQek%#4MRgMh-JRkZ$G{=6N+Kc0V4Ghl{PNiMr0$VJSe$g_KZcX z2GFfFy$E*70*sfO-dNV0zir()<+c0~My=6Nmp4%i)Oh3ZB=Kh+x!ub7djXEHF)w8F z-a=A2yPFGZ-z&{rb;|P7K}xtg-R~$|R$TShJ*B%H4GQ+J%kh{U7G^BtHTau1f0Me) zl$(sn(a2}XhZ_|=_iVTDQwhTQl9e_yv3n^xciq(TeCOf^4j*%^{`cV?rh#XAUO+gV z3(9St1|OG1Uy&U!;1y06vFLYa=JF~ix`DRVhPEn4GJu_sC7oOd!VzPt4!Xm~mGjPd zAEo)7_Bd*{@&5)OX71;UawkH>Xcf0Un9R>sudSEq(|mi-<(-U3^XCK+bLWCaaXDbx zEBNMq{~$rUS7bm|jE}b1`ut{>l|OGvd_nGfR!cGiAjS+TPq{i$9|>LwwE590#@Xcp z?zPFxKXD}z25PP^<3U;%(m+?0_?pY1qC1qkkyf)VhRf}$Y-;s$*uz|J&QD={dP-v% z`>)D(JEQxr-#x6@OHc5;_)UkYLejQva=1TFV-?+~$rflM!pvGpCJh9lUBDzBQ=wfHfRmiG?4+W%m) zBKlyTs^}q4zlx$C#o^Jbr1-t4R?84imQ~DF7Y8aNp(g~aYObc(@!FggPDy3^8#Ml7 zonlxkj-xD5Kbob3Jz(~Gyk7y^S!xcJ9aDvAJh)hba!BApxwNr%@UKN-PrX{rA;#%<+UkFS48;|482u-K&z~$bpw>bZ80(7({AnZWcs{i@QFqN32 zc>FJ~73@Urzff)v;L6In~2|16f6EH{mQ{lO|FSFSn?NdA6ydx&V-8{N4^ zEq~vy`@Gs8DW6)CDH*OhrsqlyS9quD6$PV0 zvFF!tuEYHJ~tvk9{t4s*`pHeJEafd*HSvBQ)G<|BVTyOg_L+MM2= z`);f7m!8F1+Z&*S!E6<#&@S$-u<;IjW3!0GpbCiS5Pn5&qV zsR>_w37zZPrWavZcC318D=1|hvmc}KF`+^D<3m&C!{v@V3|y+1t~Z|~_zvXl&u-7h zj76$dgN*pxz9w?os;1ul+UkGYz^fx3Meof7-Fv~HssK#ga4(SA$ngvU4@~1f!@XZ8 z56G2LeM!7(Y{tVvO{w%SMD|jfe~)n_Fi_ox&8?63fP9Gdr+A@ywGE}8-U(PZ3TXR- z++**O)KA@u3? z`W(LkynqvKXgvs2!xuljV}O0lJ=v{X$hDZO@okeW(yZ4XNf6ryQPo%kjl9C`)oI)J zC|7ow;(W?QG^GlU``b!bBBgkgZp!(Bhn6e6v`eQeETics+YT^YV;YWg87v0f6w8BP z74=;oK6W_W7neGh%YN(s8w`P8e$nb_iavKh=}k?W9=29zzuoW+5Guh+ru(lt3dFW4JEzk}_ z;dE=PAYTQcKghA*N*?b9x)c|Gjb==wg(1=t&Tco-f_qazf5flfKR6v|D^{4n{=+O zU?COeYU~voTrW{l7?kp3wnlTpLD&;}@v9;w9K4i}0CB$Fg4?+UYfo4Pg;?+Xy!*Ae zFX4B>U?NVNPO62yx!RUtQh*Ca=}P}ES}w?F!1_&&{9jF*U|&;DJnqjuSP1s#YAa(6 zz~Vb)*1MeNf@SxS=uhTf=nWEtl>NpbRT|(uPjEgN5}{Nf?T`?Uc=ra&{K=qdvPV=H ztAiMvXp| z8Y1GI&*Ga0iwZDpOd+CJsU#kF#PaaN|9f2Bo-d4T45hUlz2rcBu#o;j;^3PTx$HJV zoX@6=6v40#MezTw5c6n8jaIn>5DXPV3Nk1@yIr5XR3<%q7SbA)Awwah4My-uv4Q4S z?JsXyY)}Y1O2G3Gc9d}-53hTbfc69|3(b36EOsE<4g)<>gheF!ON|B?$8jIF)CtTa zh{;tg&jpEMecXG!UsqrXb>CZM3dl4vI`##$jK0%aez@FJxg5ARs4IhjDGd?EeGiRX zx{T>WAzLXu4AX_ndg-T1#mjdx0tTQ8^l&rvn`^ScVID4jY9aB6i*N#T!WNGr`ioBd zP%-G7C3U~}a|P(qz4YQ@2@n_H58gpSPWg?y%QTEa}kyik_xs)BFengMBF#>fD`Bp^=USr4NY^-KUUELe;@pFKJOwui$Cjr zz56z~d0GKEWvC-pwY<~!a)2#mgVu-guhNH{J6m*I;5ju2xAi6_bEksi7pPfLg8s&FQGTbdIu?+Za+=lwgw2 zx|d6)^>TYG4E+3)k||H_xCc`x_Sf^JQzYB#l<3I3=*gv`g;MFvwQ)m-!94BTPC&-fUg( zH*)x+1p3G{S}yRh8TP(9J!rwn17`LTG7T>wQjzUU1#)kFKJ2gmUx;ME)I1aMCEX&AFT?(aaELHEDenIoB!%aoY04-2P3 z_iWUES%N(~IwQvC+b9H4pjwU)@TU}i^R=k)t_JjeORmD7tg+9FC8kb?P6H#1B@hoP zCD3o;(ToMoCmU}D^pTNOzlttxoPm@nb>4ZK;lZW|i9g(Tx2qG`%j0#B&g;v9QI=KH zb#U7{lC-f02t+M10}&hxIQ~cEsq%VW z9H7@Lcrti1L6s86&0%3zgR|Wkj`@P=YSKN)4+8%F$n7y8s>lYpk{@L>>Nkcqd~3{X zuza~-M2}$n4^!UOSYA&%zVm4tzE*@CSOl0K)6xf}R3v?){)8M*3^4$P z1-#;S+#%)I(K%3+da%6>c0qOQCZnQ?xe8(r(DNN4<&flDt_m3KK|?JaKwlFtPJ|}0}XCsrQzwc5M9xn z`U!X3&}?aw%gHcE^$whX1{R?nw(SGemS^6H;RN%R6Tr=$g7EpW`=B7UbDOIQXF(o! zA!xZ{o;^=KD`*Qi4kF)W(W6jukrmGRT~L(LZ{#|7pR=7pbk|!4dFJJ7GMfEw!cdz3 zwv37NHK6hq&Czt6pQClBW8+cJb0xvo2&A7_FymXs%RaQ_ZMb1-xDe#M+DLQXm4c94 z9{7W(s6z!E%Y$9Zp16#A&#$gI3X<0ApjD#l&6K&ZgtGs%YXWkxQ}~7}hGLCg=2!ciJh2Lx&+yo4+#ue!JxJ_zI0x_NV+ktf4H=I4 zs^un%5nPX?GJTCQ7vM9A?qc_&LXBha8tkIkPyY{1-yKhN|Nd_~MjVt;;n=cAA!L?a zBs+Uo3dtTh9I~=9Q$~`RUC7LcY_iELWbZwG*SWu+-=Fv6zVGAxe!X9>>v~?#=kvN= zp9(+xj-k2R7In@2Q=vj|r)0GL-L?|CHIg$rfoHWUz<)n;>Q)}5f1X)U zBNBLt>8NXbZBby|pv65bmtu$4YY6_)VyA@jMS;$W%pe zU1Rf25-As)YFCetnsm#i_(6L&qHEGQ_*l5k^LPn>*~u8=4=-aHtF-;&2d_q_ zHuoQW9fhf`K8F!=m5skvyUl)hukV|c+s8~92ia|PuOBGwFgT`UD0(X2?d;M>-t6v* zmNI7Wte176b7TbYxMM=Otua3l;h9kJyAOXd{Yhs@jq65d=R!^MM5lv$k?hm?@T2i% z$s~QRP`P$?uuX@I#?t&&MaYZiAR>YZz-A3fP57&Fmmw|MA2B@u$ z)7}N6#1G0<#78|+-HTPpeiW%c-=%DxO}g0?Y(0zX^Rs2#aTH`xoOBa!MHv6{SWf@l z<$0sj;cK$KeupqOH_IRv?r;_3-q}hLa-053K9K-rM`r1G7QZ5nckz8p${Y4a( zaP8B}^X+)~p<0n9QDNFA3iVzKgEchUOFc5jo%+F5uMJ<_>`waDes#*QC&@Kwj(q5~ zZHlMsB@wIc&^`uJkP3|ABsav1Nc1mPYHxjI@Ewuj5^F^)J*@TmV#fGRL>!Jf`j=U! zWT$Yz=H0}M$W|tAzIF+#q3_?ms{;hzc;aroy)s%>=kw!g8HLNcnWTEdSMH|1o3nT7 zZwlKm*ww(8AohBE*IC>rB|WBq+dJ+G?Gg?+7k3z$?r43TakCrcK6z8vH>|7s zKja13#@143lY%>;F+8cc%%b_v3e3i+d%?+UAYoxah)|Y!BfM@IKKD{1;^(kU%{1$q zIw(t2Bo7B#Kng)yApZ^KCz>&SdgDZN$Aq{ueR0>b!~D~>dFoH8m;k)N8N%RXpW^Wa zUo*i<^848p6iMb{Zw+c-q2>3f*XN&#wtEq4bA}w8Z4G_@*h)rDl!qU!MDs>Qj>&}h z<>vn4`v##=v2nm}t93qgzsUq^ruXY`#L5|H81N9A6B+_9WOi|VJeGTX)e~{kZ3D9Y zL3oV!a+O(cnQc;s0Q`DuHOa}|5>K&qQj%7{nBEYrsByhh%&H_Ol#uETnPYWX@{-A) zQmw5aT<`C+byRNz9joyldveKZeC=|Zg8OgG7~4V$@jv<2g!8Zc>&mXX@nq;=i)80^ z+U8e=(|#$Evp;V%tt~!odkXsF2=)a}N^-9g<`~p$@>}KwE{KeGRy)Z>8q#}=zi+v; z?)1eq{Y@|>^V;y*A|sCW-SU#rHB&8dEk+uydpr4oL7ysa<@As}@73>(nH1rBS2M+Y zb9a8*6i~tbUp7%3{7(~}bQFAFA*kTRA;~Vc_kPvLBWYhSfeHA$?h{eRUf&5*JI2=r z76#pG=++lL2kt}kDRi<|KVt4_yk*E;W}lgce~-#u6`vgVk4!%Mn*EHf%8H>I<<;&b zUE;Ie-t~-A{Ze|HpSv6v;-!wRoZd^!phz7i6X;vm**))>H1vOX-_`M?h+D-cGKI|t)^_xNUwIehq2+*aVS~f`Sf95nv_r4s9v(ow1;#Ntw-ejG%p5^Nmt=Z z3t3a^2e-T)2JJQT;m=8*RwBy)Ig+=lj@Iwge+XPoAC_vDo3cdL8Lr!D5gm!2 zCd;4O4OMfHB?Y53|@+s1BJHOtM!Lu=+$RN{M zocg@(pYd#`gdw$hvArRjd3k$i{YI-TJ=;H<>AHbqhH^q%hHoQyTAxOh7Zpg37;3w2 zGkh8en6kzGbdzZ2{<;M1YNX$)aePKpS1;#3U(9Z2k|oa=(*&2tOgu^+FO*I&^;;B6 z^h_P2__KG9)jx|}wQ{X}*_t#`=8owYs zIA%APe3SD>8N9}%-b`}evF%kdc{SowCmoygA}mJ#wca9{85P1R9NnS9S?+OhU? zqo`G6=w{8b&+%#nRk!ZT&>(MWLG=Qw4Z&0fUk&R$QEtU`Hm%^v9s`@jMpWS!Q=&7m zW5kZ5SFP2F#h$1wgDtmgCrOu#0^P4Zvm~3OQ}UYeE|Hx*ik^s@oJ!}P0_{c%L{B*q z@!6cD)dFHreXocI{z6fqOK ziaq9_be`b*xB&}=P*u=$H2*^}4w;5+Jems{Uw27(S%e9h8~`}K@KTiv5yWS)KHU2I zSeItjdG0I-CyPt~gQVuuisop>oj9?;AT7~JO(m8s&yCbn5t}B{Kk=jJFjv1@o9%zw z;h6r7+oR@j$Ko2N=^3hkg22e*Q3J2?zA=0ArHyA{G+dEyDlW0qA*A(69^_hdE|M$g3`%sbu|Anp;{g!NwW_|hUpsaz7fR=C(%#-i2w(R2^ z@Zg!K@JbVvNW~D@fR9&TmHOnba$Aqw-tf5P$H>xLA%9hf-4k6L$}S0+`a0#NA*z>^ z0Kdl96%7+|E#I?DGFiTgohEd?>1?Jo!dgc(o6zWerKEP|FzVwgkE+)aNnX-M3J4Z4 z($cm+ugIM-2W&|v4G5NI0usdZDB7ABgFMG$#WBrkj$$?=DfbGXeu?|xiU_-6`ke$v z((>;YVyQUP;PHbuAT0%hBII+zB5zy>j{T<2BZ1w&^L4^ZsB+`#20(Z+wEself^;eyl_Xt zjr_P5#)02lP4(3a3eD=DnF5X2!pP}`P{!%dL7qizKjpc=$kA(B^}OYTe$#nF1nG+`WMN622Uel| z_>0p_BTocPO}TJ(U?Lw2CYq~BOnxX)PH1z#H6mu`-!Ij6oqt)ze*HUrfqkJ61;cSTGK4~xevv#j5QbR1HljRxryuu|uH{88(BK{UU1W62kr znsx8JQK%3s;{6Z&5YH~stozFlH+ll4pkqp;Fo8RU&V7)bC&Z;1%YL85gi|94C3~_F z=*bPoHKq7&^wYt-_{+^+;h^@aeS7gJ>)sqi>3XN^^No{wDvaD)$5CA4OwpI>T}MyZ zYv;z^%lMAf6fk8%ao-MW$%#_Bd7F&yHyeh4NF{<$)3Ev9JK2|YRZ{l;2f*VSarA=2 zLGOzz9RCk;%N(|yNf{y|R)vZo?P{f(8r!(&*vmlUS|ILz<*)Ym{drgrfwn(8X0aa-UfNKQ;dD)=NY-SXm&qULO7 zVK0`$lVz_B*a)$yF%Y}Qgu8iZ?nZMGX1GA0*K!bNkMQ9Lv3noTl0a`$TDiw&W1Tv0 zTEgknyWZXyB+EWI-0ldWVxy-_Ln_pSJPmMw=o_C{kd0QhfkDYx^5KO%&WYFd`c9f6 zjj0~>)`UYdOUL6wa)^rf8=rV}a-wU_j+P@qM`wrA;XL`J!RV5*j0>pT~Hc%S-$l-o-5NQsrpB_dDD4WG5qGUadX>jKH5 zc6vQF9SQupor@5`eo4PWDulc$xf|V5`bya9M^eh~?h-3y4!;wEt(L~soyc7)vRZZ# zi+LW^x7V*F5>qRjgcTPH#p75_R65TKD5&80lkcniNd#E6`|cdb$9a}N^>R4hzz|X? z_Yho9PH5WDTvV@u`A4}cHz=8;qrqUIf+hbynQ>aVBD~&RCDTWkOIpuu6MUg}=Ym4s$#T8xZl{?+Jqr?k**<;(bM_xv;TcLv_~J7sKz#(htUZNz)hv1lq|vo zuDN98g0(xbA2pJ-Ny>}rj`GOC142?R?l^f}_Hwjbu~I~jWlNZFHeP5Put|pV-S6`i za#7aW-vbDB7aKq*OnwEhV$%XLZAk=~q5ZA9X_K{oT3ECG3ek8S*2+iw{)>OypKGNI z-;a^B6oZuV7YbzOOX8+dz`KG7g;1znVpDSc01${UB})_ufl+h4F|c8I5A^Z)si_I` zgjhF_ZRyV z)e`s^AQcLlnFe3D!U^dyG0pGj=Sj|#^ZuI2&oo(CMuHHU5Vi>GfQ`3GTUO1csi1o_ z)WxAN(dLa}@?Y;~D)EF=W@n(TNX*}r-y6t%00H^WBl#_61R3w>sp%_KsDE}#=EPxymW%Tpa8GXYei=rMtDAmhO{SzcITrWZ14_8QcKH` zcfR3@+DbAhHb+;bj6`tsg7Q$Imd=0w1EyUFkYq^JzYZVR#5}Z%&~Ra~#wy=D)AY(_ zx&e*FO0LXSyZ_*Fl0U#*3t>|#H4ZE~_7;@4*B@EJqF=PeZ^hzFyijahEHEgs1~brq z`!6tSVZ)Y;gd_&KJ32r)?Nkg&@t5;H+_K^5tvlTtzlZmE7r=Xypt$_67;sr*@0D+$L0rYo{Yu;(6k4U?>RVuT*}7(x8}5g z?tnH&equ#jpwDNhbrb@S?9*se66f8^a1~HVvKFIWbr*HkM0M;Kt(t^BOOxl*sW`;Tqwr zhZWCw?#FWU>H&>D2?FFOSeZ|bmW$?-l&eROjLhU_Y9^(H?_`3Ka z!YUxK98}jB03&a)N1j9kj{IgY-Tb5fz5MFR!97^?N_>P6Xzz)TI3QoMt2AsKV)bCm zOL&3@Bt9XI7Nq4^`|e~H(w-$J>hcXWF~>=Vj&}0Fzij)oaK864{P3p;5>-j9W^Cza#q9 z3CDSL?8f2AcF$i>4HMW^V{_>H(|q@mK|g81!d^s)X7p1SIoi&?{Qp;=#he6i&Ke%g z4fZwjn#T$VOEoq(Ysian_JP(i&HE*Q0l&C`QUh`nU~N4>kuVEAlVA)mB0vx83#0d>y z7Z9|~pH8^+Tyx}AK*$GNL)=6>>d)3>=G@;D6wf)PZ ze_XrIpN$k-ob!epFoxiUvpa>5gg-I;Y;?o0*!;Vdw>k}cX;b*p%aF{la+EkWilePA zT9dGRZ@^l%l-99Hpc=s<-wt>v-|{YN0m-Ze_4yxAAZGCRpVM3vFg6Hv-M|E!zw(#` zk7YEhAk=^Eol!K*#6N#KJ$(WeMg{7Npyh`C*U$JL{-x&sN>U0c(;HB)-lqJHEFvw1 zE3fmrudQscN%^Q+pN|q#MTU1$yz@rA5|EDbwOhM0vR@f2LLu2M>2x-QIQIF84MYLB z9y;-Wb4`7Bbp`C$F&O+3;*Mz=P7Q7-6chh1;>zOVtwOl^yIfSS$5ak=6pV1KKYuVo zYU;gu<8V5V(i*bm^W(jz+nLSEa^wd9EYvHW{T|p=huL@K&wd*PT|0pPw-XDFqv@?u zP~itRg*4}$1n@QY^oFxeIY@F?$s8LIy-v+@m?*a?we7KSP5snHnsE_tCH2?K^NNV$ z(&xvX$zQCq9d2$tJlI6??Wff;&q!s8__(!}jBgE=`j%RshPIy7aCM~wthC4T##1ST zaLH*iH#1`FF0f!03(V?GpD%RtV@)@c9_&us4?ab-SEHB1Fl|`-2~+vMU)lPmiBKqK z$t|;?-nGG8Y#wah$6zojU=FtdKLp$m4TJ-hMyU-FBq>dB8p~Qhs3viTdf^v7{6=p0 z=eM(&jO1Eye#WLo1GZDia=?Sr*{}GWRuOlEn$31RLjJ!yq-$BsWV79H;6atszLBPaB5D z+hI>U`JHaR(5NGpX3=aL@YFsa}jaH4Q1M_&D}1XM`9Oj;^-xskf&c=IYp-F*8)m2dm! za-X`V*ypl~qTzA_wRwlCfjZpvdla%}7!GdZngR^_U4g09ZyyyH8?3%Q*w4`ZWPHQ0 zZnpqq>ch?Z^?f2c%g>K0q56N?{wAFdDbn5fBsf-BSMlskQ^46m3;QWd>jy}~U#OWpWtLi3rs~b017LmQRQTE)SjQe$ z`@LJ^cLDry(_Bq&5H^tcysqpS^4cE##2qMYdhG|e=sFuiGpQ4XMml@Og zLJP^I1CK=Un{EY8nd5%%C=k7MQ(dFoe<98=cBAWJR3R#C}k6B6+#QehQ4sF1sxYNonGR_yaFwS)8Rl!`>l3Fk!vy~W{&R)YzbhF9!1 z$o~yfxE8?Md!;Ppk-tlKFV$|VZyzumT=CV^v6(*iVkjKr@pTAZcKGUEdFFM#mKU)&#qe4SrOKMhN*5uP43b?$W!j%Wzt_e6|}&rJ>rx7r>&bX>R8Ci-D} zQV`vr7YiyYgQ?@QV; zqw9C~U~^3$_U>nr!xMRlA2Yc4(EajLlGQJ#@$$H!Z)NG?^GZxnuxW+OiLpzhQ>Jkx zzf;8cwVm_Sza(FRv!U_4q&O-QX7tew?B6QOX?ME8`QtRbLbULRVVm%MzQ~Rrk#Ct{ zaPpf*hxJ}&ESIfo%Wz=ISQgbV<>kHj*vD=U{#!)(6BWD%lt!|S)plHkwxMaArQ6BI~tTN!<+3G_ogv@$+j z4H$gI7?^hu%X62=uzvk5wW+4WSda8!xJtXuBO8~+9yN#GE=MtTyp}zANJQkG^KQ2~ zwV)r~HQmBIpLP3V%GC3;M7~HgJ-K2fM%jXjDTuSdy~m(Nk)(9@ty`>6EB!) zq<226E`fWWK4Wq>nuq@OMP9wm+-|K`M{nbWADdUC;3~4eVfF=!IC&u=Vfr5iENRytuJndz4?7TAW zSpPX5#q%}IlqhnTOCia=VOyhq%DHFp%{!>+?s(XO986co+#{Sow`_C$$x!E`X$b&r zh+P@)y_JpAq=q=|`;EDiK%cH$mk(O}l!&eeXm`4CN{^}oZs*=^*mghHF0Woeh}L?D z?jyAKg~sSJZvdSO*OQdI(Z|2WqO6X0{>|wS9o?-ZmuCKkgI2|hom|}Q%hToM628D1 zayT)QT+x7`*pD|QniXHNk&_T!W>hWZb26jjHyX&Dd@`8#_xjK8E_Mke4x^>;NIp8B zjRtecrC)hjeuDD8{hz0tr@oddsjE>PCIkCcB{7_JOJh9DWuM>V_Ny9*RFt(`*qmvdrDhpl4kW>T>_DrX? zy1e$rVd@8B>48nU43Ij4ko9V$k5|gJ0RQIx-SL>`N9iT?Xh8A#K9ynOfo&}tXCC1g zP4ms+<&}7SQ)lWwpEQ!GGbU&LHo2An}q0bIMFrJ~Y2PsLi}E zc!y6q+m>N|qyS8>kENDqTqFHFVBthwp7{(Caj8Vhk(6wx^gz8NX$$G{o=t<5Ns%ah zd$E6Xf!WJmcI~C!rU+*?QLjKnmx~J~hpsoj-5{;49l1FrP zO8#oKU3>$MzEz!d?g#5r!u#SKN!K4~ANctTm$(*6(C(K{Dw`WWO~{B^_s=gGh4JT& z`DRmL@}IZrYky=O#h zB?@A#rx!E2j&v%2SSppl#Q*v+L!26{kyX{4Hyp+c?!`x~CW3$^=`ET)l^juEBetqGlHLcQj0aYq(zxc|%!b*~F~x)v+owjg{;X!+ZGl(h3Y7Gxq4J z9FRzR-pg=;hxIu|q<7E2{OT)jY?jVAx{CD07!v8F8KkEPk4wd6?2E27i^UgVV z_#tEKt>~X-g^3x=8j1X`Uh4jJ`*)*XSj$7c)VhaFaVm||C6QTciX>{xglnBgaH9M7 zE4Af&zSbin40#TSci&(Pgs%AL<-Pofw?bx;FRpxSe^;^9YZWBtlnq3}ovVWMBpO1! zji@V(TxY=m4 zGn#%pG(0ul{eD@f!~dU+d2!HGb|PB7X00k9RH$21mQoab=Y?zV9AwG z5;@JwTJn>8G8^G*_-~ZU$jc^oeYu+o_5ozx1VHoB$~@vHibw!wKg-bWKRjkOkYQ>^ z-Ww<>hhQVjEIO&mTA18OWVNk#n_!F!)3zlq>b|a5l#@6%5{k_HWB*Rr}t9cY1 z*EcHadUW4c<++=1zgDeP-tSC*)+y2n5KhFY08pO>giH4!zx?igPQrCG+aqV?ALBbS9d@?V+cco#9ZL#mNn zF$5GA6px1ra+@_Y-fptu?_&HY_9xyLOOW_>7}}!0{6I# z;c7#yjPD*7D%n7G7ef|cjvr;?k*ZkZ&mHE=J^szE3u01P_s&<+1#45Ovd>|K0p^}S)R;d8U=yg zZPNOu$)T$7iW*q>Yg?OQRkqlUqp7_ z-&Y`bDKm$D*zgLXCo9rLWL0aP<~9#p8t+k`FR1#HY?taxUYM46Ktip$y>}~K{OKrZ z(KV{wl)FIv!w0CnQE?Ob?81gMskU{j*%~|JKWs< zxf3)Vr_4uSSaU~E>tmb<`Jb7r{L&guiWv9SuKN~Lol}1vZGS?ASA0B5(WW*?cfJeq zfDg3#0gLH_=V8hYh^J8m7?!^8>1X8W1TN0s1~{?doE zoqqGEBd&tCxx4(IsQTxl&o;e_xm$mM0N6fY4*UHZtF!HkEJ6ONnvVvDvPDnRcpvx6P5~ zd(vTw^dn#yRm}r`T1juw_)Nb@?-4Dm6+I+9t|^d>4PH3E%DCBfYG7kVuz7ea|4 zpAOfAmFYuAt08p%J_AmIYP~Vy$_$B*YIG1|&4w0jA;}w~-G{2N(SOtkbW=Z^Lf(-y#zZ~Zq;fB33uE&2Z$z_Zw3ew?Rq0hl z2cuSDb+LERbhAI+XbbA!-+VF?6#krw?W=DEdh7AuVVl#lDKLNocCi zktma(_)9CWSxRj~S0p=gF>&aMl6*oX*%OlJ>;M;>-|o3q~$RA z0l$!zD@1{ayFkh?ZQxOs%>I52!mH)F!}!;(i4`}WyU}+R@i_7%9Uy6mxb?F>>i?E< zkw;IdB`QE-Jl)W4)OV|p)WUx*dVnlo#l6~3OW5MBJT2ZbNTAo`zgmx~wReaOlsp|c z@XW*B>X>KM=`wnJI<;h6Kws*b@esJ|1%_KiM-{OSs}lOXFdJn`6=M$S7hZN zyHL>C$Mp705g1`UDLiLq50O7$WEK`|oSl z;|B6ge|6v6#jFdMeGwouB;HWT)W6KqOy{xi-aW2$9QZz9#*x(5Kp)dht|z<5onCsA zW*wdQ7{P@;_*@6d2-e{Oe&Z}qq|$@rp0uRz>Vmxa2-xZZ%Y{#>Zpc1 z-WAxbf1JOQtpQh)lJfLbK*3!iVX7I#^Lc)DX5YGjEICq(p{>VH2e$6Ic5AY=QCyKy zjKAU0Ie?fV9(Ua;q&7U~YIFO5x+`K>I=bWTyB5++uw@^-sh$z9?b+0Z9vYo6-eNK_TiSbA_;vm-F?D#6X381AK(S-GCTm zFj9dj(RF!96&?%A*Ol7$vAT#mvw@o<0n9Mo>{4ab=nYb|8&KwI>0K$}Mbz?Twp~2X zCy@yLVGYn+7TnOZ7A?@ezzTidvEVG=xJU^Np!eQO+AyghKoY%S@CcWS!5t{t@eD=c zB7_mV=$fpz*;nRj{zFn){aL$lfm7?%NnRcY9BO-u^MS(@_J@Th)Z)*ilefE=I6@kq zF9@=RzG2r5tsW3d3|LX+XrmP{)uVVQ zQRHx*{aE^J>PGVFfdFag{a+v^*{};`pwv&OH2mk9Lo*Y?sTJDh0{FI}PfSnv1{Cmc zXR$FTqGExGk2db5LVr;eFUo&|ld2p}GH{RB+Jm}Ca@w)Xh zzM^<+uT!+AE&YC4M+R9`v!=(lI=!O!u|y7Urhf-R<{SkH`8p4u9s|RRk!=6%w&VUY zjw5@F=)T)V(xc}m-`b98>fJO(%%Y<|{>)OWknacfm}BFICrcqYv($q~o2BMc4b_3+xXWQNT{p%nWMW&b(@b_W2+M6Lqu(D)v zc(z}U<#yIqO%u-7@%$X4arU^DRNX(RRKoAPig@Xu|8HI%#_YmdSGTxLxo+iGBySS$Ik{6L$ocZrah%z>gL21X)E`C?BHfmeV~%7pU0g+YjJU- zD(Fe0&SCAVh&PIK=Um@rr6u?21^)63#a$Zj&OcA7Z+(MRk3mLhS22HndgQ7a#~lNb zYU9M65Y25!r7GY%ixX0(3PNsuL?s-x#jp)Nci}r$x_Zx2aC@Ogf3gHZ!VP!A3o2)X zXg3(1)1ER&ZQmCqpB_}~1&S^Lk3S&@jBcjBoM2H>>&0r$S0!I?+c!UjXjX@R#KXS$ zLLAsJ#7<=7(658HD6+s}62V~;uC=yrzQb@kHgDmD^?Xa^n$eY8Z+Wb=f$0D7nhde(OguiTDnh5Ae(+#x_IVZ_T zkiHc{mjZWbGhDZiuHwzbaP8}1dC=9gseDvGR)*_jf+H_>-%CKf!6ns^Bgdii>HFOV8};g z%vmr220+=xOJ5sklCeS~lB^!IAVR{?24P3iQs^<%Rl=kDF~xq?4cU_e>%h1}3L-l$ zMuh!HVO$%!1>*-(o3DCYVLQC2P~rTJDh874&x6gm!H;6>?6=z!`17g#PY>pz<}*lZ zF8aS(!A=Mv7zHsTh^)L#7Y{9V{dyc8Km6f}t!}Axl?;WT&aSVD@(J^za1g$h%}H6X z3mt;(2yAtdrFK0`F8Xn(y-^T}P%xz@vfqikt_e@IxtNjWh8cljOy-dZgwXqM7&QiJ zLyLy3G#dU;)AU38eb>b9?E>Q}?Nr}A!^1Uzy#NHZnx}&(2?Ra7DCmV6zKZeoqMZ;K zF{=G%#6Rxk>wS)eD4N6m90=HGFgoT*oJH{(Wm4XPK30uXX~WCn&k6{P;z-c6!jGr| z6aIUJYM2sPro812QuVWm%KPuj|FeVq2NV(B8^!)fyAnQIKe=Losn{I_|Hqw2Y)TPY zX;q%vpVW-ds*2W3s7YHeek{3LXurYw*qLSPj)p}}m^{?6mQ$I7Y5XFMPT@!ShGm~5 zds;8`yP{-z-b?BG9sZ?rNnDllFe$Tr{Y%UvH6s74!O3cPI@3n=Y;!0Z++ZFH&2^ZE&?BA&kE*s& zM*S3?uVv0zou$q45nCOz@O&zP$AKlowRZpgL@Bg=L}(Sm5rOHg13xLVgN9?o{?#KB zM{i*o1-vNm?x;d<`$yAWmhSALI^!KNdhxo0+3=+nv*LinUXRTXb`z|Nne|Rsn=-R6 zJJudX=}Lz5p7DG_T7FRBxjnCz9;CJMPZ+FvNNO4=M*2uBbKm@be-EwjFM4r)Cp3zz zWW||q6x-1o`pABK`spSNny$L8@q^EToRv>Rze%cDpU{xcy>&f+j*oMaGpP~<4sk0p zj~PPO6@ZOhO%tR9hU{DZ^kleOs&Zlr(2SI`tsFw+F5*0Ud-oPn@P!kl`EDDa=yzXb z`IG!DlleI7!1zP~10wTd`Q#@87WfXCVFFJK94shC$;G0r!N2jjAFz%pwAN$2UnB?M z6x>h?uwz7&TUc0ZjoPKD!Y+$nDB$S*3J$hf>1;WOVeFEJho< zu4tAftJdG;1cugUudfMTvSt>KeEZ%2RBknoira+JQqB(G=IS)LC%v}rIWfn^_U$=# z&Ond!cEmU0)ZoQBVula*H67~?KUH-+ZD2#paG;E76hfr@8wShlH2Nq?Sg=2y)shrIEd(WHw4oTV2-7!sTWm4HPY`GT#Hym^;*gjVtwQMNsPG1r7wb_7t!}s?pusxRbtD1LS;J$g_9B2!`tB6;!AsW0u;XLrM9=2yPOi8)SYwZ38RHq*xTKZ_orXk!G z`k^298`gL@=Nea)zE~a8^$at_-6EI;S<&`lbhX6^>JSR@6Hc2JvY6L;AF&Iog-Fc>G6u&@`*6AEOrEHZo!qSDH${=XhMDG(-dyE=|MEPQVtE z=cyoXbxv|AN0n31rjUX3d3i>2)i z`|>zc_UND7 zd;=Q4zrxY%=i3|kmw!PFxD7s`-DVby&m}}RP)pA*T3wH@;X~6}T}708#Au)skO~20 zH9pP81K9oVMTXt_K+?=OBiL@YQd_eGLydaZ)A5@~GtZY04e+ROwFmfzj6A77-Af6Z zBJH`?T+vboj;vg(nbjmA3x!K_mm0HM0F=bHaz(aaa2Znhqvh~P41_}EXlHYZ2@DZ& z8_1v>dxqO)CrldUeN;0N2pEO%R2_AjRY+q(DEGT#;}*r$sO{UItjLhN)YvQ!oAWB6m)H2 zrV_@~cG$Vk_SUD0#_ZUOZ=bA~Q|0$o{MSR!<5kbhZQd#3x}@X!v`^WBm2_s*IE%V1 z-K=%Q=Hy^=DXWsB_d{BcTzLEqY&vA$*p#9)bd>+!o)`Js2o(;CBRInQ1bW0vU=7`q z-pR4M+I%C<=9Q=Yz|iiW{TEPGQ`DE@8gAi3bhLeF4D#&7eHt2$tFQiCJkZ})IE8+h zraM)Ov+s=p9H0)+QGNi@?4{>8QzYN%)wNmHX6fIFHmflrK5huLtiRCB#(Ig5_(ZNg zTnLbu8zG7rCSbwg(;UiyB=aM{-F~vhlkG%lfadlr6`K;qn?lj2#SvYZ7e0)XV_{Es z&=<5sD&ZYd6*dq|k=>_$b+D!kzF(^m4U15>pR1DqH+Knv5X6HgZlg z^?_{+Jw{w|<%0l#;w$B4$j`l$IyqYMwe(~!rdh%UNfv+M6C%P@G=l#gd8me0F2kBU zm~rdWmG~Y4Zd8CoR0Kvf{_k%*Nw`d=l@5r%0q^JM%#I98VgMM6<+Xjz5B?8dkQH4A zjbvIiM3s&6K~x||o8`u9`VT6z7MBU`-pY-IWo^9NF0XJ=AN3Bm#=<#*qroUmzTg3w z$EX784l&v{B<3;tn67aSJid(;cY-*x^Wk*B3$3h6s|AM!WMpz3}Lp;A=>D!=vU7onm4 zK0zI^f+&FrV@USKvpbzPPiq>WEqp-C#DfqB=0tqGoZz;`6BGe*Z#ee>mrn{jLHIEH z_3PKAYw)Cl@cUfCk+Ugs5qW^{9tZY{6OP5YkriO%lCFfrqZ^=cWPqM{=25@ih=2aVtZT2aJ}Wfsrt2srWZRd4gE98o|1mwBttvWm zkB)ZMt`UYM#IPk^YdflzkLEvouKUjtaXHnWsT6&Izz{=d0NvG9ylQ;%U!j_6=cSrW zTRVf>;t-g1T(^&kqccI+X!Q z8o~VXv`cF1Ma5(C(6`HmOM2VLFYde zU~?v+uWzmpHJ3?kjk|=jv5=DXVd-^S8c^kE5=qomg0GH8SY)iv)In>I5R&u-JDX>g zdx>uH{=qXJy{N$}58!WxXdwgnQsL*SLajnNz6 znw2IzVk!98&~d97|CPHUf`1HHi70=go)w}gH0R|LI`1~5iE$34FjYr&z2JhVTvBJMd? zE?v0BojK{=7C$XxO<^z@^6Y7Qma`*l@l67CQ>ja{hu>rEf36( zi~0e4oC@4rc^}m`_*#VFB+HRMa8ShKZC+}c3JHS6XgM!gVPtxDAoVGrOrT9^el;-5 zU4p1%bH^ap9Qth@Gb_-UlhpkacKD&)^zkZFJCc>{Sf)c!f-|6iu-OD--*9N)>?%BZ zJ3tiq9!2KJeuY7`3rPQ8D0z-CRJq03*>gn3W9-PYBW_2Gt+#gZWqd*OCM(q`qh(8G zTK}8Z)cg2o+WaWM1U*<`fkSfkd@k#Y?C*zAu8^C5Spu=%>e*LDXv_3)XW7ol9<6XJ zVkc4M+}01`p;Ujcn0fdr$n;`ziN~wvs5H42Th``#(Qz4IC3;M9G-sZOdJ^iIG?479 zismq=+h`=4v1Il4cL?<~utb#fZ3%tZ`))6=js?lUQ0f-jlURoBLBC zH&>SYz)tEOSVtH@GaG8v@*OzhF2$1om?@MuHO-Uytq4u`0h-V5pA!B+smKM z>`FZt+~Yh(3>`#;b@<1-8so^fCBMnRdS8n3v%g^ZsQ^Pzal|q4@vcV+6mp+A8bijz zy@^r>0RZu%kgR879v;r`A_>WH)r!*0eRLQ8m9+)^MxYXzH+^9fv+)Q!(fh zHXwnvnP+4X_(HZHunU)pZ5DYfu#6S$_=~PO`rqTTc}rgEwyt{<>&v!e?y9(u+QsW` z91X@{=GpWZ} zMl26&LoYqN6ihxO32vsx2QutwVb>B+WKrV)Ydi)}G7^z31h@(XEnslY`>IH;)M?n9 zGY-SL)rx~wkvE{N>8hVJW*&fezHibrt0(>b*E9{9ADY0}OVX6U)o=E%vpzvZsy7a%*Fps;^MJ6dcZf8cBFhKSgc~O{B-HFqVK>Q~UKcaY0;T9dR$h(B3P`-YsJS`&lX($Z|H5!D8?LxG2KW*9 zDgjBN^D=k?sp1tXQP=zu#6+7?`gM7OVM3#bBIrJ!pu$$h&VqhiZC*4QIeiwjlR33l zrO%7Pl>cjoyl$KiU9Ku1)^I?Zp_)0a8xFfjus&huK8DZ0}|2?s7Mjj zzkY4%ljbv<{EXT}_S*Qq{UW`AQG#fuaAXwqMELvP_e#xm?OD%3lMm${rRb!dd4=?s z0Rq1I38kvu$CroCp{6b$g#7fH^g0ke)CaAE$@YHg_7=m(GFUAzryBG_3F*gAY~qWo z3pnQri7Oo*LzKC*y91hU^6$@tpORwFZuG~&byQ!Z3X6D`u~-q%%?_{qI3koe3bS2qC4>Y)8X|wKvTm zu5lhQvpD4<>?qbk>;!hcfp$==^QH<#xFOvWRpzjA7CV9wLw&n7VUAKJqb&fk96mh7ea}nR$04So| zmnJazhuWMxFQ(`xl*ps<4=q7YsS=WnIt{1rl7en{TxSFE60wp|LCg5-uYd$ZlwOn@u|?10ZoDslzHUe_Au!U#G{4gK2MKGSIDnHTWq9t9s8Ov+N7s;8d(?h{YLE-X zApoxvUJ`b30Nt$^=?Tz^r>5~?G-u&Z%Ttu8l#+*mv9Hgzi2Gs&$$}ZaePaxJB;gG9 zG0&(%EpgWejsfAz)FcF9qONsll#<&1|AHPm05jo~z;e zm2<*p&730$;U15-B;@;F;bECoXgo)-g1dB@p`lcVC`2U~*na)--?iuuI0!g}Y*urV z3;S=PJ@f2g(y>rCY1$h_w;evCX4#ZFsl;LdhQZqJF~(_a`EKKcPOWDDK2dgF5TAs` z(>|tlMc(^0Yt_oSvOC)N!@8$&3~^iqqvaCbzd=%G^n-SUB%rOn$smxd40JVv%UR~J za<;(f-RupK6TE82(W3A>rBq-OM)Ac$J4RvFjCleqxgLSz##^8o@lZlNsV#Cx3$r+C0hPs`=n@jxWi-ig9^Car~ z;==pSK?7tKo@0)XH}u|zUdAa@A|^-cjcGQ8EWri@kObaS>7 z3#5p&+KtuUx)02;=!j4W$z>|dsa=_C-A+y5y5e`4f;9eb!>@Ktpti#MouJ40^gueR zyHJcm51yf*I!7nt)A7Jdx#F?_s4Y@r7a8`8uCG>_bpsbx^Jc_su2AxJ-pY25i9KIL zMyyBdGWiifg=XZ~MpA1i(D0iIbz-7mm?eR&IrFuS_&YBgdI$B>`QJhAkgOwFE1+6$ z7*t^*7FSe2(ZlM5TV?_Y@ta=iE;_-iZkZ=2)!;kXT&2onv(mD(%@yZD4 z0DwvZQn{{tL194IaF7N>)eX>&{NN(+21_0w_XSki&KvCYK+^F=6*TAQ;C%_e+8yW$ zeWm=DLHI5mh(5)-HN>GezVOnsSvfl!X>W2%EtgE{kCcpS)j=ZBxr!o1HVs3_Q1*$1jBN@z|L0CTd+Ib zlyj*12arf%A!2sR-LY#=2a2MhrRR*3ggKlb1MM>zO0 zOAgua(Q5Tn2R5Okd>$7smoLDwEhGRMZUb-N^aG6}1?XM^5ou9nd++OzJUHP5FvW|y z#LOPCir9aW{-hF}0h-aGH_x$xV}3kqTxrxT=M!l?#HdbzZXx7Z+LNeMvvRDt@4ZeQ}`f=C~2Cez>xf>6j5vf5`3_NKogk( zj^x9XUXyjTkQ$BuluP?;oeE4-1JJ5L0TvB*4K1ju9gRlTK(9k|3r#qLp7cAV4hEbs zpx*V_4|^?2&s@`Ja31u`K6cn^hM+h0zAXmF9?mpbJDvRX>30FWNH#rP@e>^YtA^EJ zLeTWC2#W5ol-L2uX<)tx>$G9Vspqaa$DQ@Az|#cKlh2kCl29w0S< zA5VC{_$4(6U++o#){LW@{x`${2VHoG69@LaJPEW5IPc*O?I3`CzywKGrZKSo1pK81 zl%+DsMx@c7fcPb2CGt`|*;@;Ae~m$kZNJ)ylw$+}@)KX&iZlq^_h3mXtDD`mUvtC( zMEC?#9`JQg!FP=TmH=abn*hYWcB25Y1l?Vsm$Tqx`s<7ah;{%|fM2oGYGmD@MgB(w zL^)HTzfUQi3q#dWK7hQ2HNM7ZoCPjJL=3)Uj21}nEp7b{z+OT z!dJg8^IQ&f6f1<0q_6~9aZFfD`ba@bC*f}e2_>VWF~APKsWA*$UF_rovHO}>-OHgwmE`633NsiTbK%&=hEDfjT~xkpDVN73NITm{*`Cm^+at+txi z*qtfbDsN;@@B=0Dhuh4NrP0WFHidlbXT_hA z5Ol)zuYgjL5|yB92ZmUsa2WM>ulnXO2cv+0Y^jUWt0h0iYxkZMsJBJDhH9w9=1Wq7S~MOA!qeokN9=(c85n_( zl`EMbui>;y2SBoc9)Aww+cC|`^}kl{A~we#qKATgVvIG1B_Cs6Q3`cpowG2pu}yk* zdA_5Fa^EYl#-ScYQ*6@LyuK?suk?OIpS9mmJtrW4uulEwe1|!;>0P&ntX_B6xPF*v zv=_5%)Z|7uly8}q6Qx!0uQ~k6EOCSWqZXx68fb!Chg|cTJp*fgPpe3g6d!r50ZWz3 zwVsgw;yJN|S@L^cBu{zETd_z;llEhEI9#jzLAMA3vLrSicR_z6UzJWlIO`rxcQXfa zJa8QgH)8;(1Pa8Z0}HUKb~<>z8O!GOzE{*Z_MKjk&TCM$+HbH#i02?h26hly%|BFK z8jm%uf5E|`zBqk_ zcRIo?0)-y69sJghUhK9*z(FkX@sXTdAcOpPjWQb8J*Pyaj=)_h-2TeB)J$r@LMNf?{lNR)F_8%Z{q$p#RgIO|Sb9FX0{~@8XfD9-D zT2DhmAWSUUxBIQZB30m@!U0&eWs{bpDi>VC;IAJ=ujJU?(VIAr9^dVA+ey4{nU(}t z($48B+6vcHoefEUGFLEE-A6c7debI*%#4{eq{=l&Tp^he06?+eVN;{NkND==JMA@> zcCNNhkT5153Af)_Jy^a?TBT&&%obUC6fQwj0~$>AH&3NN-I)uVqgD0dpUSz zteP<5W${^mP5HI{0j@ieh1K}RK=nGnx^zeH3;!T?T4~mY&1IcWGWa~EizB}p0)S94 zTNL$w(nOrK5l>)iGJtZ--Zz}aX=kv*>d4!z_AUaq`l3(sLbu>f!?4DneWu>yGSaC3 zyZQ~rZM@Zo2$P(S!F|9+mLAWA*q`qzpN^g41J2Md-t~*`H!?M;nC)~=@E9oRHEWrQC|2Nk#%o5Mh5x(f z0C@&}&zbZA7uljb%GTjt60cutSS8Co+TF@5JRUAoY&=D#D!51mK?#wax~ix%is(eZgNzHg6-`zeLvvEz4Z3s7I`yj z#!isfn^dNv3vA|1`Q7=S5jKpQ^u2FVSVAwvW5LSARv`o9G5GI0|6l}R5WNQ&`v;DB z3}F62_@C($sw`G86mZ@uz3<%}l14sNi8~dXXc%G~nX_q(5>Ro$MIt}~Gl)0wGi4}G zHHAsPd1F}jvvWrMtIu)F?}iqGwop-1zQxl}-!JW0fS=z1?!V992i0Yq3+rb#ZNQ#T zus(`CWOeid_lg{B>PQS|o*-o$mefCz{4837Tq}Xu1$}5NMFQ}SOP#{`2)AT70r@m2 zjQ~#L3_9hJ9Xp8dBwPrLCybA2-vBpYEQ_%;+{~|GaEnHzI>d3CJwQee{>W*Rj)^o% zQtp?#=Ilv2c}eS@-x8sWz>iB#C{es>wzD+kvwKN}GqYb5(yz&MeO%N~ zUvFMdm{n7A6i4Ep?s)b`Lt6fGn_S;=igx*FnKC|}cj+NS-A88G5t&m*VVJ1JVsoHs zG8IkT4oU@(G>?OsEdMz*>EC;i9oY!)l|3R|#JF5h`-oa9fg$`92yq1R3 z@VxIEDS+%orBuuO;Di=*bC*$GRIW)cq{)-JOOcMyUVR+kuX& zswI!Yf3NxuG={Z)Wi_7Dv8#9MT1eo|D z4eA66z+#G}peTS}`$qjSx-Z`Nh2por$I- zU{Sc#Y#C6G1a&C@6EOCdTILsEO2Ex(%#3Z3rf=M=!wh&GRj+(_QQ2zILD7eS&mszv zk^gOPg;apjfP=&1J2ECQr*-5Ah1ZQ^Y@MJTr6p9yM{ALjC-U*f^Pz1{^NbomVjkEW z%~ZzP=KoD4>6>j#0nGk^qT{XG)zJ?(eAJ3J3x?prQu51Z1PoO1oH0kB#m_;?`2-%t zZ-%DtrW7fT6mFG7tJ7TnKW|{SK5>wJh;@-Bb6SUdG0}-D+>} zA@YaaKGt}!G2*lQT+>kN|MoomI}~^mC4nJueQ?`S9r-ADL^cj^ef8hh`AGma5&hc; zKWGGQv$)W-3w<);EU?6p4#lAaW3yS39nT#hQ`)rpjEQXh_LM67I!OP- z{t7vU_LM_(+U|e@xR6*e2mamIA>4C!Vtd=&0go;{PqI>{Il)HX4;sEol3} zeJDV1MhqwA`7#Nth`q_K*Y@afoIlhPzw}{$3ECO%X&&%fm=B*{N-=AiWC;vMpN ziLSn^pDA6r(f^r+-9`|=w*PHo`R{6gey0z%ar67WFRNl?```@nTln5DJP!uc6nOxn z{w8i@it*oiJe^!#mHK_X)|`V^c+d}Y4c_?=w^(b^Q^AJ1T_c=n$}Y&mk407q;m6bzAxU;$j4g8HhW5hS0lBM-y-4^PB$ zNcSFtYHvUB964UrzwdSR-e(J7>tgjcl?dc^Y92R;a7a9!Gv0w0f!<0 zZVlJwfCZb))JD9fQS2GZ;%}7tLlC?cKV|ym}}B z69*ND!1LqZ^Kl;*Y?9dyxUchX1Tq#Sw$7Qp_6Ykj_N@^McPa(Ur`GB@Af|EK#Qo8A zlBE1uqtxy}sDvQ)QjB`Cw>beBc3Q^XH}DUuh)iHP9Y_Gq%IXI2&}x5ml||_=EF?MI zmcs8qqv_c6%G*g@Y<^SxUF)5mxX42d#;|E(uj(PY47k#6WXn1_CCufQRg5;t_xU+K z)z3uhyxBJl-)F$@iMC+%RGf9)(+v#_1V48Tx`mV<@FGt*NN$#GJkMGPevjOmzsx@D z{2*c6?h%)F8LhD4FBPVIeI-|wbfCDU@7*&4UbPRwK?B$FK@4r=7S@QF(TJX7?fh7q z*Y!UgCq^5-lf73-uTp-r{B_^pz~6H6bZbfnt!b$fBPrOTorAKh;_1&H*TJiE=&pwjQknees_&NI#7}%wS<=3r z(h1-EicO{w4Td>9a+UM>6${+GXg@V(fBvPRI@dg=AomMX`zmVYc{ZnjC&%;kO@4*9 z&($=4d&i~s9*X`oo%@0IIB*uInWsKqm*>_*jIn2}hPlJ6SA1qC8_(RnjC;|PDwqZv zz5DH)-b`n6NkC3NcNss$!$`}##M=WdbHko&BE@_=kABti+-Aj!pD&5|l-)7$sp+}T zVh^3~R}xMuml3q0@5@4Ts&r)F;1LX{Y-8ZyPMF>Z-($1UK&n#BRFSjJnb|EctGMs@ z>O$S0L4zpsc2|EUv`MjfvM6%-^gBGjTk^V2VGL@##8;0td?W$3i)~)2eoM9bTNNM<;YvkbdQ_)Rj9tP>i z1*FNXrZYmI-B(>k^54=L5h5u%?W}nrm8z9!!}Y3QC43o~nOZsZZ1)PN4~x7J7-c`I zMEQ~w{!_MKcbgm$KWQ0L-_8qWS<9Kl5fI8gx}}toWVv=7Q0#5!N%Rwwp#8tP=N-WeTS;HUF%t}`B znD;p9RV-Dc4~3cGQ4OmTIQpeAzC}`B1~=2i|LTkWE4XI=0j8=tC^GMViJbY(JQ*Su zk!Ugd$NYh2x!F9Jxw1s!Sib1Lmb0WrlRC;Y=LzHM&boM0x2LNT-9l4@722= zhb}7*acohk%L~t78ewKS-H(-#MOc~oR3I*LZG8zd3#4~#6}J2!7T*U|T@q&TQKq?t z3b2K|!`i7b8p$bz{pse(G|Rqbd;ZjHCnydmoI)n5<_DQW|LoM)9X-yzDnh01-M>cCT)V_u)N}rhfb%X^<4+K3EaNjV--zC-oQpmh` zxOR3v`=%t3RP&cw%F(IS-|Qm78=s3|#HE3P+x4F_-?LLB{b>U?aYA3FNarwoO0^cH zh!Dl;C!`c*NrL1_1uDjfi~STQJ0-Cx%N5#t^qg_XoN&0{bD`NhR`)fzh%M;AERe!=2)(D>^(kxgdCDv3&Nf(_lc)y(OlRDkqT*(L7sYK3Qe^e1(O z9*FN@y^Smd{RZ8o7Gi+q<&ukq>pB(vWpcE!KJR7Ra>5l=8S8L^sQX!yY7jsBu*bB= zJC!L5M)AHL67fD0`ugE*S1TVYpIZ{!tkQoy3c3~ZMqdQ`RDLwI7JVF6`!BLuthT2(PLx;xks@X!}m>dCDG_KlOY zsX(dX=>H0gRdE$2Kt#*!bP#1!HZCGmT1CWX-`A;2MGx``1pnsOqnc;`rs)O}WRGO* zPF}uHLn%QH*Zx^Be_moGA)}Z3-R=1Gjb@%%1v%S8r-`a-ScAZ+BXchgyMQ}{>;_GQ@GJC*brVP#g=5#I;xoLWqp)07RX{lKkI+ALKPc0O#m93#(g zyVnVgnIttz)0$J+I~6a}5x8BtY}w7k`CA&W;m61^=oQsQCB`CorFrPavYC$BETOjG zs%kHdS-^QVrh1lQnm*4q=(u|KR-kBxU&P0M|1$F)R$-c7KxxWpCD=!7+wN;W+7Zkd zQ2h8Ou5jv4Of<|9*_`XglUXT3V~)!Sfn4biJ{kV5fzjwEtN0vetrF{zTo;^fr;ChJ zjniJ@YBYxgH%morw8}rfKg|F2sLjm3^{$?`lEyu(I&78ZYzo*v_qyB4^ne&0Wnfo& zK04a)3lwi-+qe57yBl`opUh&C(Q(-8A6O2xX`H*O<5Z40WpZVdtF;YsF?g);a~K3z zcDx?&yZ$9wP-%Zwk#H7(^{;#?VUg^fl?S!1|7&!KTEn-u zqb86uA2*s3$ud)&M#hhzsIywU97RXkb`|SYVKQ%+h1nma-(y$8f!hgro+0!|o;C44 z{WGuc!&`0*6>r8}#^}Ho<`8qkA?;=*LD??%UNvF$j~J5jZDa15Nly5P ze&gL9W}ZDfW9*&+^{dOGz1?If{>^@3;-k+EU~eMB5>(av*xz1F|Jl{7w0ve!_3&`S zN@l#5H-Zjn@;HRqP@Q@=D>)}lTxE3K(ELj!zPa0TU!>H%U(-3CuKR`Pi`FK0|AEV+ykc0ivEAj4i+AQ<%K7Sj$gflr%g@@L>P_1f z4he_1X=e@s*CO{rCoU>@t7e*-j5h}kEnH7I-rvdbyqJO9+x`tvJx_TGNz*2JMM7mH z<8Davm{?}(Ni~BMN@0Zk(8EFRc33si%Ch~yjK4l0@8-QG;r^Y(&HZjcYp70NxVA%G zZo{>#jL2~m!eS?HQipVg)M^Tbti`=^8jjD2Ed7=LX-%$g+`0S#cAKpFOC}w!r8ScX zBExre{+LvYclS|=gkd%i@gI|l>(#O?ry-h;=~f&EHM4N7-hh2sR7JnS{O`Qs!MLQ| zAVl0>;jGpkr~$Tf6h3W-HT~C1Z+jk186WAXAO!LX1XMPgCg~y$>Lk?_rwrDxy;MxT zOrf5&$BQzsHCGzWSTiIt(R${N{=WT#R?#z4VP>NtoKuAerA3f8wxfmiBF4CP*h5G zZyF|}jw02FX8uNg8isN77IGr`If2d!ZE%_keY?sn$9QYgKAxNYIGNjaL8$bTAIU7Z zP100oxC~B$LdHM1t#^A_&MH(lj`2)Pv~DWH-jWTyW`wF4>CQWpKVfDQng+t+@L$P(M2;s!j_p|9~F<*BMD*Q(VR!q#?Ks?Qtk0n za{g#GgyE}=uOg$)@ z%rLEL6^DB0`2$ba_8KdUWFXE4b}dB6Z{Hx>quq|^=Cv_}jD~@09?hWTETZ%DTH0HP zFzG~Zh)8CZlHNEF=`>7-!1L}p#T^xypFvwVuKNwH?S#NWRvQEzji)tp`FU9VpOL>m zM^=f5iCC2fojaR!&D6i!+M}0k!n~jP~fJRj^!mxan_PUPvsPdt~g^j0H<5 z#Lk$OhP$C+!mbX+Gi&&sBN9tMsoUDbP26GaE=FQ~H(fW&?rxJgBl1)3o%ASStt&0t zGiS5|cGGJl*|@Q2s!G8E1(^!FGgfp}0uOD;c8@xWQPIZF-YYeh-Pvt!`nFoU)=FAd)_`AHKZ^7dQw)5xLpp3G<03_Bu?^I43Z_5ea)eMc!Dq%rjRs zsQUrk(7V^-wqHv5zIK>u3g1+hVrJ1(^`{G3z$I!EbeqF9pwVpgaQa_qLiwH_*mP)Z zCwS*qL6pRM20aBSVXaa1ggbNV7bpdBaRI36>@WT0VxEc+kFa_FYMXs}X>m7H$N!6( zEZJpUtncJs{NR+fbs~3uW=?A`*E{+9m2V+&3?p(^A5UFS)q4WXPupS+>Rx^t4|VVU zfVdps@nVr;qP0zI+@@K(#8dY(8PxFmDEC9A#%q>kK6dA%^Qu~jvJoARQ$4PtW6Fm^ z14}iScz86utMEOeMvd4o@0Nj;hYR1GA5Qg$J@!fVf#*&0s?ma`?i)K&b3x{Dy`MSd z@2go|NL&nRXoT6#1~EN$6aY``&G}X;;I^xqQv~9gou;)1A{!nJe**L)RA>lKbk)7(tCTA-S&I6jG0Lw#7 zcYKJZWPnty&Tp!5dcH+)`g4Gl?fan4PymDt==$7K0#&1J&dD1bxxVZLyDl8cI71ES z(IQ(+wS^1KG3!-Fg|rb8BK83JG|JWR?=KKfR5{qI!oskpKF3>hDZHnZ(?m`kkNbNh zl2zi{LZKU+?ul07pkMk z2e3za+<$#eAwTbA!Hrb=Af$Foz+lhSar+ooRxbGwke?;tf> zUN^xriH#E7V;IB)Jc~-d?gKg9v>mhshW7MrAP#n6McH#hp~{7WuNYf%L`TJvY8>W1 zf1v#EJFyF@VZ!(*pnW#s?RTct{weQ^++mYIWMy_r)tU>f_RsGOIFNOPA~#S(S3iyX zAh7KvQ|}FlSk}C$(%ML%A3<5l|BFp|M&%Gfc7pP%@RT- zTxb5Q#|+nBtFN1qjF5+lZ(1~ppOv!m6Bdd)2HxvHFTM%>>qoI9?SgxT#6bkypViB! zjOOIXgsvYq>h8Qw3Y?nxtF&(4oOi9Aek{v)m*EiVc?Mg?jzuY2NV{1s!z;3V`hvoH z6xXc`g4y=V&T_4*@@C0_Sr`}c2wUEdm(i@9adbyqw#cbgV%I(LmF8Suq5ZI1v^=Z# zvP1aQN&42+VFQ=Jn;w|LRY8f%rW&&x$kvEm4U4n4d^VX{eJtW zUiOi2eW)f}ZUf{CJ2I9%34DdkJ#Xk$w07Cepwz}{cS?ACHeP~*2@M5S!31oCGM*tH zGKT6qlRN&35U#tGU*lO~2!BrerY34qi}fesEkyGupqAP0;>*90t90>41KvHSd67Al zdOe2n42%Hh&$|BC|0N&G_C^$b zFgj^ISD!}nI$G-08EX)Ddl~s&Rp>mpi2HKf#U53Cj=9v3;rK@DD5L&CI1|lr%-uAj zjG*XG=oz--lbd(k7EHs?-_kkLHDhCHeNm(kWOI;TyR z=`MpTAv`)F=hsy7CGpEFNBQ`aedlc)%6;p-&S^ql-X8bZ;OE(ug^U_H~vX+-nwAwaa;5 zGH42#ZyCgE9(t2iu87<9cH+<>W}ssuVfVqJf(aEx`#X!8E-vDiGO!IQlk0xDdreCd zHD$UR*H$-^X=CtiW4(z)sk8DCp$h9LuV+BJ@>!`R#d^RjctbTw>}nt!z zAYMtt&pr^6MQTQQEHi7J3N(6BqHm?BiG3P@s`&*zMJsxS!6~^{ArqyUa5a5|lNX=- z^$LQa$U$mA%0O+hASkaBO(}G5tG^JaLqO(r&TaaIlY$qG6y@ohYW_i;_}xEZx3dbC z``t;^66yIxAT1V^P5kH9oZF(GjS@q8KsiF9Sh|C-KYhIM#g zK~CP!Xz!nhD01UmRa~+asKwe_e;yncC*Pll`fB9Y{PsDg24J5-BW%c zw=1;G5_hBYv)ekZt-0X+2ObTJYn^H9INzYTiXfZPZ zwd^GR6EnGBCQQ9h;=K@W6&xy~k<$T;%H#St`7!i(z2b>2w|j&=S9Tmb`i`q-w67F| z*~z{?al52*ifFIDSgO=@$}nZ^by62haVQKt78@*DkKkU8#T0#&ukwvZNBEY1{iW#K zE5@Pm>Pc!uKI2jX5zO$2c;oJM`N;2CKMsO;@BD&vo!xpD-|MQv^Nvp1Vrvfc@td~X zYvsaa=*7+Nv_r4F`#U`aWCzAevZ1~$(`}n%%wPdlvEG!Z);pQrA7A&4eq#zJUi9|x z8NO*mrT*x5lDRsR<||W*Yp%^|^(82fzrtuy<=SoWyl#hM2^(L4 zllj&6@9q`{sUbQ670=Sf&O#@?T&J}?{H!dKad=S#Qcu5!-~G>9X+sPNB~-nYdE+)2 zQbR{W$;zlKs7NB?bQWupsud1%R0KfkCb^6jmaQgjhJn!I)pm!F{HUFS7o zT*b;<0c%=L8!BZ`O{+rrN@{aE

Y+jP>&HD>c-p>pHtKF6>V{UDYR*51|! z-z75Hd)@P=?Hab#o&0ME3&OT!X z*Hs!`v}d@?Pz=fZ>qpWh1oI?a%{#GdE#Y_7d;72WurTUHNcqnB9;*b-IU#l^5|1>` z210XA6QVBh2wVj23@FOJF64RHYVr4>DXwoD(peCS(syNT!n!}X_)_8>5SS@Mr$qM! zq_5I{<-hWb!$F>V-fVhKboR@eT9iJFd+t?$KvHxU#F;;O~QHF}dx|-w-L8016 z_(Zv)4ncb$(xLQCw?!QtL|Q93TD#FYSj#TrF+y$4`|A2JducCm{yt|Ds}xJN87}~e zh_VWcEXZ9R{Io)QI&CWDK88tJk6@AQ?JqGOJ5luivbF~yyzSyp>bGxgTq_DjoI*Hq~*qHBdW=dMS_Ujcpy>9kGd!#j4y7I z4D=paCkbSN_LyA~=80oVIhGjk*ySd~xXrh94?PS?te&ntke2z3Z;}wKN0cldoCwI= z3%(xjDJSmDKCL1VD8fB^pj(lq#uAMnPFb~p~9 z_gun=&r8v8OO|4&1E8IC=kK;Jg#v~cZ>rN6cu-$efH_mqa|BxdVo%5BO>XxN{{rMkzoniT9xB$>MjrM5j(62~rN@s?7Df@Os| zv7Y)Pm`>N~+Si8fobs@bhr1je35Og!<)!E->T`B_>E0&=HBN+Qy-Ys)D*-RluKujf zYM6SAP(A9gl3c#nD6vK*6i&2dZ=d~Z`WwM(d*WpamVTs<#qhftP6=Ii{WUTLlDWK` z)mlL(NSUsI)>kL4e5G>B-g8vEC|XA_&(4auVEJfaAR`BqjhlK&-+8&Hji$J^b+A{tr`~pWj=xr&<#wWjg#- zblyepOK38>47vK(UQ%+A8mV^)JwXh!XvGfu3=Jg}IG)0{)I(CKQS*zDM}=XwelkGc z6D@}FKiVZBy&n;$?X>q@u3!#~#pn-=m3`ZVU}StWd`DHj`L+EF4?%MZk2*}#lWr{x zO5M#V+h;;eGZ(F)>eQ-^y1ueayw1|vRjpOF$7tbwDHXdi-+E^}lZ`0T71cId5TIQ% zvFonidM@UQA==REvYb~kM|a$gkOO<8eSUd$@Bb3z72zF;Pi&W=wqGF1fURJkFpf|b zVi*Cgk~zL8;y?X}Bbdzhe5bn$^45vb^FR&`jDw zJHqczJoQ-RE2v3MG7!sD90bW`tUo(>i6FkwkGDKlkCvwM!>Yq;*@wD747}cI4o_xb zPC?;O2@+3xzaF8+q5F-tPYiOd)Vnmcrcc~Pu6Q=jujj-$$J*=tr)q6tTIAMe)liH1 z3Ut}euK6gmqiud3qVP>n8r6`vUh}fKD|R_O^$t(i=<|y<771If@{QF5cMdHsWY5O? zF0)Y+W#DA{B;lO%{6Ho<(Y=H0DbA|GVKpV>Ba(hZz&=*Kw$9dG!*RR8 zkLSZ8zefDrry-n6zi5>Yo7@O3k%>6M-R|eEL;YCt>HVC#tTse92A$KWB`4eDe2$9a*J^zyX?jBu!+sX8+5cXnTgBT!Ia+^6 zh0&3mtSXiXa)8=mDf2q!v8rNY^VPoiDdQ8nT7J?eHdh~zK=s$u(J(}oiGtbr7jQOwYLVasE2sjl(Q;Snm9oKkupAh^w z@0jIp_52z7T!xUmuf-#0YyOe_CieNT+Y}T*SjIq1ql#jRyJ*wWrg0xdWeAwrr zXpJ?hrfpvwe|Ma=`92D(Q>(r^Vig+h2*IdG73w=7(zjq)B2iRd=QfH&>r?GRON#Dp z5lmj0te5vDw0^L$E&RoiOLPxbTX)e~foUd)wIN%*3R7+KK_5MS(iz?SyEP<=Mn`e}F*;c1sP~tXcUlx4 zz}Zi1RvMq+y*u=}3)7fv3V_~67!Fo*Xb~b(7#5obqN*2<{S~yGfNE6H8%$w@hAc}w zs0cJ$xL7@bQ7}LTd66mt(y;C{S=x>wTATdt9Whoo!t?v7jHGxD;{+y6sr2K^crclQ z|7e`$#J^XPSFm64|9<@B@!wq#y7HXXN?KyYLfz*DO@`xX;%fT4E+w0*!jXTji3yUWC|Q&WxsSUeP(=K zH2&{Rt6u*D5dWJx>?FX23yhy+z$mW8l>qXT0B9Tno=ujlO0j2bzMA#qQ+y=s?x2o< zw9se9F=-^fleDvVCqkdmeQ51iTr@Q}!mDoe!5{LWe`s|UrNS6zHrJ2&&A=z2xkGc8 zX?p^@%D3QJ<>x7vMcGPAij@_f{#btuc#AS%R;Lm0MzsFiDoh4|Sp?5ZJ96kc_H;HE zEp|BmRi7|(K)o-bXiOJ!fsOXYRlR7##p`~541P7Ki5;}oi3&PWqr+w&*-4j*kkz*o z(;i&dm3{a48;aJ52&gh;Uyop3lP{pT`#Ir;AH;85u`992d6NR6xLq_bh1R=__&?>F z+mkJA(_&1-r@TEDAs;UZ3{WLD%P-tP?2PaGjmglZv+r^iHBphZifd4;M2e;*wtNYz z7wQ*65rfJlBO}Pn>{;XEh~=^>)C`h-YABeG7>tZ|l5$(~Tfg)bv-+$R1OqoQ|M41h z)<$<()lPo=WWccmOa}LytIS%T@3i{bfH@v=Cf(w0jK@(!yK ztL{O}v5A3VP?}WGt?X-bH%rLitig=FNk2VAhUewnWs0e8u|)i01JNbYB%?V)^ZtvW zG|h4KzrrUgQ7MjuB7p%g{Hw22eBMS}5K+P~eJvrQnw{pYSd9Sac_^lO51D955yw^H zbiBwF3d6bg#FIx9ZoFG&>&C;1HJ->HnBS_K+9Qyx{?XcKHRg6J?~X7X!6H_iDa(l? zNFaMrm!M6;wbqmtLDwgp6b<8<<@9ZbZVqvTyHsRIz1?|c;T*Z%q>>r0f5_t*W&_>R zO*<}V1aisMuFkC6jD9=fmjUhoz2wcjJ7$!7YmHXUgju`*?=&68*`TO1VhATT_E^cT zZ1G`jQuI(*7j9`E)5;R~Q}A4^h)l6WSH95FS<%_#W8`z7LnJhO$}k_I)JdNNOa1To z1swJ|J@*Q~!HlA=Kpb>A!?{7o=Rl>?Joqc?m41xjRjhH|xI0cuY80B|#r$Zmf!3L* z#47Z#&Ew)`H9F;_HA^I1J3{oZsVZVasP#H5`aY<2btG@M$|K$96SVaj)ng;2-+h&+ zQLN-9!qHoPaMXs0N>G4)w|n(+eu(EyH_GfCU4dvHkoA3(7r1u>rEdrUB+SVY2*(!w zJ7W8aHd&^4Z@Xco;l8{|?`$fC`Zu0y#=m`jy&{ZDmBT~gs*+m!g)yI_S+H?|>0?hb zH)Cb}IMVchoclF~3N6%G?JnEZ5H)Q54pU&cc;N1}P0Z|s(jPivRWS^D`(zDaz3XmP zV&eNmjZcWJi&&D0LYE#R){8>B9K5=Xi;nrSsa4WmPwm6WIOW&8709|#)J3w(9k%4* zBQk+A@M6S(sw4#@th^n=+NJx;fOfNZp0@wUY}t8`b#dxX#=lu;tmhKjw5?hcYZ|EK zD4yPH!jzTu&7sWHvVBX;`#U~jefFrh`j%gXmV^$3CIbVH_V|!OyP2jvoL_un~BS zJ`wyG+A=1TGZO0KIc4mvfY3Kxwo=buzB^yz>C7W(LYeh9&{<1J*hq}-?p?GJ$();F zFpnE<#(;Q;EdNLis{Glaw}V@4iqZy}O=FdMtI|Yg{iOA-7e)ACI};;~f*e_f=}Xrx z@S}Ji%H1E&5H#B6e`D-Rvd33H#&blC{z>xwg!e|p_T&j4Ho|L?1uahr4SD4lRbJ5T zO+Pl3)BPck{mj8!0WfBCIw;X4ydd(+o`*8%QRh&{-4*qFzH|;=?@?==j7m`qMjdI? z>?If8Y_C9@wKYj^kq0%W%}rMas4#94)_3+XCY!qzGBQbp$baj?ewV|H0Z{(z<0<*K z49^!npdE5g?d^$S@GKNzwk>!){+*wCA00<*xm}^7|EDvSILzp!&@Y8eiV#*$GnT`i z!m34hLK-=mh(I#5&9aRjPOGmkzlMP@hirwaTX<((nQ!y@{A_?sEX+LZ#a_an(CF$n z@6&S$IIz)XYnlR>18Z3BqmOfkcOuUDj*lV{mIMST|9i~=mF};N z53yo>Ec)eNjhU4PG`HJYR_2qRvH8`YezPCd_lmu6Xj$+NeAP5YsHs;m%=}}iP9Xag z!YkfyTE{qfxd$m*QdTyxzG^Eb$M=|=g2>j7?7jaADbgM7os==ufRUB0KNngn>qvv> zMD|MIc~}xh(tE5R1l2krLuFj`6=98NyB^{pdXmLfHpazv9gayyCZ zWI-01Cw2Z)mIue;yi_MkMRzO#&6Bug-qTO`_LOMolNM*NG_yb z3ef33e7|l=*8MAaI%pa!>v}1(==r-Y^)`|>Es*QmTDynD;P*Fe{r&iOs_29_pMM5& z8gJ^dRYd8O2aY%_(YIk&swK#~ZpqZpbSE*Li>8Yuycrox-tw#bOD{ZaVXCFk!)#== z#W?&;vDV+S!O`C}MD0XRp4z9loAj@PH{(&d=2-vfRs8RD5&`sEh^a%e7zMxVwLoH}&|`;c#eC@+IUs^jhCfsDR8+7; zw7nCMHDOXIH!FHx=Hr0>M?Ly{tB)g+P1a|I4>g* z$tgO%!1=<7g?@(h;ekzwmL=s1mN!e1C0WZQ}mb})W#&Q`MEk@ z^@VWwI60D2zfQw93jSKSO=pg1GNdn;k4vgnvG_J}bGjpDY8Z6Eyh-sCnx(aBF`v{T z)On5lyga+B#&6fIvg>Fk5~Pi<5M|nevV-x`o$xa0XG%hg-h`D(Zo?|$TDJv{ns2C0 zmJi?Z_U?tQ`xInMl3Wvry_C4(wC5eXe;la47dD2+z{zvE0cm!2LK!CRj~wjYHbd_v$4@QkQJaXO#AI>*NWseVEa93Jq&H+)$`^*hED3Z6|3|s3LyrqMAb~Jr)o&dy*Pz_>SjPA!2_AkDe3rn^{$43*XCnlgb z-ga*<^Ji_y*p1nm25uFba$c13s_s>B&3o(=VUv!#yt48 zO+MkV$Nbq{l-{VfCnW6A;@QH(g}%ei1RGlk#cwUn6m=~2itV>hz2x;xyA3}3AHL7y z&=X-nB}y?63A$5vAXJcB=qt4KUhE}zx42tDYgw=V53>Zcp-IDt!yaxMZqOc z*H@;vEJ|ZDqHZ4Rlz*@?y)a0Bi%CqL$I~&xPk8) z1Vf-pP8h3u4a1)(eC+1iQDU19-PnPe865a1flfQl0DSlG&5+5^o1l|F{- z-ELE*F$QM#ClC}G1%2Y7`$H^MeNGmM_-JY+FXhfwSI_UVDI;%h<5)> zc$v_BU`n!7 z-J26-V#AC(?XY=_gBK!ur!ka8LvW5deZ7)Vya6%GJbNZFaF^WXFfz<<$j{a&F#vtx z*jmefx8qG1b|<6@hj1;urHsa)nt;vhnou!|e|G3A%J(cxHmpsj$=gJYlcSS-Mg~pv zccoUxbd0)F)>!crbCwCHq1RLW=G-DN4e+jpk%m#=vA6_%&EcDsT;}u)`7Tnqz%I&5 z8M37)O+0@krqfn68Rk8-6jf9>XBc0`%}eGK5SAeq8%RmX$lflWE2Q;axu7KHI{<9C z@QPE>?2V?!;99>zUo_+k*<|ZTnPGVVF95KtYNGWneMV8ar1#MUvW24Q zJ@CGHUo1Nv$7Y0$7Yb`x4w^}yXsjPsbQ9iDq+$-%xsM;M6>FCK@}n34LB3vBJp#<= z5nmQja&DaH-YsQpe%Bm6oX<>nmdTuYVxL6uxN2c6FUS&($oA_-0^9NkzdiMx z(~jbm_f$H(I;q)SKD<<&`OxEQ=7f_-Nsa6J4bd1kL;PZNBYt<1GMb@~2B<6~Ypt6S3o< z16bEOuEit!Wrw7o3UrGx5Dp)dt%}!)&$LSDsxNb0kI7JGc3 zpZ~yl^wPSza4bq$eK3e`sW|b<%%RQzu|=|cWcn>COs6n_haE@PvgW)5b*soQla_Dp ztsK3@evz>RahQuBCdO^z3AxM{r82{+zS50Y5^L>Wgb*1+#}?U|1sVfxcT z;^$JAYr6>&x%ZWEC~vsQLiO#?+fhTh+Nh0-uo$0~l=E31&*2#LuQw%!RHF&5Q{EfG z7&_H`Hq+v2RAg(`mE$n?mV>uIvgUzohtUp58(hbjDl^DI@jcUgy;CKtT^fp49I1Sc zC5tYNF3oM7*lQjxBN)lBOv+>aj*g1dPWNk%lWx5$r$@x*yHWwfg>nINg)kYWPCJbE z%o~#{zHEJsy_|bT%A0h93gXObay3p)lfZ`abY@XA_}Cugi@R}~libX$DvpYo9g2D{ zsZjE_Cz-b_a*RJ_Vcl}FDyZTj$RPP<5%$1Z=jmxsc@Sn@qd^hdRil}7J3VsKw^a{c zR|%A8eosxunn9n(ZGIw*p){gc!=>F7zlT>GaGPK5VA8^-AXs8*wtKt>^+uhKB?ikL zBRelK<>%KqdaGF3o(sq38CVUt{NfKtbB8GEy!OhWa{~J{=Es|UYGt)CWzHkP81C#- zCSH1M^TQKADBKh1mS-`}X-30piZ0KUeJ%ZhXukKE;I+6h8?$3-H}<1E_;CHhZJqi? z<;No$Z1WQU(i-T7NhspNw0v*M&~>^djmGY%I#9y7Me2!97((w#>-96KO&qGguj*9O zAm7WwrGDdD#ddSxrZ@-?7mRW|$Z)CoSc0BDiLEDuRm+5TGUGu)Dic(y>DzYlyphtl zKkzvDLul>043BPCr?pA_L6I9q>jYnlaTM-+J@~3U;pB(aluE(HtBJQu8iaX|!n3%S zS=SXSXwU5M_@c~e`@6cpMA$5h`*5JHkmF*ndSAp{ADy^okid;QEyd`LoX%-}O=+kF zneW<@OKR?u6Yhdq3Mym%J0d6b2vK!d1DzH3%l+=Dda{5%8{V}NJyBnlKI+T({ktZ3 z#o7)V)0;mwT_qRKY~c{))JPqBT~A%1^8BD{*W%8ObPK)ag)9OltwhK+Mn=~H)h4T$ zCEFJ~XR6FF+nnvmP#@twy5FC|V3>~Wd(bGMQ4(|4fpkT)k+|=KXH*>H#(5xiKWZ@B z5L~0hCw~tKv%#-DPfhMT}=u~KhC%v_=C17~S0N+{L zHim;(hDnRi3h4ZjU+y_|Lo4i4w5}F3k z6gvPU?2dofK2d5A)wp$_Vvc#%aDNNghFsW)Pp6=@OVvQIpt=2PDfxP^u zO&%qk>>7mE37)&N5#6tPq5!IwCUyF(8ocOGtzR@6dp(81FE@qe?Hq<*`4Q0GFlH9F zpSI_fR=k+Y@=PV{!n>i#7rnQVFgJkf>d`)^+O~^yz55tGO_!myDRJwBH=VYQUfnns9a53#YnmKAdf1RXyRar&QLNfl3KU{}>e!}3`DCzO|OSna^`8u~jUB!F{op4y>JV|v`f(_*vWDaf8 ztIO^)b-Yh?HpRez{=@?vI9^sO!3XFIilyBp#E{hVEJHgp1Kg>jFK&{KC6X#3E|=3e znxa-MrDfC9#VGkSzdO7;U{80&qj}WK+^lPyT=sC}P}#i<$Z{At*O6H+Uv+Y_c( z*)~OeHr&u;#_P~W=#RV{yPac)Ti>3tHH@wvxVJWHp2t;JP~uM9N%_$v8KR*<>7vnn zGU_7W9(f_-5yUrh=FQ1bc^QA(QS{qL(FOb5Koxu^c?9Ps=4@QTXgb;Mcb!K?h>)C; zI}5J78ctH4K>Q0MsWEmhx=1GLGXEK1iG5E=r^1YBlC-$l2FL&2Bc~D%a?z z+2tg>WGR88KDNw!z>bA7tUJvONqrqrSBZ6mCE*q$+Bn*Air<%^9u$RF%I1}*b}{4p znuSl$Hd66oe!aD-u`C?-vhr;NGovmGi!*!VdEu=M?XHV7JF>A1!S;xg*GUDQG74My zktlW85jh6^XTx_9PI)$+(QxW$(QHAin(FbbM{fe}9N5ereVHbu!l0}$YZ5uGST+-S z?BDE+iYb~`McjPX#0OVT8e50*Bl8;vowI$R%hz=gf-5!AiyJlVZPhS7nC7bOf6 zlp*_idgOydOzc(+DaV3a#k=QsBNM8WhIqTHT-P5kFsB1BhDdvHy_fJ2-|?1qtNjU= zXoX3P{N98FO_@s!^d-&XfZd=w zo1Gi!Q&GJ3{3eGzU>PKKI%N!@U3Yq>n3~omhcA10^JMc*_dVd1W`>I~{JQDq&$n}b zVk1J<%!%TDc(8PQVnvz5vR8Aq7d}-FI+NL4O}Z!-)}EZGZ2vU%{7%S{ea(a9u9F`V z&2Q<7T3<=)1Qkm=Al=>6^&|&bLi_txLFKtGdA`?P{x}M*ssK^${~OXE(>`#OKLmxT zvme#^smYZODku@yIIRMvI-hKnKQlv7i~>#)=Pt4PYE+{ZU-$8DlInrtZFj`8isBc; zgmnC~6792MgeYhf#hQ{cb<7V^0};2S2#ewyIvIySdS(Wd4Kw7yEP5OiUW~*$j`-%X z*WNdOzqMn-(n1-%q)Bw5t4^eUY-Cs>;jwU}X(w_~G{tmwwnpf!wRq{&e3wuCEdT6t zuK?e1^Sk$QhM9e@cUNfxd5U&0JE^4*-PvW9t8|WfRis4BEQfYQQ|z}=R&Cyp$EZx; zL-s-`<-Z)GY+oMK6o=<}#Hcx|R`0To$v7UH5D^K zaU?P95x{5y?;oE}^b6=`X-=0gB27;fcrC;aT}sitasM%ikX6OgCo`V@b7&Jb+wN>L zgZ^`GbzovNIBWyu05Cu!K&E-`K6LRGHih%%i&7mbmVDpb7=~}vXDPDydTxP6gbDk% zVy7i#*j{sjv~uhgpfKu;P!wbWWKZ_sqJAgV9v>SUJFfwL+e{0xa#{otLjaAkTL3fz zl9PcS5vel%tM-@NX`ISw5}t+B5fPx7u*x>JU#Mui92+)&ZD+PimyxXj&IKBU%Sz8{ z8xrX;jj0PB7!$hSC|hlAQlvoUJH<6U#|#KVi%ay}rvM`M>==}CU&{h*ilM!j^it&|Rf+eU91@Sl2#1O3N2DXMWXmN#_CVMY!P#SxvWcTYI=vlm-`n!5u6b zW9)1k8e@4mIk`qd-`qDc_uk?c)6-`oEjWD{I#I{^S6828DUkn4E_SgROY6~$`LcN1 z;GPvvaG#mfRK#{^-cuKpVJBU_zhwY-8e>=HTcpC`2}{;+td?PC;V*jZaL!6p#!-aU zrdombp+{}x@rk_8kdWb)DOPIVq<-OiiYZKD8cv3FhD_$Rwe_~QD2%YamsOCLY1MHayNdz-7m4%_9;Vr_SV=4bSGk=m*%o_QTfaI?4 zS3NOkT%(@sc>(8pFw<`1Yv~aqVezFSD6}>r-r;U1Xi1s1anpQ6Nc(Zjqo*CCpZn_M zo_n5IM61`l0q=qRrV&ZrNGBT&cyThzm5(v-irXOCdMDTr10YLp-pHU*x!UdKWH{tR zZ-m9E?k(B*41FAIM;?(I=)P;mF_|(s(X8Y7G3;A_gzX-cfLZTdw=N)U@yg+wc&_q=Nam+`wgwnae2X=j}Lbx5L+%;15ld7_AFG4EO$=Yc7 zYo2~5e({!gc?FAJajslEPX{Yu6ND1D8~=jWg2E=a`@3qYxy0T)Gj&=%Nh;PkA6cj| zv7`imv=~Z*`yBmy=F`qKKl{D!_i_IFTh{nS3G%2c=>bya%6>KdAxjr&Ym zP{To`OEd_vR_d}(y$50qGtc9Wf;vvsabS-3GY>|~j?r{=yhU_P7&58WKXe6bgum4> z=m9zK#Y!7U7KiA;)MJ*E&mBn6tbb#0isyya%18k_Xdh~qwnP417w$K7c+xQ7K<3p& z;XVi!pO)krC%1h+E^LVNj!C00-}nqy_R+x)>?eRSi{$uB&PL=pA&j1W`_{>zNtcB9zJj0bR@sAAv=gc*0_>9+_NrQMUZwU@{7HV zVPORnn;~?Vr0-f ztSnMI@{@TUFWjUdh-joHe#4?wCclyd_ZrGo)&(8QRvfF`d_aBE(}Rgap`FHb2_{oI zQ^R+RUV|hjOAq+h3Y&)ROM9Rsa1kYXCMP=Ck!qEEulqW29^FV^K1jNI;>HBi4FLj0OJ;KO7DjMFgCaKc-m-SE7Dw$|9{n*cK-10J_*&CC3)s$A47 z=n5%cnTXXLfo$=+_!47vpf+!JsC`;m2b5NpU3`RV^xz+y=_thAbUXA(?SKqdy~^8N z8bAk;MJd^|U?E(Z^A*5tZ&A~u_VsKHBlC_X>I#yOg;bz?mSR}ik(9>kE ze2i=5IF4(@?*)8}v}98IAn;bfeA#N5w#gbPMJM;%RLpV8d4X`)L-5+JdrKb-Ip)#2 z0d4%nZpwN3rEc0@=8X4OPvqq|OIJBoy<)?E>P`5hz?vM1U=fg8APpkrQ#kl;!M^O# zHSsAhwQDri%J7}5a*k#}pV130%X#NVbUZJY9$9O2#UN?8CU59(2JM=(N|6p9W=vgL zVR3MF4h*&m+J5vljRNs#`cKXstmiYV@4rp%={peU%TU^Jce z<2CjczV{=PMhtF@cR~y3P48RAa!Fbun?!&f)&PMJfL|1U45GTV~uS* zBK|FdnVtiftEouP0=!cBm=LtTe&>VpP}+LMGlk`7K&Ww!B5?m)Fac=0+xNC#675|f zJ6;t~D5O0{5vnru+3k zO@*dzy$jC);Mzag+j<9>2|*0jsY{|ikQNGx zg7YTd)M@9Ssd{H2e!(Ye`y+h1boqpDDE^G0Xp*52)@69TY)f!h#++>gTrchYtzs-e zfTwM>7^g#wzGYx6dUzJs@sPuB@Wa=+93#*KB^tCPi3F9>n$~x1>lkp`hjnejnFS_k z8PxJV)YXAf2`*2%$MraHr)5Q$afMNaYFrvH08o}J;r%mM9aoJA0ujG%(7OZ-wM+mp z31Vcu?*#mV`%${q5OvwbHsCsc!e~r)d7&|>yYso42ma>evRyz{ooX0r)=1yT#hlLc zm(|%own)R4kT4Z<-mpxohmS`{(PYY!eSiS#R2;J{zibjNy*HN$bGw!7?no0S=wA7w z6yt`+<+;=HH&$IFzq6DJss7M+?X3N93meG0;~(IYFR{w2&#_%N^|s%cq>K6)Atqe| zIzj2g#D*H>tNy9I5u?Kb8q7He zReZfJ)BvA}rvgoIhBfqAM|A9ifgyZbRg@4j?5db#w!_-`q2M~xRtq9`Hy%T3Z|lSD zbW8v&9#IE1`|KWs6Lu1#x+iO{;c=LCF>{I!C3ao5^XQ7#ER;nLKBh5CFBc<_i;Cn5 z{&_bdc-MfE01^uUEx)3)UH0DrmE4ZsBJ1LQ@*-R5*f`rc5ox5%gn_0vjC3^;!ho(O z>|}p6TkYBf2_`imyB^Eo_#^TB+D}P54_D=~)js%$gOnZFNBEZRvIx3vOLuITNE^ zYDxFgGoQrCt!raWQ|I#trmj?LbzFktHR`q#h4wLI)gI4ve`x+CA25ZsrhRZf5o0h= z8g~H`jq>K$70JlXE+R!?K8DTQkvk|ZsxI?wWT6AeR$fsK7tpG( z%NhsH0iCMNk^s+&Ho&65mK6i`1miOg$(w@1Ca!ZZjW#Ped&x%qNtZDCG(6)V#zOu5 zr3J9~&S%8v!F=XIkmg+ip9b?7=$}YyZ1-$x=k3dVOo5$>vVBPfX_Z`)E{K}mbG8#I zScOJxm=)ZFPwTkmBZ?KJ4pwWhFR1ZiM5BOx3`(gLjdrgAhMN!74eKQc;G1^?JPFp3 ze~s&t3aGT7l$xvk@^>)DIGwki0Rx8d1%U4@0kc>|-;6<(pA~kKXtNJR7q8$D%aaa_ zE8M>EdD8<1+RaEm4IoViPR^$it0~=h9{Gs#L!o*G%L3^`k1`cdoAul`_wNxK3Q3VI zYkzl5LE=^45~6}Y=;== zW0%HZ`T1nd(;_g#;kz*R%_@@(2G90JwI9HMydFd$q}TGF3+vMm7SCwwYgw%H6G@ZFW8PM?L~+E7vpu&auoG57)mt>eHaa@fFbG zX6w~df@>Q%W&y=7Cxo~h8?W&Luq~n;*4C*8T55bDx86z*xcHH8ub6G`Vnk=6T~?*e zzKC%eW)&XImSTN-pEQs~uS%?A{f<}zj?xS?qdT6y9$~jYk)de;r8fzYMm^`e=VKyo4U&j zZC;znHDE}BctE%?@8qaWtZIRSKNRYc%A!k@5T;vhp3{B$GD2r`4I_K-eUV(;VJY&c zDdY?byG&iYX3@P=%$zDu^<8_HGG86|w4)-~cALn+y7G)3*j(6tq<|wNO*LOf>v?zV z%zv#Sqdyi&s?%56yGUhN@`4KVBz8n9L(z8XU3KesxPqtLQn1@k92;V^jW0|ucQ4Lm z{iZ6GiwQg~8x}n~1N1F>Z^!8WL>+Gdj-hr?BTNsJszcgedqsR3iQar#YW`b3mIt%C z0Hw=1DgznHn=ro@660H%vzhysy%RQ{J89}K$kyNp=448Hp>r|9h3SN0T2C-9zqwECb2L(wU|#82SvlaD9!E21Ovd!a@QKvqKNLOCM_I_=`6o5z3ee&iH~dYX zMv$%XfeUX8sPF#k==se0;0lHj={Yz2iky709D5=&X8f!_RRXrv5YT=-Z89vhj+c4N zn_jbRa{ntP?5hY`rxbyRzZ>3AU%@91W7-3atTOix&YHSBFSk|IqtrN`jDS~ z*zWE^&)^ni@|Oz0SSNiq*V?WCe_q4`P+q=i{#>**6F?z};B?v;X9V4pMth6^%|*C# z;mhR#X^5BQWHtY-OP~TJcW`%(K=Vma{^nkqHG+U0z$tN8Ytn4 z#D$0Km26+B=_!Nj8K?3-}Jof;tjQ} zyFOENPr7aMT3E`DO=oN3Y7IyDmag^0_PKQ(cnYkW4R>W=o!h!1;1p|rt(SWqKi~7a z&erL)>x3WyWb-KI5TlPfQKyp)7Y~xt?sFj`bW`}!?leeX(jD-`j4g-=IWCf%RQ)g= zhv9}g741ChMIRwIS&Mcp?^BZ6fHoZw^)`{`9OZ+UlfSi%U@~n23yM4FZG#()whr7on5)nCNHqA;AdQ<|f_S|? zbxkgJ+^^SAC!)F2L;2ET$`KnY&zGdF<37lu2n%6&P|VM)Jm*J z{jM&x@gu;`Bdc-}Ds~0t6o{AVtRXYoA%b(g_(LYn)|vSeFx=JW^F_}Zygx#Ta~kb$ zOl>U8^y0gDmUk~X%|I^zRnn!Fr>syXsYe+R=I+o`4=pY3|q&s&-jDRzrawL(Ff za9_pv0n>4QZ$X0OgbBF+S?%#cg5~aX4G^<@1vyC3LY!?}xRi79hxICWxxKG|I%lSD z+o$pQZA6T|WE6>KRb^Zb0rzuXeCANV!$o&Xu}TsRXHkU2f6-}ygfA4_4~}I!?Pej* zlI||cMo~EgDro$O?c>E1)00UsA6JZ3@#ZRf@my$QC5cg#0NrSL^df9zcB^`jnVs%r`~k zJD_3>fZ>6*wq~jj9a}`QvL)$C3+&;fTARR zBV|_&$cEDfBJZi)GdFmB8o0$R?Yi5;CpDXWNiQESj_{(G+e+{ACe*k)0|hQ5(Hpd+ zhYp>pmj*3(5`k__@I=pxW92~E@uG5sph2UrH1zRIQ5%Fbsttum2d`AgWhFa{D{3q) zRPgQ_(;3^@Hw==G(m;?qcq&nk0~@>!Eb`Sae<)2RmXqTEMm1#VpG z?V(Hl#l#4v(8A(iQEk9B_7S?}L(!_Hp&ONb0m7?g1~pD~9R{3Etd~tCXJ0V2U5PA6Kyx5T zHZ|NQS>e92FXtYx`wR(TMGTyA`AleIGw%6llIhZ>0jy!2yB)CB(QF4X0PP;!Z0F*C8PN7A@dV3EQDD8~ zqbbLQAK$uRuv*ERD?tT8!!wHAE$< zU;@ki8qi1n$TQzB+t^2$myKBi9*?IiQC{7^EW6r*UJW8do>{9skK># zEpL@c<%(V!Iwjlfn=e#xz|rki7G_qOf$Ap1=Uqs>1TVHL2f)qT?wciMgFrq=G#H+D zqo*mR2*&L=pRSkDY=?cmZ$9+;t*Jb5vgy|k6HpMiB7b9AD;A2=Hn`acf8SLF_gzF^ z03QT=fpclR4`WTop*rq_Bk40d-P{IAvGCTOrX*AA>g`rJvr%6R;W~Q*eaplq2-&kr(6j9FWx5H_hL=o60*iG3*7Po&N<@;0|}mjhF625 z7Pbtkc}zClC*5U{x~2CCuD*CDbb}9Ns6XLz-8+o7B$l28eZtaL-4=yuuZf>rXPjp| zFu2K{u4(-KmG2yhQDY~=Tv9^75C3EN2g%ST%-HeyhUV4Go11$@cy@D5fp%dk5XQ%h zrZY!dzPRm!Jv7zmZ==uzGH1BvevCsKcTr6uR3Js2-8EMS^8_v^dVMN;nQ)WA14hUn zhmcYvZ;2?to9}%tbijJ8wqej1Qa9mT)_qVi-qfDm?W{-LnRIwkhS*rZ&*>{i^j=Qz z%)mmU0>ItQ&L>pM(yZHX5nr!kkuxvY;@P4gP)b*>aO;m6Q(oip9)6a6f5pRCLf}aRuI2uRv@7 zIhC?*{iNwmKMr!6dhYvbbQm`=!x97|2Azp=y2k|xa;DbvL-4?W@iZ{x{%$E)LtLM1 zSXU>sB}cmw7wxnM!koHTzG)(&U2{E%dVK_!kasK?cziy5P%|n@flLxCz}+03Bxl98 zs)-~T!zkAWq2rNSshtz++#Aj@2{7D6d@xqgJqQ9~0c@d1>UH)b)Yg!SI2zB$n5TN7;U|*n_<#LIPghLHlq4=L3csb zmxR5|5e@R{02{n+_Id+*y$4pjzulmDT#)_!HBT)t&#oRXGcz;m!UiQu4%Y+Ae;2PF zgu;G?qB^|si~sRTsSwY8N2vUbL8R6aWRB87A`g82pD@zAz+VFdibkZcnQ;+2q9I8D z-~RqPs{4OEC$~WZH#I+5UMS2lRN5&pkkO>E`gl|!}>`qS0m*>AA~U)r5GM8Ex9FR@}g z5DEf(k^_Kmk#F>$H6-34|7*|Bfk|Zo{J)?3R0x>E|6dCQM$_{7K$!-@>Hn`KKtaAF z?*C_f06k+EATRm(M*plFR`$=&At(Q9aX_Ty&*EHV%Mknb0Q|LvR7QgX1LMzpJ_uj& zBjfzNfa$3JzSJKXK`z=8oWJ%HgpO6?`D`}8e*Td=v09(MwvvEBAyOuS{^wNd!3zJc zt;F{C4*Fj^h@1=bXZYdwvWMOMYg-(Fh}=UJ6A^cDsr(;eyEfh6W6~Z*Wo=t~)p#4E zD2=X$jp*-K-ev;M~n|3wU7 zD1Gu5+IdZ8-`x)b`K%1}bNh6$G7s1U>tTD|A=Knu8|=r|eo)@gg!q(SgXp$r3vYT0 zVarN(c^d>dSFJ5dJO}=^>YpvizWUdG|C5c_B|~CO5?!s0nDkPEUasBb=2nE|rF>Ea#5FoOvC9bmIo4!`&l1ERhCmuToY z466;)y1Vi{P%N^Y5#>LgkH|2X%jB}3Ied)DruSKwF#r2K3itgo89}@i!~T`^r+g|x zN0R2l0@|!`F>Jak>RBJc3_+8TVi0cmbKa;F0UYJ$;r*%8)&T!B1Mmu7`s`w9YC%AR z0=B>((>bnq5k_Zm*&&V^KhORlWFc>y7MZ~e$L=O_4W!g1xZ$Jx{Gk?F3(C__xm5Kzg*>JO!K6tMIKcKiz@Kn0gd>D9%wdQp4 z@IR^^<=^vm8$OIx*@L*-SXz}HS&4QBEr5H?eXPhRK)dIrR5AJ(2!nUbg$l7L3LYDGBKf7# z0pF*xxMh)HOFD>M?92wTcLGnTME|!kpuP3iczPbk2w2M<4J6+IK-gv&XFuU`>sHmO zoJ1e%8JZbms*Te%KR4Y zTNrL*q<=0+5ulF4@b`qKcq$I(${AfE_gn%%LS zVl83LXZs`i=69-{w+%o0{t7~d@!TLR2J2Y$C&(G5^q=9#n{8WCkZ^*WZ;?opl~m4` z=hP>17$GcrpT+JaDamh;@^L~Vp7kElbCqXT z%R@>QP)a9GAdZL#WUsZ2_-&2Q-T+r*1<5&BIArc|vTX02`F(|$&^my|ejljV&KthR zraNJ-@BpOGDE`rWGp*oZ%mEZ0&O<0^fa&gWXUlJ+;t2tGUIe&gFhc@HmPO{xh?gk%}vTxu7 zR%yS|9iNymu^VZb&`NyY;B&*fs%bWN#%=Gs)@DFV(S*8Qj-DHSnCSX0`&0f#em+~p z`HC9$C*7(qcP*V>Q?5RInNIlwSP7ZJlz8Yju6a?udZ0ls-lcfwO28HpX1~K7(BrLTt><8HG!Mh#SkSvVVbo9d8!D^Va(msN?3ORY=?M$>HWELnxok zWb9y~p!;`f$Z?+5=T@>%Q$CL~=Rh97D4Vv}mx31C1_Dw&+w~yq&6 z{fQP%$?gQcK1B$Upu263e62q%pr!PEFR_!VewlSr3YP#NO?(BHONNd$>JCoUxSr1! zfykcy+2zHd(8^#|sKi;JLDK`BHo&ENHq-=M>x8-f%^k_`lhrneEPiH^Rkw|Crv7}T!DC3nDmX21Cm^> z?|e4tG9H>+!#z1Ux!^ps3Wj;w9D4Wuy|0J8v1s2VOGE2jj}F$g?c=Yb;b=C~2q`Km z+Cx`5LHe1s*XPR_$c`@XyvP~a;i%u&*wK zF7zf@v^Sgx7+L0gB_ck(XA8v6tJaA@F`1_=%`E_n#1KEUwrf;lCW*6^MhHFc;nvwO zREgr^;?%~v0C~6NSVe8uG~G8M`>?P*qVE|u_&!u=Dh>q@tIl05%@ZY=s&wcg3%$JF zCFX7_Y^1#pD5SNZ0(c{kudZlFCR(!43*)cvMF_XSo0K>^3OH@-sc6(xS9fo}h!}!y z{fKP?XM+_iWi+ccSW2n3U{cd?RI4SRo=KlUR>}eOsTSeLTi%vYsrMUb(E+7JV}LHT zd;6net0~&7?)}E^qusG+lO&}!k`wn2xK~E8E=k*-e;hQ2Xm*&pT@WXb%!LQoS^5j! zv`Xj1-r+ge48Wy0hyVvcpnJO>mLF{ zWHx*{Y57N4frKE9WDv*vbXjQqW$OWs!;|%?VJHW$O2P00UNLPyqad@bUgEZ9k!4d* zk~Q+S7Q;Oo$70UBG^K&KBNlDti=AB1;>U##2i!~CLGbq}oC{tDD&&2JZbmo0o0g6T zTd67Spy!4P9~-hfm-BIIeL(*u8ZFrmr^JG|k96?8Ao1cWz~#CV5n$(lx}(%K{4vH{ zC=EOgKbGKb7uLC#osJbNu1*zxj(XmX%0_I{im*FZ$Q%3u>oABWh1Y@jI*rn0FenIM zGKaiyeF-+Pt@%YxkiGC8zBvR~jiv2IdrSR<@Mmo1Yo=>A6K)netPFhsvYy@G7=T`! zFD8Xid~F1k{*&Eo^9PU;Q&}F!$Oa~YDqtAKR7Sf#45Hok4`yYfSvuD5qK-bKge}>* zKwi`b*WZxuyWV^pwUr;v(I{)bPF|<1Q-4LB)d}7GAqP6wf>#}**xV~}0<_X`&hi1T-5Ie@oVZX~ zBS`FnrEQlu5xD^`(*TQ+f;(o)f~_KWs5#~kyuqOQ_am`VfaAI%*Wvxy7Btwz-|I7e zY~d$3F@v@mg}c$fvG@< zm;_wnfg(cUHb#&m3PW!kEV7y1ap$^nwmET<2yP%TKrS1e4aQ1=j2A2Y;}`5e4H((g z#lA<2L`$yY#6^by{aw<2{kn4+$E!DKoVeB8&WjGuZhctra{Tu5#Eb)@!X!khCLfrU z5*1~ZV*`Dzrrp=gf(T6H@Zx#Qz+@-I<^u3uPqsevJ)?A6uBRosma*;8A1!t^|Y4EPy2-Yx?!+TGpWX9GsQMR}qRtGOkW`<_CMUKrh=cQ=nC`EhR##aw?F zAdN3RzQ4Yi6d$lngW6eU(Bw~u2&wIQVBiz>KDtU-&W~dhumT4H9Rv|{d!dl_MXkN> zsCPP^ik`SjuIGA$S~Yhobm$9)$qS%kla#}%oVQa`psF#p&Za|L++WmUt7-I&D#YYC zYq^hN(-#a98O4Z8zVVT!kozJR&G6oLTH;mW-FmRB&TT_5zPFgpQa?&9OqHL%5DTWD z+{({^f1JA~_ca*)UY^q13F-OD&=8eS)E$;p=RY)r<^FYs7?%$Oz_=l-rtDlG%X5L|9wqK%mak47lB$ndBOXojGKY&tK(1HMlh2wMDBLLZxnKce!9hdGeHrDQH7qI_0SQK9wnTB}z!zDv31 zw(v8vGfci&iyd?p`Dfc#XMX=@8Cna4+dH+&h49s+$;KX8K(_eL~*p)+M6&zJJ4uy+CY7a2DnL%cF>vOv&T zf>}34qINhM>v+C&i=9usuJ5IRAK_jO5btiW2jg7M`oPG&GKyNxE1y6lk7DQ<;WwgN z2}kb1k0hR%S=vE5nr`Vw1b|6=fhs=RM$S+Qa^MXtMYkCa4>}yS-=<|C`y~9rKwh)+O;JZWiv~sOfh7)3R?oGzX<#;_ zfgt1nzy}R8{ylCh@bt|ZeQ*96_1nZm5-4scK>a};f@iEac_PQZOPGO!{f4edfLHo@7(&*?X4LxfLSF3 z*1T5m-^-7#J}L)lV$x8D^L4eczX*`gwLkAj4)s7MjlRpXDXDuBk`YoPbviTT2l?*c z4&pF<1BzNO<*JRQ<)QUHkDbqV>~xPYVp2!#X8XT1IVJTOQhGK#=%Mh zwRLS9=?+QB2Nk3d=`KN~ySux)yHiRjrMp96Wg4anAdF?>YbH7%~{cVDD$I zHP@Wieb2c^u(>>Y9K1HXPcMvzVS30h?vAvV@~YQ~$$xXn#oYhd!Ioc1_JO2O$-cxg zfcwW9#JoZ=5hmmJQuup$MfqjH%_`0he+g7H50I%p`oqtnJLQ(#asvtfhG}d}w(W#OQ0&rUGCC1E?>35Yr?#jMQz76OlnDV z?y~x5ke|hMiBjzI{)aGJZie$)OGgqF z8t$4#WBPf?MFfd#ql4RD$4KF(18phKmXdfu)rLogO?)CKh{*$Pica<2T&-maA;rJ; zLF*T8XNDTnaZ$8d&i66*@Pu$+bRom&SNC$I`#*s{_`cFhkh=)k+>=WX6#ia{FT?IJ z?v3}a+~}3ItVhoQltp%fN_KX{;?Fiyfn*Z7!_^oqo}sjv>yX;vr#^29@zwUgEE6U@ zxs6^~4$DCmTa)9NYm3Qx-DO3w#FwmrXhcf-9AVE{2`4QVd_8CI8096owVi0>GW}*- zEEh!&cGg~HV#Oty_vB=3GwI1je5s)RY`tW=PLX|5H8nrk)upa`e^@2BU7WzgbV4cW5Ln zAs6X4;XIz{qcbneXOt}(VXZb+9Y2U)rkALBOQgg;rxX)HnIg?T_;6VHU9D0>G7OXH zT!8yk}(yl0?Zym~Rskty)>87sylfzXpB5M{uQDQGAf8b5kUu1)A zFg3?1`0sE4!Vqv6P-!=N{2dUSK>oyHImhx8hTzwX{wOBTwZ&tSewP@=Q>o%WWE0?BbpCe{{)ep)Fx%6IgW9d6dUfPcv$Ze6x z(hLu(A04b#DgvEi23y`9w+FnYo5iUjiqM}S^>SFrjsP_P%iaH=hW~7Ih(jjei}7!yNm*xpgOfnL$;}biIuZe&;1G)mZ162# zZI|`OTg(@4v7IQmG=b?Etqz>u?i|8+K(2gIwUK*;v|!Tn>y|FyrFLgzYUP4*jAr|T zVYG%<2VNnjOcO)1-rfwZy409YtUXmq85NKB5ILS4xrtG4Y8>3`L9eh{Fn;*Y zZuhpIw!|Kz?MeHm^n4T6opfU9@90G$@zSOa>wB#TCf=1JfCI;AcOsP9MzYcSF?~3l zR~!U?`K*5}Fhl|%z2pElT2UKY@Za#HP`jo1>4qd?&dA+)s~_p5?+b#bS3|GC~o=d$|dsvg%dfx%uFX?QA6;pRw)3cf?AH{z)D0s;9t> z%`UlCw!_Ew%daLlUm}NVBaww-3!l}@;esp$A}be$$4gnV=t{1DSSg*(qYJh~G#P*H zMqk`G=pdWShkDjhwcZn*2TE^*j~0kq2M0qvfGPTcT)iu*RlCKjWP{+{Q`hlnVGxi+ zR2O=HFXOr%%%qnFz*`QZ3d!kkq3=*|)Oh7pFF2h{!Mtl0?*|XJ+l!K9o?h@?NglxC z!gzRjC#jYwuvs3`*{w;JnbVX08Qxn#bv6jT^-F3-d$ae0H1<{CJ#ACXL9J7L1q#io?#_0#@{YXF>>-IockKRw?Cw9j#8tZU87Y`RnuAdfAUx+H)^ z@>M-pbrO(!wtK5Uaa(OXLRuTueY8gQza>!`(Ope^SV3n z<$#F(D^uNE$);kYO2iTQa#(PxL@BwY8l7AYYRW%&22180uXfzl z(;<@Z8R$~HNwh0PLCh*|U8pn+C&Yy~2-eDGiJHZ?*iZ_x0oN;?lo|}|V5Ug8T|Mx% zjR8_Q`C1zevC@Xf_fAKocpad{Kfi~FoRobYTX4zvJXSm+E@1@OR4Wbs04LoQmXu-b>_TT|K;yfi$gQ?-|0oT4=gRQWt7P)X(nPLZ&;EpVB zisNa9q`hEgL+#n@=2o%stB$$cyR^bcU*SESKwOtE#5*YIa_3rVGYqKZs)(SoN#0rU z-KMZwt|*3gyfYjSwroJlsk-Eo|0S0&%UDd)d+~zJf81!g-@13^dcyt~hX%B*bxD8u zvhX?gdr0idahho6451jy=~5Dq{#=0u%6w3IUc|@S>6@t3ET=w1p zk;LKg!Se|rCI@tl0EVfKD^C9MB|3C=7(kWRM8*LOy}t*)j_+POt`O=rg;gJ)?UKnf zgN%Ib_Imm8Ay2f5QFFmQ42PP7(ZS|;vGCTZi-D+V419a{F$}F}y zy170fGIa#2pey{hJ#;V8*12>H>r$MzA#5A|)`aIWVVOKz4EBGQ&z3Ma4_zGdn#le3 z*)gFFbCM&e)Jy#u3zBTJ=n^nrC|mOnK%3sCzf$?$b+TFXAe2KhmFo{Lh-YiBbdU$O z+F15aB2^ZVnVGNV)RH%YIP`9)VM(o#m0Jw&v+_zZDTtb=<)?TiiiN#vgU)}I>OUm= zgLOlVXB(bSP`G)Um|VRhd5xTqc>qsVX(Mj4-><&AV2r9hAsR{;xz;Hu@RXomn`4-5 zEe}pJne`(#!=E?r3xX5|c>45^VMS-uVw#)|2o+1z62Lw8T>F6l16~Umq>A_zpu6v6 z-gmL=X>!l-VSw>7bG0M5NS4KBMSzc?d}zRrrWd-&?R>2oxaiJbX=(km3uNDFzX#lX zgva&t7+w>T))5%If;2?~Q1GM1+rBX_P*^P1Cp;Pa`x9vKajE43MAV`1ssa32A8G+g z&?nFy$ex5xmsWCJt65`)tG;q^wEWGRQN1>zY`|q0^tRY7)nTYS3R$qJ?LpVQxyqRI z9NZm2Osk}cIa+K;`UdBLwvJ*J`kZ30M*vZD7Q+9J^o0ZA?d3oUd)ePRqAU9asuhP& z3>=2Gn}`;^_6^dH?kwdcySZ0M%dl=dW2E)R?Jo~NX~uaVb%aoT+o>-FdNM-_5ijeH zDab-PKGf)6rh(Zif-|uLWt?bH%;_jS9g}Uo_9G@8OSHcmJ*~dN&IoU5t87?WgG@j0 z7zTXU+kVCMvI+vn(lLQ#N}hdugqOh6AqPYRRRIx#yV`adY?T%--rxiV%{d`hkwz;+ zWOis#CR{yTu)L||vOikPzQAl2jah8*<`)UU^wsN*=$p#qf{0Q|npKVkV*)uqr}o|S zQHAT-Z=+aOp!V#u92o`K%jSN6nrMgKT#S~*u7@G3J*Np8o3iV`FYQ@dwkhWYd$*XWc>)EL2Qe2l5Y)(F~Nz;~Hf1 zNsuR8uYxpzo=;R2+n}O=pjaRlpAD+x)PtmaluoOWqo<8fQM(iaWdJl8SIk3^fHT$L zarh$B9a#X=K0NT&s8rgdRb3U0>dBq>kJx@1~g4^10T#rB>NGiHgzKS0PibSVizyy=S zwhVd-=vd(#0^t}vCJINv z=rE3dkPyEZ4DgGsxsu(2vbSA1{Z*n-*xw2vt&OFCLcoftFf(=E6!kc!|c1utCMx3wHwDh8I`MDrL5(by+R&ri6!Wo9+tl{1HYA9 z;~K2?;P-TF`mnazy|8*21E1LM!_eCf7r3j9*ZM)Mv*TOIe;*SfEEl+iseD=gl3G8a zm%a7qAlaEgv3cPmnRA@8XRbpFMiC81c~DAx`JVj%ZIK+vGLwot8YpuW2)bbCZ`A@@ zRSV=OF8q-nOH}HkIqIN%HYxAY;eVGxT@eaifHiLM0-3+E-X%2JnuCw3(RLl4?5xI* zuKjrM&)Zik$j`0`x~7kC4bI)$nSa37QRjnXFYg+3xo}G08?5e4*M)vX=N$ndwMh0K zU;(k~X<22Eb4a);KG857K_wkVftaq!TX6R+-TY1hessle?}dSVf@27XghRT9{qY%~ z?5;dg(tIDts1EfKvbM7Bc)mMH%x=(&c~$`sBJ3>S<~C{;rRO*2=IH+6gaK{PID8I@ znT%iF86r1GyFA=+lAY1}6*CPYx+7NBPac8DGPI*`ce7WPY6L3wJ9T^m@(DG60BZgl zvdiHCSc?Mi<}W1$(>~&X1FjV0MJmwNf6fVUX2fE;j5K=@GQY95)`M5nO{8{@kB<|h zpWx~OyV*KVd4RSvkxqp9WFc-338!7BhlK$6T$Pr;7lDlP_uIu_b6`S7qlOXvh;IO7 z-IXoA$W}`2868y9TI=Ou-Fk>R=$56ZK>`f$znPW2E)XBU8-Dpg9f>KZM4~{Emdax4p!lNdv4}8t5M3?4YZ{c2E(B9KHVZxqJXo08|_QyIO7N|Lo7k zZ7p8}Ay&3yJ}FZrHTHhczF4)-4RiyMGca6r$EmWY+<^+-ANvZS$MI-sF&aa~Vgrt> z6cZzo=nb&-e-`Qb82j?10;J$9#JF!AKnyqpor^R4B~l?dm@$suY9U;&v!WGJ)t6V0 zm0Cn)FM!D_8?2#}0mYfD1T@p=%kc)vyg#)yM(POvNKAcoArp4N63eo;E)` zJywX{KSE(0o(@S0qc-2)ReLx|8$P2}F~E;cyA4#ydVTCvm;}$fGchNCnFgP=Ut1s` z5}%Q@E0Sos{iQQl26PiITZ96UD`I9dMNpRNkabOtwHsY@d;4HDao{J}lrAMv+WnaG zeHbWq8q3!rB!5YS4tpOKa?ie1*1;nNf&{ocq{B!ngjVLek0R4F|Nd-Ma5N0L z+pa_c4Q_ap$q2dV@=yB@yc5vnK_!Yscka}d z65sfE)$<-e5PArD@yr+dx;0a!XDmnjKHyoo0E1#7PFEC3%1`|sboAh^cl{GogfN&U zpi9oG`AwaLPXc|vuxI))MYw%Ct3PXNGTsU3wf-0Oz=Zal%zK;p*V$3u(4bD2hxWfB zK-#XLuQ{|3VW+4j7$aL=O>zhaObU6T*C;C5dPw{sk0sAs+r@Ak(FY!3po`^WNvjjd z+FO)N*nXS7PTVPC!1jYCc0XAYsrR_H^GONFkeIkFSJIklk7#8r9O+W~f{u8K=|t|f z$AD9;8IE6!%=y$%(&1JpX#eBiK>N1FKB1`wC(DZ@{J(54Q~;H0`9Q)qcm5TG1RypnU?q_9 zy2Za(*j_3rLabyELac-fSI%}im@AB)J@om+MP(zoO=M{~TPgC-#RMXPj?NK&U0K!B z`OAe2@a15b&pQluBSA%zPgzACm`=x7WQtS*-Ln@t@|4s78jy>}=|>(Nn{Li5?5*YO~2O zY8y=*eTqjmn}2iv?fD;!|6S$}p!OSF35p&;0x|&`{nCo$8|%=7;;Pe)66tEY`4?f> z?0l{yT-kg^4Ou|x{=a(3zd&UzaMDiH+3EvO>2kh{YuJ|>I_E?pYW#r1UpXws(?IH< zD^o}&KL+mNud6iqAwoNDV5;>!pK={a4w2ggp?zJrTE^;%4#E{274H`?x>a3d3Pq3M!E1KMHqe- z@obgcF!5}lPzLVWJ(Wg26tE8@fev_y4^hOj+O$b+;rNVo%lqRE?1}{%p>|vS83}@H z{D#kOzC_xY8p>t}aRQD-14!Rz1`Up*>J+ZEdQk0s0I2Ib7-=FK+a8!rW`kH|dZZ#% zsFJ}mnJ=5x9Z4dd#HcMtmNX9CO%E)DSs>k(p|EU}t&ycT+9yDpI4rT-1pZ}<7tO{kn-B-5OivosOb+0)Oaue1`t@HbLj{cT zx?jj#9xmof?8wD;2%fWlxUh(4)Fv%dsDM5nEj0wW#cmF=1uRHmY}~lPR3}yr6(;Z= z-7o^CcW8vn-{=iJt-Sucf<_jG2j2jl3DPZ<`&PGo)WjAaQVzsmO0oD}Q&?4pZm=ew zkg6=uGN_av))ew|-OUya2^{xXlezv7Qs5w$*s;_caek(mw0I8?rWAOQ=Om?pp(#)& z7z{ZZblN;_PI_=~cts`p6N@k>Z*tIT_L%&RmaV@AutnuYHTB43DK+IMyW2aN;cpbuV!FN{GXt;Q|Jj<%~17_~jUY1ck~jK3^eA zOI)q!JjaAgYUx59J|ivfD3jEKvJ8dMbn28>A>ReCV+v$6LY7|YVncQ?od&>wQA(j} z)BQnKhwXM0HkMYJx93`cZ9R9D_$m`AB_R=x$sgxT)*`xQtIS!X}Ais<)Z)!Bn9?e|F?+k^@{?Y4FHO z0lXfxA23Xj8jT&WuJ$@%-h=Ww)ZTs!769avR%lr5Ve-Ks1MrCa-x>Dzk}PqJi1VCu2cw>Z z)8lyYs^(8nrnv+MMQMLT!hVwnTK~99s6aowp;*H12eGd=Z-7vmjL@fI`SL)xbpp;j zesa_ng<5p5Zlc6|8k4^Nfz5;C{#cj*KsUN&7)rynFj@my=Z3(GoaK;ZYO!j9iL!6^ z?2qNR0AxvrKBvsS6X_pgvy4Dr6=M@cw`4hw-4=(`c8bS*sWcoC0zq4`NIV2i!!n*$ zIY?P(P^Hm-`4e7BP7|1I+>=Q+R}N$RkR!%QCr;8d9Y;5?TZncLsM^l@Lp-@Ti4`Mds2ToZv#MIb=eRU0iU+haQK!7p}x-)SuQ8* z^7|Yoz4V`u#a^TGQA!DZ`ka@^VY@Crs8T^~7>1)zWQ3N??Q9jICN~Of>bU|dR10;s z{!gHtEOvDDYXDx2Fb;B{wz$L~#KdyOJz?SV!^aXeu5USXbUrKNw4S-d)n_xFcm$KT zIPelpq;#`35pO{Jc0`Fi*V!4Pz@Z$(nDFkTJN&Dm~Biy7lq;4P6(+G2O z>3kK|)@pNgtQ$tej($JZNK4XuCD%O(LLJ}|bd{=`{m2Cs{gEMX%@pD`#?8cL2Qw`bsTIdlzi`Wbr4Yqx`& zBb5|bs8q0zbI^GS#h`k&Igs~v@`g{RB7lO=px`eIo6heo3JQ>oAx5PVjZa0&CC}lp zl}pqw|Cc(ut06N@=@nhsa(KIEP;WUu?oE}twLtIxHG_hX&0tZ z2qtBk0~-_-0#={o$J|E58&Tz(*mXxkDO}|7UCJMC0KzL2Igru$YIt{bQeeMa(L4Yv z{5deYcSWW6 z1fi1@Dy3mQHKEDG=QEiMw5jh;i8qnyZoghAC2lwkLdGj z=z2Zfjh{Tyj_mvZe zOi#HGz33oE^>wRcMFj2*IiCs|0c#Qa2+<_q!s9I|s=eeHK zRBjP7)dnYeH8%lNs3h+KihaG`Hf_3a<25Z z+~z3q=g00T^vb$RJ|c4n^O%+gx%cc;UhUZ|#>2!ly(E6(Ner^U%V09x@97{`{5|cn z4O4|tk8AB)jVhy>n-CK8kj~jEv}B)AAC{>5{qs3ZE zOFVva8~SFNUY$jmtfIqqrPT|F$M7XfUQ)&Sfe+7VnYjKCf10f_M2UTfsX#y~9-TiY zATJUPc#wM(2x4Q*ws>iUcI>aQm&7K;iqPp&?NI|ik!}0x+zECW3glQIH_HVVT-XNq z(Ta^gjHA8%E5I4S_4|C=gbn>1n>V6=-Oq>v55p8g%%#iJ6-^fX)a7xWfGW#jclGus z#V4s0jD@$?B2{+$?bfegXWR7gWk8tS*%a@or}u$q)GK_tsd07Y^88S&pY|0q#0}%C zXn}lACJ5RE(Q_FD%tkDRed*t#v|Po>%@M41KW2`UjO+{1N~P90fCHt5kdxRZ2b0!` zcsA{}PIih}#zsjO#eUc#b+ShB9mu3bB4our=3y+RW24_IP()%J?b7S6V>O{WrDI~O z7V1j*%z~lgj4ik*50L|&VKq-VyYgoSgvyTs`to!ybZXTx&nZ3$A`CygxzBtnphqDw z$u(3Uwkh`XcrXRoHuL~>SIGW|^*8;4jE+88O#u7b{5~`I^7T#o?tC6QE~9eXFt1H| z6xb|ALNS^RXV8FpBsDfg7nB)&LCyzY}1|3-xxk7-c!^T+r81e(H0s!jpe*_iMrxd~vzNu3l>YSJ|y z`h&-i`B(J}se2|^t1?UIaM*n2&?%I`JpL7SR`%5g8Q}2~iK&4h=i2|{MUV+QOMUhu zh(HkU9L;b8Z7%khpZR|F((Vc6RZUe5KuDx58T^a0`?3!5PO-6E&$lb15-x-_T@aK;juMArM*|ct)Tt_`U2^d!ATfuXLCb5c*jTcrK|-!zSYCl?D4_e;|VF*SDkc_nsv6Th*+6+8@0Laol-p! z4Ke4tfR93(DS75GC-g^Q{KjOu+T;SK%pT&9f|F zVrbenCu}qgKLu`uZf;nuHsyY@JLE+QU~{6Xm0MZ9S-)+T+|s7mVfL78o*M#tS{bw4 z237nkXY%CQQUBW!;C+l!AH(s80*9tdJ*ukdT52TSVd4L|E3kj@>f6BC^mi~R7FZed z>G-{3Ge>8Gs@38p2~y^4;5N$t_omv%Pu`tUr;({Ho>jIe%NRESzGaIFw+qq}1FX2W zDg$0vV*Do9eM%k@K@qbBp3%$sj!A9tuT{m6@fmm|HOmv=WX=8dcN)GGjFj)-Fv%I< z$;~^Fh&_2xtN9H#9Tsn*x9myAhXcuDav*r-v`_b%$U3ruNLV$4|_@WHHyU;bkdK#ZC2<$0}e>pkf zpwWQjVT26ZhK6Mj2x?I9H)UW|#r66g>lc$|gm5z{2>_+)|NQD3o2o^L<~-h<_l5S- z9m>uV2xwuPAKJ0or0rL#+U^g=0W|+|DbwaD_yzpQurN zK5EH#$aB+XURLliR18PnogWxBGnu`;a<9$NzXq%kUz;9LAMnhb;FDsN%(#VRibQ{c z^JeJ5?#v--;#@2nsvqT~=f6r(JQ`mDYioP|=?e`ArF^LLeir}YqfYDHk=5X+XH|jN zAcKGdCf~;3BKs$SjVFj@aQ|G7_O4P&yE=`qc3i6!Hz7v7`URA#&de!lekB2)B(%vubelDe)}TyUK10zcSJ-X1n!CJO-`R zfeJOg@_rHG1#$r&df8g8N3HVTkhDfaRlYi<_!oD-!XQL%!Dk2$!q$Id7Z|L-xY-x? z;s2Ht<=&58Adxr$)a9YQ#`#DKWUl{?wDDcL2V2`eLdAR6k2N16Z6-J^_Tgl4Qk-!a zH51@QGO+Qg2B0mvaoH?X@D##LZM(!2nW_wCcNKE=NNSxF=?%z7g;=_dEPf)(g}$4l ziwcBNudU>EJmMGR9{E6{S;5vItYs~dAByHAU9ZoTe7#)baU*7AFxq&S6>^<(3gc;WdOc>Ag0JWB>?^id> zd1!6z@t<0+dNYyLtxQK+tH)h>n~dDC+TVXHa25^H62vMyU`qu#O%c0seC3nuOtJSsI>afgVwpe?3T?SDD$2yj@N1nM73Mh;5@COBFj!| zxq*rtJ$>J#Sw6L4t^}*PhbSe8=O|#Qez$)4!}1~rV>>_ivMRz*|Y(V>13hetmPBm4`BIc8B$&YK{2P4uwvV95~@<(U+p`3@_$dCd|2@1 za?=UfYc)7O&YAW|`%=DsnI7j2$3ZQ;z-Q?fRsAI!A%2QqPp9~3uB6+O(U!OTWrwG( z@#!af_ObGlaFYu=@KW{-X)gwH+0<%Wav|ZEU)~+b=tSoh{BF^KFIbyig}Px9!l}~n zFWVjCwFp0)u1IoyZ;IoX{zYqTGLWHe!4$d``gX(DA)QZpaN55mH|MaeAqBjY4R)51 z$MKd~GX2NbuE9GIv9q(lkT7#!C^*?ay_-pY>gZ-OFViJsw6|uB#T-9F&I#DOgqc+m`!Y-j;)K z{)u^F@Sa(Li46a_FZxrbi!2TcLdicJ^4NloQ-%|${484EYo7C5Pe}7#L^nwD)DF}! zUzwWa3GX(Tf#)|D41VJQ9t=AAL>iorl>SNz0gVJn;d_$`;2#;Ki5^!db}$Shr00^B zaO?(_`Q+LIafFeF9&FL`nH9*edXCa7;I4mex!^q1=qf&Avzo8CMefCB&h*?*;1R3p zdS-Uu(xWB;nNCBei&Z9NMHser)93kA6DA@kAH`$)Vp9`k`RlF!$d0Vbd9%jOZAh&1@39u;4zcKcsKKWHxtQO0nfBZIfSg6BiLv z*x-txb2T$t)`yi;-xjo2J{X@{S}qWfjcb_!q`=m*(;fB~{eNtUyA$rHpc&*Fk2KHJ z*huD+jZ^xo{TS)DHx7sCIgkO%`-C6yEgz}T&1W*X`NNy?(#Iy?)Tyl+18^wJSmw&N zkgpeRK7L?m%54l3lTkR}X)$#~bHsl6VUkPo>T1}tdlYDp*D#p7&Os8hs>h2v_mCNp z*-?hxN}qxnB$o@bdB56M7Ah?VBQsyUzwjlh^^L@Ky4!A?nit9(D4*3(ipK_ZH0SZ6 z@8Afs>15uUjaH_7b0g*xf(@6%1dg_4zd#9AGUHcsY-}tlzD#xY3(@~M;J@*EO9A`3 z?EhA#Fb&(jpfd0Pw$cJg)XRNY5ZTm`D$8X$Du)0WJ%cKDTl~#pumDZSZcd`kWH|Ua zykkJIVC$_fs{c=9#l15}3&24|rcA?Y<^A4&s_whtDDl74HV~O>?l3;Bs7gF|pb<=@ zzpX~{?4Ni+k{>)sSRqSbpH7%ESGM|Z_C)1dKQL7T-<&*?uEc}Bn%P8Q za43C)dbw`6Rkc9}lUEDGO(c>q?Q@QVIiA6jLDlhk`nP7?kRPt$NSi!|#o(VwDb@@i zReyF#awhZc9k?gYEzA_Evso>WiwxSZy^5NRgrnTtbJ7Hgi-aX{R5qTSDzj>W1+SNqg)ICP-wS`7eY_1$Oj=4@6Mi-(I=X8= z1vMnqicowhx=1em)~V~ijt!98BUF)ug<8*_@sC z3u-%@!h3Ub^!1y?N}?aG!+eLr>JicQ{CCGN)~ARW@{-ukK{51K9R&1ZS>Q_o$t@@@ zd#THxP+1G*g8Kq?H!ANJ@hx`Dg7MN1zOx&k(hk}|s?nqO~# zQi860ug3N>a)rb)0o*DF-hklb*_Xul{1UvW3VFNn9*cD{WXX8um?Syd4feG6bOm6PbjB?^N0mtB`f?j?y{ z%=dXCLn9OQXFAOcAxoGy5YoMA=z+J=--783Ad+Vn_jib}_1;|iu7ztvhq5h38Yw4g zlhs_Z@F6`l+YNa|rZx;b2percKwaW~U{5Sb#wax%Z{DMuilOii$p@1TJ#=8Kh8+o$3}Y&jN)eUUeRzo|Id$KhQlE93G>C` z(al%TX*|x%Y07g<8y$}GmrX1+xMqtFt)^FLmnG)im)O(8=OaIw=lfZK)5|NgvWr=* zV)Hc`=?@=GEEcOtf00YjP*`P>+g#?tceVY_F$2NzfIofi@+z|XzFErrk~F2HeKbnI zgM`$;*uK8*%5G2}m+_DunWxt(Ot?lbDrgzhp znzKq)ntPA;WrF{B$sL6&#G)8RKZNne-(18kq<`0}lEWC($63DkWF7p+C4w71GQDRy zGIHPQ!C1_gXNgBjQ2Y0M(oGZPLK1b!j()&SQ|tBUgY9EDpXN`-CP{^k14%3cm^bM~ zS}n3~Drl$-3;*CSdcD-b{XT7*&2%l#_EVMV!oGWXitpnwXPuPV6ao*~U9?+Hbk?1H zXxBY$=`UwK9m~e3#?dIXMs1Hp?OmeZ*w{uT>(09SN8aVMFZ8ubwK1ag6ZjTl|0iz& z8J(-Vjr*$z!sC)wo#!}+saXG0i@$DhFmJr7wAlU%s0Pj?-XF*Dt&ImtI)SFL;hHkz z*>*ydD@YGtgjpRdt&y3qY(nawM{T7eK4_D6Jc$w>0{&vTN}ym^GWv9d*9h)Y?+$q7 zNz+}oMk!UEFuqp#=CgiNAEoqinF)HySa@%8y+PHAr098o641segF&BVPG}W-`FWXJ z$rqOPVA}qsTc1!C4e{vZ?yUghdLY4|I9`#BAR9(h3jf>@|GZVQB;6D&l{cjsRF1)` zfl`_RSE7$mT@D9s{fEQV?0oeRq4t~iiIZweH%-a?H_S<2y)+w(8WPD$r&Xq9;OiK8 zj6?EV*kDA5fA1N3=Polbp&fvx;wL1lR&VJYXpRJHAc&xxrGi3*kY-;)E9W&i*V2>2`w0@e%IpQc~cS({s zUwqudK>qmv=tY8y=nX8XB|P6dIF`2VDk1J*FVSx?MQd0~-12ofHsAS;@t{2lNcXmQ zF7H|>iNntzOUYL{*1*=Ab|Cy^niCkZu=bO ze(6YUdAVlUMeCf|Y*)-G>yL577Wyx5Ad^v&)qXho)k{54u(i_1b0K)EclX7A{i*x$ z%yiz9mq zWTnP)t66h{HH{T7mtJTLh*~a86^;vd;j1~4b{-WLl(9E06J)KBbi0q4A1fqR*_G|+ zhbLehwtd1fIOrIC*{gT%04$mlf05lMOJF{jS8Z0_0p`z0KS3rIw+Sk<@p@cV^SSri zP{80C1;+4*fvPP70pqYJx^EQ|Mb$U!xXF($06PS^{2>C1naz^dp+|2_SIT*4W6ySd zsi;v9t{u{EvE=Q4ip$U^gcgOsv=a7$mQDKoTg(n0v`n&MYo@P-%hk*xAq256KZN$1 zu6IhdCQ4%lNnU&DVNqD#&^5buW%R~oJrUD_kG65qLWt#cTS!&x$3NAb2s@#n;M2ai z=(_Tc>Ao-rJ_OgVj&Rz0YML=!ax-0Xos@xWh(Xp|bo!AmXc_WClOR}s8EG^iYk*3i;7HqJn$ zb=55f@6v~&=S!>3spT9;KMUv8PX)tTBdfVuc9962&;j@njI9FMADqiuKak_Wu|=<( zQ`#4Hm)p0t__4~xi&samRxsR3gHiV&_XkWq0?_sKu;8#Oa)_ek6ZfwN9o3}~yfZKf zkBfB+!nu4oc+8v7^_Z!uIm`KaxY54NH@uE{15>L&~$}dv-3{Hy)4d# zt7*A6y$IrM!J-Q7rN=~P8AP{mq%Yy)@vEmj5WbVO3}uAcdsPnafT)|Ge6*X=@v1k4 z#!@tD{8r^|x&91d^+8ZW{tW$oHb;8shJNTW(BlVL)5!fEqa^p&9=2`#6MVx8yCRH) zl8s?YvTk+6kyLudg9}#+JdXnr(1IQ1i&UA9(897OWcBHDRRdWQ>GucS{h)^mnTk_9 zeP?2SjBosRn2AOVi(N{;Z4OTS*$&O8N*T*Ow$Bi(zv3;UKCAsgZ+S92q}lXM(~+EX zy@X0lQF)Oe0v$%l(o7E1GKp^@<%%8I?%WbMr($2X_a-Fm~HEmIxefGu# ztp?fX>E7Mg?0W;6SCCxY}9!U$7LPvl zGc=dB^o#gJ`(6XFJ9wCX`gFd{5u%+TXF(Fa>4y2p)Yl68+>efKOGh)D=VZ`pnv$SA zB0xk0eK~S;<&40eCti};QjQonV(4f(`jPMk%a=yq;A7sd^hkPA3hJ8R88GSI{B$s^ z66*Nb-H`Z?$;|WE{kgzpJ;!i8(_h{L}^J; zN_2%!0Q~aci2=U@K&6}Rr(}`j=WHF?F?8M!#~iwQ8R~Sf8H~PNG;_}pi%8tQYUJRI z-v~U|6r3}ubosu$dUrTu&hkSU%p)*}#a-p-o{(xIDc9?+_`1b%mnwiG}72(v{>Lj zRjPSUuwVanA9xxGpt>Sx?wp!IQ$uJ^dQxpG8Cp51!ttO#?OJ%al-eQ7R!EUuLJc}g znW+5eZul%r@ZV@bUu2+)A2Bg2RY_Y#JVSZVm@0&X8v_V`LZrX<4tH;~-!iYp>zuCU zn1p%FA3mMrbT)HRG2+{?V8(KYs)r=eg$7&n^(|_O6h7H!*Uc9Z6f1@B>n;Qs+dK6b zUs|Utm5DG~3zf_K=h2sc^IaczZQY#^i{j4=VkOp>BRgBqM2+k5qnOja@C*Nsw3Wu| z&Q@dcOX%Ntctxith27dDo6YxA%v-DkD&@Fb`h{+63l-V#w(o^0P_2^hUbWpI`IBDZ z(aKa4i-h5X);sTYj|~0-q8topuyy&fjpdCvQ}0*=)I^UmNH{vNKPw_V=P4d~r47s; zOJj6SjbK^>n;^IA{Dv@Iq9ywS1!=MFTetxcheiO~;ZXd-r^)mO*i>?K^klk}1=4;x zylD_8;4`Iuv|4mUd=w?}ahQ+3`(#_wM$?I74tBNF?=rsDOSxhN+yl22`NLP**a7@1 zkq~-V)sSe~+Vt$ZK=HG8bj0C%zF%y2m`wd!jZ&Q!55;*10s?6$IeKBB$s5Fvy;|4M z;Y% z>_=6sWov<9>F0EAJ^oI60l&0G?4X;@OE|v0N!Ok*n_jV*Qu-9~e4UZ4zOA%5FkJcV zj~`Ur0VSs#l1b@bJeF2DTOEZnJNu|TbN4mU396GybwHPlt;?=L1X?)o5g>Hqw~#jwb|{ zsI&)EvRPFep4)dQ>AmeYFw=14gqJ?mb$lU?h&`u{8qqfre(j@lz3p~zn_FZ(!zu$K zZ^4RPT1J`U9{;6G;CqXzMOiwDOScLdMj5s42Nd!!gPl=F@J8Iv9`-aC&#SMt{hZ)- zV%RlIAjFoV=iU;I7(MIym;p%%^TRfpvPZ1>owqN6;yVS3KSrRg8a*i-zqY<>3Z(dY z*^Q-2#^>oEJ$&@HIlT2y&bs@;1%skvWAF18q!&5o=G*GWZG@{f_Rv_g>W4xeR@IiF zaXHx$bDY0Q)7^DWQ(TW1?9JvasmiVdA2^(PXW&w|vVW$i;BdR%zA9U3<$xzc`USs6 zYSW}5n$G&+SuI-t?Hdhi@nyp$dT#WMbw_rHjiS#9=1m{Thr^Sfe{9jZ1ynZu`G|IY zU0CwEq;%D&wN#&8zN#n*xy2G>+eQVDq1@296ajPLB=slh0@KfdFhXAVGGr96m6>SM zF?O+B6o{6=?&W){V-e|9Y5(n4DjwM{TRIJdl~fgxU^IO5`!O}{<;Saa!C?;()%PMT z;q`$`WEvW}jY`=D6r}v$(WSoBq@Q*EcG5_y+3UFtE9Vo=8&Am7?yl%Xp*7JV_TB5| zP^15`0_JA?B^Op8NZRc4>oW_oy;kYv^SFu$!=n~r((7hoWJQyDv)%4v0uDsP04aDr z2T=(vQQI}jY`ZzA6ELMhkt|O@hxwnP!4*(M;x1JOE{`|9>r~jOxf{DpPdIR$Q>UG*@1;+g_o?;`(KI(ZRcbVs@2d(eN4HX?cEL%~|=>!sdO*)IQJ1 z)SDZ4qc5;0Rfdpq*Fqvp}R#wQo6etx?za_$$jtldG>z4_(6X%%ype{ z9BZv(t+V&%G38^CMlD722-_VBax|Un@gl8oTrlQ_?D;Jae=zGxDWSc1rX+c`bYf_c z-lCoFd9g``qoYl~A_V}amR)yDQ0HXT4F)``Ep}fx0X_Oc1$Z*KN{z3Mz7QY(Aj~>;!@NGuP<0lj>0s}22Fw!Y8bBpKUN2EioBwXx z6R!>_A9F^c26PG zo6wpDM}#j;^UJ9Ad`=i{$oe@~0OuC`fCl9GsKU{UW81o;aR?;KL!TX|ZF;Zz@7Gp72ziG5BHD%jCldEr}u2z>U*MLXE4VnXJ^DClJ|!*Ig2FESNq_WP_$ z{ZZlKe$l*FdS{N7;XC!W9nzKDU6ck)3)I@Yxj1Dd{r(!ZZ=sUVu}{waJwoXVyqv3u zkY1Na=oX$#CW)T%K^foaU;*AX&qQbKLNQr@g9`W->)RyW7gwM~DL)C;jR@CGk`b}> z^8bT!7yyO(F#j#a8GqnUvAJM+rZ=AmZl9&ec!28$y^XE6H%9&LbK>k%Kp%RBP4}#% zHQ8`}M1;IU2scjj=k%qhq~obqC>9nR&7N=}Mn;nKgn(GlcIihm#{BF5wZchZe0ufo zc0x*v+K~5b)C;}*IdV#L)5CqK>`*SfeW*v!bZRIUs;tJmj?Pg6I6H(LmskM2Y(-E_ z+aN0$5C6p-dwhyE{-H^;868v?A4jyOYWd9iH#w|ewVFzjQ?axK&y9F6>B`QZ3s|m7 z9AUfz)8~6rOXda1J#Eyx99>(hEsG(yD-tA{fb^$ zr>c>RFZr=bJbWjeNQ_XRXUVpnjgQ-K3F#y|`t^G>=g(XG4`^lxjIM%i>VJ~jD@L_X z(4o_NVw1Ag#1EbEmvo%!(%m**pHu_+7!pWArteJUpnliwE{}jUv+|(nZ;M;$`8wca zdi5IzFSP3Cpo!WKrHJWA_CUxFbddMQj(AKvvj3Ui!tXr8E~7`+Ae34VvTI~5RJh{C z`*jbuNgU@9n_^Jtw;!{tY42;eAN~rVh}$yGsB~M`+2Zqg972VrgUK`@^_s40(F3>J zsPc0Km}0KY{S(wG66nhG36XzPV14vNBPu9}^YPP(Op<0}StIHe1}MxrcDKyvI;|)6 zG0Xx#@Rn=4LCoTka*m7Cx8Qk^WB})3Ao5W*8!+QiOl$t!(W`B<21vNS;A)>t=XCo} z^*A_FA6#bro<6OsYnAj(YthUvzM-*zly$tp*$+dInq| zZs)&QfVozQM+@(JPMh8o@u%U#k8{YasE*ppdj1?$?>`_J$juh~6yjm`NP>|nM=FgGf(QZ?Fg*<0r46q@g9e!N2uNaKtos)!D;e>oM*JS?DyTx_G{;82M z3WW5#o^-WYQ~hf{BC`{D@sCRmFiYurz{Z1K4?aLezqA>qEE50rgYwurrJkO6!9Iw3Bo1ILLos5imW^V06>u zB2;`SMTbfT0n&u!Q&fd~N8An;C`Z?KSR47h0;vUE%rV$T8&B}BgMHIT7psA{5K_<9 zDN<8{VR3D+6!{ond&Ybd1~?CNFuV4b;h|P_`(FfO&@`c~Zyro$FjQccog}+0nS%dz zV{lDure!CZ72YFx(Bb%=C-^G-;uoc!P`A|WZ=mn2;r9NVB023sBj4%Ca;3Y)cfmFNvccqmf;heRVCf>CfzLsn?fsAqCC&TS zUhLHp@Q#nAhuly5`#s?2x0mE)pHXms$6F0%3EQfTfM7r%UO_=1qM2R|fqAZxCn0U% z)r%as%?2MxVI_+~TKIHDO;iYsQSGIp(3>F2vln!VK+NX%;tFsGQ$*x;Pka*G4o@V} zYzVUGaZtu3A#oVtQ?+`R+=EoYp0({5yl>#gEQi2}VqY<6KH)cc&jZx{jv_B7>IXWf zePe?w$#|lrOb)mW?~7`3)tb1;!T>C8*=h0ieeJzLhtaCxhRK(kUn+H;tuF5++(?V1 z@mn(0^M1Hhw=p4C)>XPy1hlzH=zZJZVB~7qnc_e|mSEzqlP#(bt=+`q+Ov&pIA#Ua z>C5IZ&xJnmew+|<>_?xIVg4DwsZYAWja$Nb-Ib%IZ|~LgMEO%Ip?T@ZLt=E`b0DdG zUB}~5yu^_onQp$N6-}^<_to6i?0zO`y-b#^D}+?StO(o^_$zecUYxJb)CEbmk1szr z6H+n!3>iOMSk0eeMsekZe9s1_Rx~}%t<%qB4?}bIQHwKbJ^H%K%Rm`%X@d^eH%ax; zvjpd?T}xLJHTLJ(EDdktrytLIMfS}{HNCx{Txla#yxn7>4}J^;EwttvbO*Kz4oBWw z6XS#f6zi@d%(H2C-2hMqtk!v~3ixq$IF@9LBQ1|#ZNv{-!JcaF-kF5>ECyp-)5Eb` z(-F;M>&3I45yhx>>$e9wRDaGwB16?x3v!D z7Equ2Oqwu3YcgKve`CQFRB48cnAS+NcpLSZqjYRudVlJ(B&sqhA{%!%1BL+vPP*?`wCiO?XBSb8I=fPnj7R#Kc_ zIz>JCPLD2yhWqhs0QQ1FKmZ5_di%F`ulqeQs`A$r-Cf;B5n!CxYSDTlo9M%=-i@mD zP>o|DvvQU7bX5^0L#ZPD1051l!J7;|!QGl8ZkV`lAOeDJf@NE7)6$OmT6a7UN7IuA zxk`yQ5jBqG|Hw%e@^O@Y%iEpT&vMOr-7rufAuN%;UJjLi#r*|$joO*S6U13PUF?GU z&~{&pm@GR9v1m#yr^;&?0sw4k6o{AK--<6LGTY`|3Z0>~B-h2*_AGWR~yqc$1(c{ljd3OiI& zH#r&_K2X6_Vvt)pdB$m5VKLx9$;m{h-+cR>wo~D0GgIDhwtD)qfj>tCPWmufFRk}W z!J1QtPJuz<0P`E^cGqdF;qe{`gAood28!4-&;=6bKt)g+C{sm-AQJ>H(Rkkt3r?W;PHR%1!QmD5eC0usp zXHOhcp6(co~)M1w7qR*mHZ%R~HI+_x&%QkKgm`uvN)#QR+D{n-W_Y>4= zm*8BiM`NRSj@p5vgQtMdfjARan3pD+yc9!i*JwuJt6*b?*|`|nWD7C$flkHIp`B!X z$3#OCI`$H$iTUpgLYZh0T+`DcaMQ#`^Wsc1&KDlFQ7uWm!E8A($a{yUXc@(<t3%>kBT8Vnr0a*pMwu5&@h6eC`>wDs=XQ9-|IRH=_-2C%>>SiJz(YoqX21 zR@ucY^LYPjo~wqgcm3YI5BKC{Aa6w*zF*(F|FFR#@*Y90LiDeU7uEih8zy6{5sTNK zL-|X7V|RM|g`{b%l*-Z#W$3&jap9{C&^vGcWfIY%S~v4TS97f?BzrF-@&p;pO5!=Q z|4wtbYF2>jz7MlvlD^Yge=#E|X=>T@bis~+KhKr%s>k_2a`PwbT93cK+GuFMt3)JX ziW^d8)*24_60((QN0%!ngom~Feu=A7?wN7{T>Do{czu(|+n#-qM6NCqhMoX;h)y-V zw=8K5?FRNE?o2!NS&}f*`8323lu@4mPWTHQH+h5dY4KK@6pI#Nc#v#wEE-{(*D2L% zQ-r?%CBh+6!=I8v%kq7tfEt#RT-pxynj;NlqQkS@#8_F3ILD+X_}Xu)QlsXl_Zqj4 z^pjWlfh%lkuius{U4GG{89!LgL>)(`m0BR^)F^dwjUT|IgOKv2l|nt}FRIVp1dx~& zf6hxFqE!cf+kCKqATl!`@ZKjHZ78pjq;-2gpUG)+D)w;l+*9Y~WB|Tb-?T7$O;XTp z6WE!*5z-L)1TnMB)u`!cC%MiUaCE~zfQm(H!Gpvl$gh#)K{VZprsoq}WqfG=Q?mc* zLdB~f4wN?)Y@{MGRBeBi8I)^8wLF~c6n?`F)JWyb67YIWVSS#pHM@5?hQ^>UTQtVOB$fkBjP`pDo(*vEv1P6QKtps8uhl74p}j;UCSGkjn{ zY?)6S>NkM!3d>WC!sA3CHUsuA9AG-8IyA6J$Y^6HC?pq zG;L_4)YXTmnbOYWOO;P8>t4v)EXFN`;sy>d8lpl(j*MgRwkNT0^qr~Qr9Y1tUG_v? zTeEvDAMFP|LX$)NFh2WtZGv{)=w`~{1VX$Mx$GE$RnByLk`uT~AyRYohkhZoDM1W z`}actw8-*1-YytFeH_3!C83oT;5EmyOfgQx4K!Uc#^WjV1IZ; z_-lxebj|G_s*&SB>`N$CA-kacoXyP?*akHhgUj&RfJsKn4M`_+_ z@4{4?Of)P}T$(PV39YGot*hYv{t(v>7rMpe zKI7d3{DZ*SCuBZ?cGHT8sHS_SLTZc*@m0)349nD7Oih*pLwkIs>{7>68Z&d5KAy2{f^o9u;*#> zU1Xsd$kcMv=)wY)3NJncCB}P9w6Oo;i}%Y4OdI&YUqmY6NxrygBCj;)`;C5}d z799FHbC$kl>!whwkWZo%YU6ebN3Ku@T3cEA;3Lsc$oXLJ!*(g~$AeIRgdY^|pmu%= z(SF(t-}|+GFgrm86PVDj2o=Z7?i1ua!|4prq3c@xJkXZTlweHfn9_ghaZf5)y34(v zWozeJItp{Xz%6ZfGEPZvnjk@F>S>C11*(PZv@1{L$0uFC;MJ*n;yz5y>1~Mh)&qYG zxgWUe#On?RxnIk=TvHQPYYg{cgqjCg96lF@(`4`ytfuiQz2`Ry#T@z}gers8=ZGHr z(TCsZLld<Az`;$O0Vnt>-!o|QRA{8j3o#@V=Ks91JB^)>Of4MkG$ zz26tQg!$mZ3d{S#ALwj{NIa{9^BhsOyVoA|8J?|Qd6@c2N`&%?Jy}Y;$HU@-NcfN# z&Mo<*lM0*MAyB|3G@#Hg6n@*Py;wZpe%tj)h^POZSRfD9I_dbMju&CFBDB;Eo=}s{ zr-Z-$k{NqnJx4LdK6%3Dw_Qs+?|~fg?yMxJ(MlOG)jN8{a<-ldtD~j!JpD7;d}%}A z4b7UC`1=xaI-e*?dej(fOwQ-vLbE|&07qcKIF{7)2X2;EsX8So`D)<=#m}$k4?h|X z?Hri0!&PIG&zdFps2!Vif_O-a%a$oSK!TGSk9Et11W;F#Mec+n^t(3`}PyJ>t0j zU`T%2E*+{DjFM?K`DLhh9&u{@ctz3?*fRt3)o>7G)d|NI_9r!?8E(Onj=xdc<#;d^ zoYCv)Apo1%b{MwT#xL!NoF|8!W_Uw5oo}!05DxW-=}X^uCe0~d!slKxCS*Kb&{R0;xHv3S+X?tg=)(yPv)@1+N-v>G?hE-@x z>@ldpe`TpU(jW6PXdU`8~7FTg&X2#v#Q2o4(BFfr^jA$asQp zXw%8Dmtelu>((w|G)SH;?0J@zcscgK;(>dovHTUg%0Z+JrD3YJ}F#WmfR;-MS zOjm>F(RO|mF()MMW!6RYN6xMxxSllN*665034}R}P1k;wFHn-c|1GFd@XFBpL#NH; zk8HWmPP&G>{;yc_Jz4XvJQQmY@Opl_lYK{LG!T4o<)j(^)YAb?G2H*{bM2o5=3Pip z^Bbj9S96@QbNdxCS=~p4Ti0u!28utIYMEbYAx;GJSK2x_c09h33in4VP{X(z?QiXr7a5N5i%p=x(sBl(v$mvZ;T9g}uwmt%6H>xrn5M!2s z9TE?0AEz`JKoc5HTK^IfhpzA`fYV8_`CwP3k18Z6EGmoeVm$?IBI<~;mX|?IdfVru zD6;Y@$T0AP#P3{M*dktfHeyaO+BDG|?()P~3w>++A zh|DaDC6?;Q+^v08R)l1399U;t*IxMjfe~m9%Epu^?}zL`ORrfCj0op%H^aiqW-E)e z8<i6J*1b>-$XvT9wD8VG{;rW}gj6oSt+IS(O5+)w%?n^>bY%ggB%YNSCz^oAE@^e(JMT%FR)~ zinX1O3AMu{>m~w7mSn$JTp}H2<(uzG)jaR*DY}^vb<%-t~eZPbxLwzx>Xy0Gc@!6EHLYE^0*iTUfKC$aNVoXhCAyx)TjTj@{P4m5UJH-R0!_hspy9WQ z5aOS^m0-hwz_y8HL@jX-dh!5)azPn zyFLFslu%T6_G%D`miIu{%|kW~KK#~cg*zjiL%ZVG6;hR_7j?Z~o|9%OwT1F~Ku-8W z3Os@yIq}uC5|?9oneR-i(22|ayXYRh#iGgMr(4)$PdNm(1io%nQWl77)PEM!ErO3R z&~f@O4sc$5E7sN^Lv!9G!E7*q(yS4s;gHp9rPqD3dGi&4+3#m1FL2_IiOeJuDu8X4 zK}joDTvCATH`dvJcPQiQ>G0_{xAV2@^vax(Lj5AOH!nTc3WalwN_o{X?rZFq0{UYvM@A?LnNNj?rKbf#Qg#s#p z3l69?T_3Q2T1!7NcAf6d>H*IXOEQd<4_IZu1V2X0(6b#Jrll-h{o1}#YIOkvy^~Je zNVpL(4qh12si+!OsvB32{>+a9XYO^9Kb0L$W;N2LO(1IfcJ~Jgk?-|RLH!KP@2S<+ z3KihZt5>bb#9Wn0uaMOaSB=VPy(@smZ&NGoJ^K_wXdD{PU3@aocwQ%()Rc>70+$^HD@J% zS9dw$_=#VJzx5~0L!%Carbzk_Jm#Pe;k*)8tIXBHEt;;mVi*m`Rn~TtLyx>s+Y;K$ z`1*v`qW>|D0TRQ{t^Qgsi(VlVakpl`GqQbuS5Y|H2ipSWY7R(QadZ6pSrJ0o%IMqo z>e(kFB1oFuDJpS*tP48g3_5~>4`I1Exxm-D_ruf5VA}q@UFcrREs>PS^tL$5VRs7w zzZ7hmM1?Vuw22@+D8UqcGD5f()W!^6yCRFHp<07uJL1KiPU0S}dl-u&|ZRnsndTM-lSHzFB6XGCEo*pQ+304dyRC z*~|p_apDF$)z97(fmhq^diM~YzJv%q2iVkPYam8v5p`K~bo4_K5_51dO^})$k+E|Z zxZgfbzWS7(%Rnw8_4%7|df`doT5a84YiG7|F{I1~Ij}W6N$S7W;M%#`9v>oCNg1A< zZ8JTz%3c>Rjn8ZM$FO>1YYM&m!IwDBA1G#dWdyHNU)mWE5KDHgT!`}Djz80qvT5=P zlB!W$*BTwGr(LV64H#2R=NzFRAhxuR5mB|vUTGxBzHk;d@7tZPWNPWI7-$?(9%g*R zrK$;Kq2e>oQNBla_XfPzdMUsg5C#QAwsN}RU7D^|5H)rgzwR~|L#71Pj%%RYWLaBJ zU0>sw){s1k&DjmZ1rBMv=w5?DhqB%eWTo*fTB4Xkfs#^771-^6>Gp1m=tJ9U_s`;( z{1iriY$4V~Pka<)#}zT2b~8oR%MOpxKporjnm5EL1Ty>%M2ANL$6biDo^E_t5%FUR-# z@@_TTL8^?RV=fDj{d==~%8GnsN53r7_6~#5KLL(ooW67h^_#%G#@p6Tv&pex%Adc; zWJZ00zVytxv&Jy8biW?midL)gF~<`uU1(iDsxZ$^K$cgvNnu&n^Z zCEI9Ft!^hi^la-F)RnSowE2}^U6(?XtglwA763OQ>Gc4}duBk&9{^te{=K{rxNzkRB-PA-o zgjEo2a>RN_z797rTccBADAyTkJ$s=MS?PpUoZ$MUi--vg7aPOQG)h>n^S? zw+)e&S5ZNKLv

=%V8PcP5>U-ZHI1$m;N@6Vf1qKTf*i-5uIEKgN$% zvo@DpBK1zliYrpWP=T`AOFlRvF~?-{lcRHvd@}qnkK^`UA$70WpjKXm-tH$8Fo;ziRNzI7&JLi5Snhm*31)atgVU-$TY4jxZr zDUUePydt=GV2!RnU2u+lV1;}-R0eyxd9?QGl4i}(xV~7wGxDk?AyE&IFPkIFT#5YN z3R3DM&;M`X0E&l}htD9V@lapYP&*U#D8*!ntS^b$nDK4?SjCbQGEs1*t= z&f7+Ms?m|gA?^q~WH7bv^N6Gqz3yn|@5RNL)SmB3KTt>vSJ<6pR5?FEn+&HC%8x$( z16;;fH1gl_0@uC&y9;Eaf)?#Kf9DSNp}RgN(1$iJI1J1aQXcqOLUevd*8LJa*zaMc1E#WrwaYz9M5u3hld-Q&V_hsq~|W?#o3t99VbqSXIt zIsRvn8Z`OThEWOibK^e;*!BxqTkJBAZ|n+HDTl3=3L2a=l6FO%`(7j#UzbieKcxfg zc0?njHeswkgCkqBnns)&UXGmxjf)|t>xK)4O4@PiJWyrUiktDSZ0~p8 z6=(Ldx&S{1QShuZA}&!hqHYuar`M4KC*u~1A~@uFJ2=yi63OH6r$IRy)& zv*4dtmXL<)8~#+hPz@uxx+*CYr`{#+LlSm|Z~0?U)Ar2FpmXFCx3dqgHp@mZ_3ED zj6o}ZA@!))65~dgLNG}nMk5PaXdI~YAG~M?zRKOychq@~Sr&v%;?(Y3{-mqJF_2!= z^g;+eA#6HDAc! zrO?Aj`esz$6Gy&(2F>hZT+-%-qi`ELFypgd>Wo%0!3`6E#FF7m2BbXVa4?pnPKE6E zA24%TZ`t*--8R^^(;ihkfYYaV%2ku>mR+UhV0VN8IWI0tUj06B6vS2gxW=GQhVpXXdA30cCk`=aDYZJ%d4 zLkYCpy=a0YRbRMT3maPPLBfG3Ztn+9sxSiXUcIjm1s3f7XL*2?Fsyw4y zU3F+0g$ebjdHDJ?eWG+q<7IA78{p}IKLdA4KfM?-`;%idI3z@3RvI;tz{Vj!zIxWeIkw3$dNI)@C-*Z^B#;5>@?*=}D%8{QCPegb!Xq;FNB{j67t zTHLX8?c2@fkMwOl-ogZ7(_jF+U@ZzN~NF z-Z`dc)yI8o-cW1D&Z^p_H3(C46o`cD2DUpp>NIL~E}rtQV?CoJ?+x3(J1>3LaTo}4 z1=)c)RXA93DSgR;^4%lWjiiLxgIAN|o!kn4yy;@_2db-|W|jB{*q#HUTrxJ@il3m_ z2kX4WUL#7@aN_K7P5TIAdr$YaM#u>SyOqEqL;AcIT2fOue;YE!FmAis5M=ge?JIjQ zeBKXYx%Y@$_a)4T=-Gn}LKW`@3F)4x<`HGh-E$4URkz=7dxKN!j6!?|8Y(+MxzJb9 zAosZpJkC^|##P#%g|3bB9#EkEKxtYp66Mb6)^rJNX_Gwjc}!0F8sY|0r|$3tP(E7E z)RfvzmbC+%5d`^xdgd!4l5d<$EpyPSl{c(B)eASqL^iYwAT-wVF~;qsrk%_{bisQX zDgeLnw^qDwt!-6#;oO*Q75(dQ9O^fzNLQfW=%qm^5Eg^|Ex&$QRuZWYN)_z055#n_ zPUbybz_s`80p}{;zGehO9gCilqXolGFuJi-7sfD)O&lkKPKl_9@72L8`E-fYhDEQq zb`SeK$#C?xm9ikJ=J<+DzCV^8xw5g9Z5b^FvnERuUXfV^YqKeVmg{eWM;z`ZoauJ;F#-TZ0qdD*?NGZJ0tk!zE`ecPYH?MKI+)l z(frLzoZVxprcS9W$eD-f2D5x3W6(_v4P55s*w-mlyX9E=P>+hOVb!604F%-;u4a0` z?(`)I`}8j5qo*hYI!J)9}>W7NcA+h+}`-F~bH0|>cMY(|fJkR#>i%8vFCOHN^}PnjFmIwDu)1hTT;+kuH~qucgLg+OE1jt$chdIq%f*@53qx zpH4XY-ub=q_&5`N?6$azIsK;$raktKB5EdreMImDXuvws6g*t<}_>-V;G1y4}NCn80UQ)kGm~IU4<(XPL^YFKl&QuH%i4*}+o#>Edq=U|yWAEpJEq z3?lgUcO1O}P7^>QX`a8&0B9ur{9Cwwtr_sIanqRCZzoe{ag534eSf|_PU>p>UOo%0 zd!LD!RuCaRZTN+Riyg#plsdGQSODj;B#ECd&Vgmf;psGJePRy> zS4isiOJHIZA1%r0c&hl{UBoae#ysDa6&ALedt3H%vXq-d%V9OU=FlTR7=%ZtX*Y8x~h8an?je z#+77LjBUNAO8bi;baLSOceO}H?vQ+PU8JfB`MSt(-*IWGaeWnpX)WM0ng3BhxquC(flt1ISZcry@ z=JVrKB&}NhTQ(rD1Fe2g_#H-#xKtu%$Xr|IFkYRY78Uqe43l!A7~W# zGE3H>t39aZ-s#}Vs#?X!wOWN0SapbX3!n4F5V+Jiex91R+T*^t?*!@Wxg~!4rtumz z9R;C;{PR?j%7Y3|Zt{l0Ay#1ExNT;?f630qtZDKyOz_w2-l!$S4HE623M0sFYI5b7LWE z?L!cuEI;2={JP9cUSQ9$2mfhLjWxooe=Ez&r>XM|PE&1h0o#%L#{C-YPpiB2dvTx) zw|bXdgSiy`UG|IIT4C+6zzBha-ebOaI+qjH&TDE(E(!8}E{l10KUntLpI5O!`#O90 zn>ezk%5HkQhpB)6WE`q+vZdU8YhSNO3s-w_*(|D>%QeYbJ_+qG70txSi6obVZVBx$ zZ}M71rwEtzfUugK9wwy^)xscG)V%ikblQMNLH3UR7YqV?>O8!=nz+% zVBl?ptJT8`-sUlD^`{IiUYwTMPVypS+l7747cmSHE}~zt%h$PV>YpjHg9f)gL%DSN z18Fth_%jpvFbW-B%eY)c${x@=F1P>c@DRTaecBZD6CIBk)VXleHw@N87Dj#)%6RJn z=J_KrApI$CU&`}hA&`Kd&PE@j^z}FPNcmvgO%~JKLWV})%5e4EF~>|_Gx5n9$E?Qs zgCrDG^sePdN&A19iGN{aWC5vqRJbH4u|a-%`CW;10qh5I=8;0s18?h)UiTCu6fv5YNI#S z{f4|pRG}{DfWOu%PUE-w+j7?za(0Ob3q_5~97!>fVUlguW`Z3G%GMh0u7}c?3?B)> zAIvv|&mdJG2Y0AYhYvL19|09ms4Z-HLIil9^`QpXQ|tMh<^~>@?{A=0^Qq*Mx;*3Z zuNxOzWa<2{09E=ARCo7-{`qN((trsA!D(F}$$e)jgE#_=q_r!FW60MX6CF0+UMufZ zP1_Iqfvj9cJUy2<4bg$_T$}IPYi;o>V54X2OyA8~xnEZA@G)pVLDMO$q24Xv{_00y zh?M)r{_P)q1>-*-boYm9$=^i~)v)sqh9LO%V={gH&V+`+AQ{IBP$o&u`8a$DyL5>T zORmy1^gLo>Q>n?uE>92u73FHkI5KsY(4mcK4TXHQq}X6ugwLf7Jl_Ajr;kx8gZ1tXY~Zf_myq6_EPn;aHXBePzJF*I zP3E>5!572g%`KnK@byapp=q@G*d)5iJ7oN&!sOw%RUj{X8%1;2%Kst!tHejXM}ub^&=uVYw_;vj-%JrQ09Ar1ghs^u7&>!hQjtAer`y24!`O%;&FNJMJ@? zrwh=HI3Ue3l zgF#RGn$%8&VbXc+>oEmyPd&U05BapauF+5K)}8fr472=d=BxCW6J&pV6W{SB4nQT9 z5}rW-DjD_SG>FX52>-v&dnO9NCEc&+U}j_(fLDT)6up9X^^#z?2ERK^@f7FNG@f)? ziPpu_?fHc_-&2*P<>K9&4_i5yV(*fEAJkRnam4nyD^_tRyNBL+jQWK09y*WvP2IA{ zS3&1E>N=XOFR@PwZZH7jd7Adfy=h`kM&yh-d({Cq^$CU_i5=_8&+9d^L5hZ-$@`;T z#XjYkWdL?$_I19~M@2IR^@99Q_<#5BXARKl0AxcQ6nds$8J7^7MSofN%R)lNP{DnB zY1xN@l-{e(FP2C)TLWmBL; z9OX$8qUqS%GT`!9I8GZVQ<=3>L|k7pGBTQhKnORIe@`}W;K&sM;NA^b(;v2C(A|+K zGk`*Vpk225N~att^&*+w%8hDieDX0Oui!p-(uR09*}TVqv@<9(*WGdK9IWX zwA47x_Uo8Ih3__*joJ7aBifhP7+eazo`4fpezf~lZEt1)mGur^cjc$-uFmko`tFCU z^6W*Trw6?)csZ(3EUou}-Aot-+kX#-mEbE-%sGLG&VPqT@JJS!yQH;gE(hQIiRMTb zn?J#8Mphs&qN9mPea*dJ}cYy00s4%M{R2T#!>y zz&25ot+I=;|I>H)?0TZScRv8pqHDlX7af)L~pxvP%>od0=P z-%&42LHLvHKkE>sqBzpvTjb+nmCTBjMuP4*Cb?JiWfy-X**%u##aU%v!pc;tTRF8p z6-3a7Hk5R6g;JD+3SMVinBtIf?Obj{MTSY|OIN$h0aH(A(7>UwRl#d{qAKp@7r=Qn zVmG4HjF{!&GvAB+{+$(ByY~M ze%x#fW(`$k}<& zP)El{0+ihQqQ$cew%{Xjr=t;Z*;rUz2JLM;=-_9|@nR~bB`kUq7V|o55!2PheUEI*0p2g3z9wj8`PF^Sv$KG9nhpd}t2aB$OIS~oEGoi`ntNeEc5@Ar>X9>J zr?4}ShmGKw>B6{)baRnP9|7qyPw6(?d%e-^eE8^UG(j!8xS9pB!45m={5-1efZs?0 zeXU%Agf2zukgk>xZBAZ^kT-;G@Ej0FEyX%inwaMXmlZ#`z} z?bDBPvyH@`58Z1{B-n07jlk)&v=3Ys+N%dDPtUDXyT1|mR?umT{v6?c8Fw(AF};j? zsWOZK7hXT=7d#aYMD!o~BZjCgzaOFWHeuf`-L$}#J|H`r_THx_0++mA@NJiUU8SH# zk$g4tS@o8Jwi63^o#eJ@e%qE{+RI@x4bnVTYWk}9TIx2$_irl{8f5beWgPqR-7vmrFrIU7*)4cRYo!0a#ls&ler&8P)ly@6DTSmKap_WzEFF+=Cui#Q1}lU| z0{O*pXmLH=Pm|6VVjb(4C@Z-|NS*^u7 z{?1RJ^YD<^vo-Cs04;28Ji(0nryOd+ACv3&JegQNNfP^e0Fn_qsfo-wA~LWI_C`M_ zxDW?##aYO_`w<5^7brKI7cySww7Q@rl` zdYyD>7x&*QwRsVH1p;#q7lsEqVBI(Sd@ z_1)vRr_7E+H`=#tp^p?Dos*o!p6Yxp0X09Y_KCq!eseVRr8}}Hf_7et!mE@*%+iT^ zx2%Tw>F5xpmRmhcvkm3?hjqdRO%1`xmUPg1XLHk+|IkR}_!WNx%goiVL_levo&_8k$w!y9RU@<$- z!-HL?W*By=ZsQH~5oFb>D9+$eSBkilBv2%Wgv3b2-_$z(JYHc-vf^LQ$QgB1c=0O! z<^c}bM+CT>FJ47|V&p&{49f0_O#AV6X?Op~yno}iX}9Xh_A;NcQ{`c{*7b*#o+0iN{=BWh+XIA zV20PqREQ<$&(QXZ(w8x@2SDnwLy5j4d;9t(`4fAj3}?y^8amBmKY7E4*tp=0ElWeV zt;fad3EqE`Ohk!i6ahYpZ|W4`-(S_8E&A=TKJ#j`zvCT z7{z{r#8kAR1++78#JZOl$eWGIDk-<8k_j#hwRwzBv|91t<>C_(A)Mmj3|g@yE(G4(7@46w z-TvNtHSLYdDn$8+mDeb(7knvcWvyn>{Bj1324GOOp0Of?+X%m$6&NxRe~gHYJmi6m ztipTgP~rC&iD(mAon(Y!>F1q?@DLNjgMJncBKbPI!Unj7F2Q_zJAl?~T_#^2*G;TE z*qU=+{C$1mTg$euSMzeVO~SGh*aCN1;2R{d1Qz=jep)#{)0ChyJv8adiT+3*nnc;3 zH)sSN)HbN7Z)n+rtL3GZ;(fb%xP|MchjqK9^yXvE``Rj+>vJ%^-_o-ill$=hR?fw# zXhZ$GzCmKoB1#Y=Rckg0E4G;Z{;sFCZ32nTugUx0qvi5xI;Rt8ek>QE0XE3=6uh9V z21mrkd%%xNt!IFO* z+Ry+DJ3{KrXcbEz^I60l&+&3l#3wfnHIVx0Nc_M^#0NCU5G3PKXDrlv4K$V8^UzV0 zwb}=HbJs-g;n9|^AQxhvYsvCa|LeB#j<7ba*9Ld__Rt90XAflV>K@jewGH5Hpikm- zV@=}QTS#l_msQflGDzP-;PrpH36)BmJq8M*Md?Z*tkGt{b6jVInwU;-99rIaVI#%$ zv;fkIh6*T$M#v0THfT;>!{0<&Pw`K#1Q2xx3E#f}8gA9dVz%Qy%BWSfi$=W9f+tM1 zlzz!$bVIMX!O7b=oM_#6Eoi0wbz7$qEPePIzt_7E;_HX|T;4w4)psUCIo%HOud@zJ z;T;cxt1H%x)x@SES7M;QpWjS5yk-r!KBp2qU$-^ugZg!N9p>af+51@m{xg5vaVjA>aPxP3~#3ciQ#YVo> z*(Nzvz&mpv%7ARZ$#=j<=|vso%K-{t!TT*|0|&hN2k-G+-smfr{M9MKyBQ|d9a2b{as^1YKI z-W|UyuHfqy>4%|5Up_th_3^6t?(kxU!Jcn_sp0@DE}==?AM8N(yDq@@F5PAYl8J|; z^5D8Kkcz**zO*nFbx@&yl-HH!sKyFDLHg}FxdHHHO>zfs25~OUZCKyzH>S&cszPRU zJo+#qP>*aC==d`e%?WS9q6_Z>Np@y}t)s<#YR*iNJD+gN+; ze$_h#-ILSMLr&!@{Q5ZtrJ{%kcQRM~2?PIM>)y*U{HU%w%3&B;J3Sl zwtW$3<$FFbm{m=?-DM4U0nPab87|`f^JQ33Fn2it4J{5e2;{YVYQFmWPNjEu|EOg_ zIN9iIS`3Twq=@OU1=G=X6ApBBef+=24sZB}mhqYtpL&3kRXIH};Y9jJe>49>-rPaJ zhd2rL#a64Pb|cvFu#g>SrT>WkWz4@q#5M_?69gR>S-|U(tkJAe)A25o5`F4&rM-9B z6>&|zBCM;(nBb`R92mMxat;w)eT+OUT9u_e*=ijeukUbZSFy;)^1jz}4Q&DP%JZW# zM#x~EVnXTqUTA2KQu|7U{(&@cUKo$E1z3-fk7RRPw%`e`luD%uNA=I#&6??{E(Thje& zc3qyOpWL+GoTX8w`G4#JQ}x0?jPa98`^>%XB8;kdtxu>J%~ z@s>%(>3+1}7sK`sGA}=RiM@R9_dcUB>nq}}doJo*W=qT;Bt#5ACw~h@?m)&(Lu8-G zSAUSqt9bvcYQr<}z6pRGAsnSoDYR?gn(8g;dN4D(`SS*6xBl8@j2%V)P*yWttJ3LzA8|kU zGRIJ*e`pqU#g& z|F4O*w8)0Md|4QZRJ<_PtIPjhNT4*Z{rroZ%9h{zuR>bJ<*cFyW9R;JAM@!wXxM6Z-!R0B1bB#1mg@2{jciUZv#*M=JQ`4b}k$0 zfJfubb3O);hutmczv3-bFo;wHEk#}FUdW&X2f;!OX_?}bl5=eV5njhqx! z%hh~i)|-_{FJ#h5Um5dflAq570+!?|Kfq5+U_`wUblEbL4b2j@clH(7 zX(tJZ>0m{wJ9wd=%wEqKU>$`3l;jD=+>An()r>er_Qd$ z7NAIn`};63;hlYv;;lB1V^BCEB9Jk+FoSkv)tBnoYEK?du#R+UdsEvw`|wRfk477f zO`E&ol+~|ITIcCusUIq5n+Q-Xa2KNf;ge!?(!GUN-Y+ga|+jlDr@aM_#4vlHl4?! z4J73}_A8A6=`aWwSsieGH}A1;&J#vC`?X%(>ZoqO;1G;qL{M$b&#*RdOHT3U#;C_m z4ubFNJbz78zMHx=!I^vOh)olB_iKH%+2weGVIHU0iMvsjWLg$ZOmxhrxz0Si_bOAv zZ!fJ3unW?mc$R2{yEjoY9XB>`hqSE=U~PLkRSQe+Z|UtimnSS2Py$Y19hPrUH18ov z9y^h+(Z4#oe`?)CSzIX~y&oF#$7v^<`@O3dP6S5&(K@->gcsj8ufZ{5EeoTg;2#uv z;cHHE)bRw~2I#q1fL8429|c>*6?i6%6?zW9EXM}9x!ZkLn3xR$JjVuCtd7?Cq?5*M z#Ju$Z(K&6~5Ei5};Xy7S@d$;lSI@UGeuE=yW~9Z2?Tt*0=Vq`)9Uho%lb|EQKTwdGK|8k^ay?;vh#71+7#t+SdU!aj4iBmjJS{PsDak8;0(JG* zGRL4(uVG8Xw_iagM%c8=@^t)XRJ<1JJqC$((=8JvoZ%vepZn1s00Y=_=mZeM?H;DO zwgllyHQ}L75BmBXPDC@bd%e4}UxSK) zq-xr9J*ucAq%UQy$W?Ek-JORl@Ckw!q6CDqe8J;<&C$z0cPzZKl3KMQn%gZg>ECzD7!sVBJFVkMQXy@&b)L=zyn&CG63j5!R;=}?9cbV zvTGh^Dr*zGV3H17|7xra!F)wwKi%Q%bb{rNtO0WIg1y5TotS|va)#K$&aVq~oHND9C0Z<%9ZeOTvNruz+5 zQJ7ZRMr)+C**vd$wtH3GqVGb#W2wbp)gqnCxUb1tCD8ZX7dN=<*Q)p2dZ$EgH%1bO z9(QKbp0*z06F4gh??m?RC2=d(QIb3y6$V${2)3n{-LAQ z?7W)Io>b3=kt`Fc;RU}u$~|BdAf#&$j@av&ni!$Yh6^w&<e1YX`VYHSlAg!D8eJ=(v zp1>U_>OF-wiHW6oLs8@)t-nS;=ex`DIA?BYGi8G;U_FTRKL0I~qkIGXFv@g-Fdie# zSEt>tjfuvhK$hz>Mgaf$ZtEbDLdDcs6WrKpY10wK%{?@*>uW#ZV|!_%>g%OkES0o4 zZi5%$;Ub=gE{nhUhHUTTKHLhTxf?nlvdoZ0;w6BDC(y-1Yt~U}lYX zufnm~=$Uf6HD%Ug5XVh<%P=@R9Pm~^#1aQOd_YY`?XQ%L?QZziG+!_Xpdn5rxRZE+ zTY5kQio&kVOhJ%sd((GL<4EQ!aH^3e8J;AjiV6J&Zg4mu4*dsuAD?-YN8Jewp-K2N z3c_9oJap6>zvvgK>+j`_Xc%-lc<>vYvC+g699tf7dB1ls;t+5g=<{Gf|9xO%q!H6> ze#RDpg|?h{AiSIRmLfu)ZZ}8&Zp5HStIjop$$FjGXaWP23*oL+{2g$Sp}9JUZ~>!p z*19Rmm3_B$dftP6;Cm}|cTg>*0ic(yNu7xV-aIUT?-?_v#)E#p^<(T|u?Oq@M!T(s+r&S%Zzi!wW~EM@5PlNMiQdK%=>rsIyCJTsp(FQ^oCop4n->4TCs3_ zmcCgzyT1TDa@iddF~DQEvPE!V%V;ltr9L{-l$O(D2>uBM7&d5t{<Py0ccv1qi&y0=iS;Oxsct(MD_2IaeLrR#HO-Uv%0ctuC8umN2B5Ch=l z3B2t?@zaXOy-x#R*Xot}1a70(Q0x#2*u@^8z-?YDr5DaK(5GWzR2D6)(y{BiWeFz5 zb!J}AfuffnUCrk3LeD7CNZZ96rqTezn1=Fpcmt8NT)b1N8P(l{rB1 zu)R;T9JmXFvnfvUJ*%87$l}A{IpZf^2Un36LO9iq>MlR*neaVcLAv=e3%Xqjy*+0q zxiH%%qr`f-Us+q^Il#?*+K4m|V7xGZJuNOsY7WM2 zJ>)h%jo8P)6v8f?mBak&$bs`f9Na!j!1oPs6?QKMu3j@cn&ZME*KX!$U?jCB+__;` z>acJ*k@8lfBs;u4j z)V32330`jlsT5k~Lmp25HrsI+%MWk3M27DBKS&vCqt%{dd$^cjJQcV-w~$-7=dodo{oyni8 zYD?%1B>lcM{IT}tA~Ewokhp9WDl2ZjnSsUC*>jHyUM@Rq*qJ20x!tXJz98&Ifld(g z;!sdRdI`!o>W^GYRn-PR>F4B8Tv5P$RQlvG6NPpna z(5F-I(+7hC7;UKHyo`)UPyL@AlfcIqMCr=txS?F-fHCN{uWUlLc1T_dyd+jnoolXN z*0OgeuChEWHc1AG_&F-|M+h%-Kwyf#gV&6u*X+(4nz2Cd3e@Hr5$4Uu@kU+6_YID0 zE|XXyN>!OQ-a%f(1E2*O!l3`fp9n6rosy2`wTw9@FmXIBmL=oVwjjUGV%DF<+H1!9 zVWoNM1O2nI-}cHg1u`r3_m!aV$Qb?ELkQU#LO#*I-_?~WLx7BzoxiIuSL+#srHtWS zi#@rd*D(eKFCnhEh$G{`=Z^57*B#U&-tQ#k`QAoLvk__J2` zqu}fQ|8aW%_`p`cgXII3b->@xetvYXo9wC~c;}ZBkOYFEVpArhRiWjma(cX_*d`j< z5&NPL!q14}bf2U=PM9R>*N=p3zoB2Lu{9IE>mor=F`t%O6u}x_3Sf2qUU*G%q z!#=d8hdM^9K>oU%>4V3mLOK=?pri1MC@u(*jATIM7op0Besl_Wu_s7c(?g}@cwM9bs9|F4(!n^E+ocwzJHL9=>ywZS!~w(02oyCMi3m| z_<%782IgnNgHiXjA5iLvP4w)zC>_m*C8YlVI^f=mcMY`^IXkvLnkqv_FNx+cUPNHN zF2~g41+s*o!JTx!<^7E*NAH+B=0<8G5s=xdA&MsM0^AA_9z8zmQFh2HNtB5^0b{3X z@xF;V;5G@waj}rm^DFls#bFl;Sy(%wisdgaBE3XKZVCa?smnmB=5UdxLZ)i)aMwG6 z#p%>5Z;L(%)$j*YX#pipL)<->e&^Kd&CH8F2)Olpj%IKHF|%BLwlMHnWuS`wmm1N- zd8W#k0F<<-a$sLNK*H9!CWtu0PdEeq#$PTsXZ7}5hn1mTP*unUvUy=~jESoSP@RHt z*(lCydUu`+T|{CVFKd5%55#>Aocjv?50@HxAEiqFfEh1nVKJ4KP-gN;$W7n94NYa`5u#ud^#o`XOqSURr`vqDmCr=Z6kc&?We zt#tMk;-GX^!BqLi32e=8wL@$4+z{{@B${tdqrNypM($4Zh(fO&BT^bC5omF zpr(EHACcGN(Ms8lOWe`|j6sm83UUl8^6~+F`3dzZs*jOUhg|Nj3VW*~j%`u8knT!G z;NFrN6#?vRXI~d|riKP0!#5vyr1^Oak9L;kRJAjeuieyvGy^pO^?Y65J_eaDXbo!9 zp@9MdhSosqsxt3P@;&qsFeTocZOcZrl!KZYA^Rd0qLxFqD~GVK15dR_)GHh^N!Znu zPKG`Az+f!qnl8MfLa$ljaDVGyvYft7--)Bt{O+MpRfuS0h%;`G8%m}P#r!l+)P_}x1iwO-YIQc_a)0ZiRFYqu3T z3WnSFz;Or(zuAh+@;JK}Ai`_ELVn^3#7VzDezp7FPw-RIKuK1NlX?jk8PkgF0tBVuZp2i#EZq}b zWOfpgclr}0K+c19tP)~LZKk0fDWFHTuwA%`49l!W*g#I6JR&O+J-bB6M{}XrdaR}p zQgTMDsSV(YD4|A8MMEE?7O?v)<^mNi=kXL;wMWaW6Wf)WcL_Nm@i`s$l{j@{_5nX_ z4=IP{0FOV#B$J20IRAf3agM#6$qyQ;iT56{6U12wpS1&o(UB)7}$l#7RK(hIgwwVYSQ zL|0S^(_iFJ&!d3dmF=7$^Q5VNx4FM<3ZGJ(SAnUwtEmnc&_e9F7GP3;LfD`2oR8X^b|?awKezTjrwP zG>vCwBy_o9H{!ph#}y~>tGd{AOYAcwLS*A4jzffitwCRBv?*9>l?W9qf~&#C zpmBn0({<4sVj^o$qTGaFZoic>nH%FflcaKRa4G-t__RP{jsOW z$1%N>hzMK{xfT5D4;tOteUu=X1vB$ms(z7qP32WoBQn=#+97n@@Em_ z3N{()Hk_*pzLTrpMy{1VW9h3UF<}iPt38XWjDEi9bn$}IeWSgEH6gm`wpK(mpY!Oj z^K8u#n_Ya%yKFR5eEOCv9i(C+8q2YqIwA^E8>L_0VW9eo(Gy~$qx#lCMQ&|b+2E(k zHvDoCQqzM7>BO&BBqQ!{(x+$o#nd^Xa1(de?Xa{3ODJmi1WS_=Vgn0HH+~l1w~fV2 z@M(n9s5%N=tquET%uqduzs?QXtU3o5$vl0WO6@ZQO7}5ClSEu;83mb7V4;mja4S03 z7aj5^E5i0}R%&VbToU0g0#Pz?3d=Et&7MfXrk9IMf5muv?V(Yb7mcj2ZzQUTy|CXC za>D|pC>+UhMnSTW;wb&OO34NL?RnmCE&GN!32`9_Kj(Cw); zFQIElS}g7OkB*K~+lI9u*2(CJk#jzu`nMq8dUOq6>_Q{V^nE|n{)Qu*!j`!ML9t-T z$4t9a!fwSS!y10<>9bPk)CjY32q#Y^n5ektEYR%}8Iy!pw6 zr5YE;Gj$=a^pf&_qUl)dkClQw<0TTw=YVrf@dct=fZ$4{kuDT!5 zOKuw)OplvX1Msj0`Lphe$xf3EA$^Z8{xdI_-u^nFc%Lu)Q~_+t3f&TY@-N!rsr(qj zBqB1rpv8V~I}2YcqL+#1m6(;Hg*%n_6VCeYs{EzEp6xSurC#O2Z5=Kf!0vN5eZU)x zp+#mY`XN$O2$lwFbk=`2YI%+;enlc(xHaZZ=n|(|F7=sZScK|U0jl+l7Kwp|9&ceS zA0on6VE6yp`Jz*s@XKLiSlfZa28!&xB)fS9x3>a27q<|ioyxnVoGlVj&8MIcrkucb zP@rJ!XYaa<Ksx4SB0CR&BV>G{1nQrW8? zRIdkpc`DUyhvFIha$MoZ9PLK8!B5`wf!a%qj4h0lV2$TgCE7j^l6)A+psyO3@FWS$ z)xKNqliAgU;L2?kTzmMTnATj<+-&8}Gg z0;jvlXq5n6H*i@`5mC5gp%Ixb&%~c}FobU7vJsZVvOHoG5uL8Y*mggJ=rXJOvwW)& zK{7Rfr`rWge9K4;H#izjTY}6vJ2=~^zG@Nah3%q@%0x7#Xjg!$qT)ru$CW#P-G(l) zLV=g4z`Fm4t*@9^Pl>tO&uE^Ezh~B~E=x5#tI-OB$gnLG5Nj?BR3?WF{6`79J`NaB z3HaRjZ-Rm;dLKcUQKV1%YIeCb#TI)Sl5%DDY*7F=! zl~-aU88zw@DUTEr?*5n|qymv{o!M2s3iSGy`b}X!1hfm($;M5@$(|3;tv9wZza63-@b6)oU4iY09b2g@@a~ zFn1IE80tPeMa;ksO^-|c#MYbx8i%@tv}eUUPUW}La5D?7h|CQ9S?iHA$kikfc5@(G-}d;t z`x8Y{7c%1$8u@!dOf;&k*mncZ?$x;l~5pXtMumv1K4OdJJHiZufDxrQs@CR)^$!Ev77fQU(PG39? zR^ia48X8QXUz(e1M!ecXVI|$heopA*cU;U*OZ3alef)4ec3*V){$?}Ie@}=f{QwT@ za@?h$i-n{&o=p2D3*7@nPj7~bY6N`vdGN0m(`|euM}y5gX&A0BimojAL}CV4(em5S z&nZheu%@tZmw3T}uNruZA8ZmVi{U8Vt{qME?b|m^zV!Z7D@k2_BGxzrV)j?B?^Heu zDSA7ssD0W3R8Rb(vZRAm%$;J!a9iJ&M{;oT`0@DAn+(#q&vPbsP$xM`Gc$<^4>D;p za0>~a5Q&G4`J229?18=J;8x=cX6=qg*y6MI|LdYnk?^ zq_S&V22FVF$;fhmSMHdR&q#SU3tcMR_#}6#K8#-+m4dAa5S-?t@3|&p)Tw{m(XxI zH(WyS*E<))c^{=n#!>Z-qU!zY!XlKELRufR!c#a?tJt8&aHq%7F>)`au?LghAcJ&{ zrz2Onxbwx-LoQ!yX>PvWnqvI#r>k%w@3VQyo1Y3foe{(+PR}Yj@+~0|j+VOOIRKN| zrRDq?{eK=4c3A8Qfxv7!nGiKRb#i?C84^*=3Z99hYiu!caYeBH_k1BAvQZSH+T7Vu z4W+>cxjH>PeVi(Nsbt^_JzO>$>l63V|L5!Lgnt9zbO_Ml->WIbAX_y%ove~{R4w8X zzU%$wJYt%f&AW03-$E7V2l5&k&R!w%^QsLPp~)b|st=GTx?$KreUMpX{vYd)3gsn) zS*%B^%nIr!fPAD^|AK^&FbVl)nFwKz1A$All|ETLRtnu@78&D8zNDzjucy3~-<4Kp zx(1yVZ%ykstaqu50&S~p#6rMm0)L6y`re;!0!3Q|qOnk8QvQE`v#B>*Z2_E+^Kb5g zyvsQ^9fA2OePYg+0iobv)TcYsChC!N72A=e=ws;vw6?5n+fgLNdsk>`j$7@*N);<} z@9jr1e{t6cccpqsZ~hG;O3|qO=ww?DBIqx0F3+s)13?3HtM1dUb{ID`3Zp)-lT}VRz~G&oaXqpoF>r;xQy^cs34| zT^u$hF$7)K>2E%(=td-1u^agqpe)O;o(Wmte)E*>bqtcg)ocfg_4?hwgIC z^rID#x-t?;YM0TywK188AK42m)h66+cNSG0mIofWt&jT-EnE4E7N6RVjaI)AYOl98 zA1FY0Z7`_Eyyf=dusfcm@Mh2* z39=)0b@v{-)?>Bis!LK4B}tIPi)@zYm-F!>IR*wsX6!H~pb>xIu?mM#lACUF529FD zuPMa5RCYB^e{9NgQ7drhMDD})*t?U=qqL)AQ1yc{soA=ZHS6S9GEZB58qW9>|Py5ipw~ zQ<{StEkCkb>0=p2u5oA(#{XIGLj@gYI5xt4+7FC2> zLEec0qcoA>&A-+#GI-gdLgkXHQ6xPwy(zXje7Lg&YU%!G^=OEAu0MX7ze-^#axAJ4 zZAP+0mCFW;Oho;aX?Hiic=q3WsO2`C7S3%!zh# zwK46>6EEGA>WO>bKgQ8HgjEp4NJu?AQb}pJ|5>jd(_yOV(hgb09j?Nxqer#~8_Yql{Zwiz}j70vTJ4T$cU8MMzYoS^W` z{kgA$@M%&Y(v{t46}U%g$OMggpMrulp8^yAak0sxA;<))`0>ftG{Uqyjd=}k(6p&b z*x~B*wG7^i_^#rzL3ZmY*M_v`pB7_>bR39G`O_S_@}HUV7^b3P)Ev>bOFeFe2nXcY z`;uILjrBge%cpOpx@pboF5L&;>bq1%$yRGJxix81L&C#e+Z>K7LJ1((^3WC|GMw-V zt%!##sN{4-(UY*J&a|zJC}aUrV@bIFW|cUic)$DWyj(Z5+r0dDH9y_*a{N(W@1MGW z-auppyO}OGXy4uX$^Ag1x$@E)e{uYP^)#W#)n)R%3%j^<_J@;Nk<408PCG-QHBeI$ zy%Q7?y>}@|Z%ipZ#*NFa;t)3nBY6m7mr&b8g&}T}eDn86h{X`JE1XXe+xfDDtZHTCoGL zjCXjWue8X--PKUeQYv9Us2RS@pCF>M+Ku;a@Y49^H`Wb?aPPwqJ8V3>t2#P5B?A*S zQIN%fB=Y{X0Q7-BBIjF2>T|)RJ#I`X-J+&=ZTTKf4Ws@$C8~KHS8<`&W2tejIJUVG z&RQs@&97IeJu$h-{M|!x?c~4LYSgY>xh5AGx7hn7HO3c_1RdhGH?AjkgZ@9Fz zpRp>peLsI}cO*z){$-05(xd zO}){OR*)ElV@St?c2wEUhIGsZ}$>&iSBl__n=EgB?!B3E>`8@ zUws%(e4W>ES>R=DOLj+57D{4CwY8qmwCETu@o5AicM2IUZ26$4iS|2oYC zl8UNAu4#JcN6()BfH9rL;Ju1kGlbGut&Sy&cUt0EVr670z6NONUK<*)o!m(JrsX3t zdfna0Y$%sAzT$bpUh(=#@aL#Z+|q6+Dku*p^-0_J=YY?9D=iNcn6?Borf)tB)N|eL z|M>Wm(?hl=IiKA^wXQ&(+jyXhQ~cpG56X9!530&&PW+862?K8EfDnU4d99FAiIY9eu%f@GgR zdg`tNb@W6sUjUc)KDT~Dk}0mUEE@qI;}g^|&%+C@*Y4j(!@^U7nfcO(s22#EPWi?% znnXthyFtZ)8;xO<-8YR6P}ZNnB){=sChInre9y;D5yg?KY$}-?sw_GR)on1T!s&&D z`^0k=vjl8rvDi%SS~^brHcymZ0w*zTRIppGV|65w#q`0D+cZIU)m*ah#9%O0|GPv| zaIEkH!MT!xV0aX2e?TR6VK9o^8%6=-{i0T(n&_OYblr~FX;~n~Zf^Gz7(bcm(@?I? zffdHB$JVSmkFAfg`}XDgOzBID{VFht-gy5M#7?~8Lg|kdu1C=i+TgQ=PY&cT&pOf(pSoY-~jKDCy&z|SYOxJV*z!s z?Xn&P;~FZ@RuXT^sO_=hgLn?yws;;HSlp@UKDNoTGxO{J#HmSK zeOu)9J0&oZUxRJP{?K>-%A<%*k@Yw9D(PX{#|QR1yU{q%NtitS7lV++0M`|6{eY*- zPmnMXHa4~{xTnAmPltwk-9RVLwX$O!u5x{MnORQEz!?3At5}kSP!7W3s=P;R zWE~m)i`h88@2pj`^dppNZT&{H>H}nuM_nYQ@=Ot5sggbMDDDIqlf;aik&5?y^`VHp z`Hc7c&Usd#FO{YFweatdL1#0m&~x!X$iLBVIV!OqT5;ojte}s*`}KgPD_9@O*fO?M zutF0JsBvh;)N**B3QdkNKI0a$b{pB<5br|>(a7r*t1S0Vv;}+<$v@Vq0$& zrM1bI{)U%y&(-4aQzqk3daC=k-_cW1P{`KQv@2S|q3atPOEl}TNo*8fM!LY05J`ax zuoX}xibpUEv}?{O*oCE*jJL5yG$I;N!}M!snb8Li#G54}NTn&E2hUe{A>w~Hy%qqX zWI~YtTvH(t7_@c3n16!})+5gWMn@pgC+xyoN_N-5A4D)q*h3@B0v#R|bPWVAku(Q% zwb6{ZAF=W~6>Fv~R_&$}KR8*8f~9~6|L2X9P~bTd;u=E*20>sM!ZvZGkcdCQe6T7> zWM~Ppj}K$tH8=Y6f$uHwxi=4aE*1ivHx+6b{k=B88t8^Jk{)|z2q1Dk^y%z*qiTaYimKmDY*5~k*ZiIe5{-Hr7c z^IS+Ri&PBWZPQ7~9lK#1;6fO}sip84tMyKVuxl2a*dLC{%E@FSZj5bXD(hC+Z(HKDn{@)O zqYC4Y=CT@x4svVq4S!rt7t=NRzkWqmm;T`JFbgaWQ0&r6Pj0e04z(8RoxNgNkwg!w z%9m_Tf1r5jN>2tD4Ot%<39;oc|F4qX^$FJ#(tge-(AUus4w-W)WR)}=IyG`3mt7xy zoAfGP8K9xvmi~lKIcK-5mO|5fmZnU)q!iPW$$yq7HvSXXD-7~6w|~rRX@;IJ+bhAf z=gC{=!q6s8WZ3A{CimxTP<$m3=e09s7I*p{a1pe#@VQP3JVH}QzH3V%E{bW>QsM7U}wwKc82{;`xS*L5AIa65Rz@Fe`4o*l zNDvs_gX(MGBa5`@l(y%&S&9^CH}m>qVoKIz+J<&sQJ1bgR5dS4TmFjE2{e}b7hX|y zLkXpxZl`DCxY#T<@^u%tH#bv&9|nRuaqnWGtShQG=l)N7T9)q%(*z<gdnR6>Cv&*%Pt4($v$aNKfxk;aJJ#IC_&_sz!_`LS zk20b2yL;o;>yhlcX@g*&KRx?Wt&(3WlHYS=xHIzoBx<=`iFOg(UzxK>;)6InYTSLS%t?%-a6ZuEikZ ztC?T#KS30-6Mn85=lK1J^5k9w#c)r{CBA5p@Q!JBsNF29>?md7qf%&iq1lrjo7b z&xix62On95W_~^ke(JESnC-0XbSO}M3n@VF_81Pol~m^0)e3lY-P+aa^$eI$K}rf_t1LOS}O!hk{o;w|~n--gdom*i?q{ zNjCnq$iX!6;p8vQOwYZEyAe(=m=2i}%Efy!JJdzWv_<6&eiKb1pyrOW$k)w+qRon} z?>^>-De)c_l(m8mo3pV@X7#jZeobeE30>(O64_3gFEWF%l+I8};BEa;P z$uRkCFtEO4@;lQ{6rE@8F6U?BgGt=?>#26{mo(6CUO#?K30mDUmOnOFU#o1bV%KUv zG7;9)5#Zo=FBCK-=AcAJTcGFl<3e8r6VLUq#>)M*tdiIb$MKO{PEAiAOvOhtw~vw} zM-v!jU>CL*=~R#1)cHh<(AcIZV0j*c2M7e&%PFGpUz1IWExXu6;=dT7E;6a-1^S_x6foz!S(R@~8 zlTII+>G4H~K>s?I!3XNwdb`p?j*B&p5w~1d$u=Lvg^8T*8#p<<9y;*k&masGbAO$6 z%^8tx$4_n@WA>###FL@LX+_kxqoAXc(xW}tgHnhm*ob|^;3#%zjPQ~ee+kcroh7Kxr=zU#f4>7ZHdev#!OQ$MP6YKXrFzA} z?T;&ht~y1b8O0t;u2;C9V^|9AQ_-C4Tx;t&v>dR0b@=5#P~l9P+V=IW)))MT+0|Z8 z*ZNlwQNNpZ_JTGvxJ0}ciwG%^XK~kEu84d8Mt+80BinlO<9)n&MZV=)g%z}*Zt%>d z@4x-jHM*p>y!*rL>`2fZt$rxdvetZXF0;hjOKXv7K*Ht3ct9{=XrGWsfI#Mt#pOyj z=^9lw8$5GQ_2#c2Hpj&NH*4Pfy9Q~C{_3MErTtr3?R;zInR~2jXB8{bbP`|sAN{>Q zN+AkAJvEhjZdrh}Ec^b3+LKTWRFx#)NFl9An-BD%*fj^{;mSD!wA8|dM*YtPZR5Y= znK8W3uV6(V0hiNAi@vs+Dc zh4Q((FiX7{w*M*oOG>D72sp|(OfCANaadjC6sc$oPZ;=yOH-Zs+fXnMi$1Ir5gFQA z+R#R1i32a}$a+(Na?6>fYCp=1*Wy<*Xr2!=Oe2(Yf?iN=ccM<-DbB_!E0xly_9Xd8 zD^&dyQ3lFm7Yprmk~SauprYteScY2t3*CM;RjRt`7xMKSdN=a1+)KlKaV(U_$XPGa z4^eDcPu`8FKR}d!!O^(Rw92sesg8(K_HNE{ zmzPXOcJNr&7RRJUqd=0i5*8+f;l$JYG0YX$Mu8&B+D6??&%dX?g8m4|z{<(`qywX0 z$BMV8DH3J2d-FNBk$vv=F1dbRAW-79nyuh+dLhVuvZykYB5aQcA1RvFsJgrhRF{)GsE@cC^&n6PbNt36-ec|GMUuom^%qp_RwdrFwG0rJV( z)vXm{QrkZW*m*tL-o8a!_~1Z?({cH=FHN1dU#3bKe-YOlVhp;?*U3J@}^)8vaEj4vW#RZzos?xC($L36n8MjC_dCUh9T->DW z+HXJwQ#1|w)MH;5;PWQRq&CGZr1zba^o+kyMZ0IcQpAp`;pMnhRwE(!nI%h zNz<&|cH6j^TB#Xd^fw-m8`_?Zx3Jz|@&2Y9$zkC1;B&mUL|)qCQe~}n0bCg(1w+c( zArc`4%~Q)b=dTYO)jj;mQQleOI06rPb>;@Y22~V1Fj|Lpg22cn0ASXxcD6 z(-)3QkGVj>0$pD}6wb9aA2q%D7|~_EJ7~S1)A{Z2Ap}9LUF%+^;tk0xU*LJ3VT$Xt zzQfcjJN0DbCR$aTOPqlC7SmctK)EAW62Vo#~AcG zg4dS!m7m48l$o#WRde`61)eY3(sD$;nw}0Z-7W9O7>v0zv|jMM*h@9>K?2KtoRtHX zv`g!2ThOxP7@6HMQxtdiD)DOF?>tx3%R{(qs{dbcZy6V5_w{{C zqqLNCsR+{D9V!h~1sy zpP66bWt#sSQH^!Lay+h^A6%A3Rj2QWg30DLay#9|xCHI4_#Pl3ua6XrNRXrqk*>^f zSoQh(f9mYl?YdE4{>pSNvJX_ObMM0z_S1XM5aAN!P|R0Pl0=LTb?CDHyA|wocJ-rx z;J4?ApLc&ArU(4i%gksGe@jz!Lwqu8HwNW3IQ z1plJk_o;UY-neEoP!lV?PsuW#{$=@fp><74!u`iD3U)ODb=@hOYivIY9;G4*Y#@vp z89IonD(~d*uX&eyytXb&^Hd=Cp9mRjDM7!JI!H&2nS#Q2gsbr#|Fb;<@59x%AOWyQ zDM^C+o6xWs3zyb8KTM0iW>by+ffa29i82)`-D>X+5QvT0Onx$LkCrv^JK<0VR`LMGfW!2FeU96p~8{~7|{ zy1?s4FW$X_jt($*F*{Iu7EijTyHhqvf3Acd-aL9b`qg6Bd*n>E=NBVQKr79C_q3}S zW|@`G=Q~G(`RXxuwubyrqiVWo$fbP@NsGnpiThGz5qs12DV}>3vOIbr{brBDE-5$E z0?)-SwI@f~1yej1vGuL53vH)UlTOy?FC zC+rrccv{2g<=MFs_T=DdQBWoz+Ik}oV}|hY3pxKWilcDC1`JT(`w1glm7d7J@cr6s z8Z)rNZwU6PVCmb(h=2d!ba9Ey=;+b+!>-4^62YQU0)gToaj(Cp(f7CP|Lo>Yb%Wc(VVh?-qC8pzaUju98-KaYuEsx)5^*Z06CC>dNzPhn zW=2O>Aw>Q*(+%$btti>yp#D_vJOZ00+xNA?fZlO!S-?3yzPUz@vi$sJrSn4Ml%ioW zL^`5_m;OwAD+q5y#z)?~d#BB9(Tr0pwGguMcbkZc-`vqx4Yz)lLR1{9Urc z1tA8}qQkLYr55F?(Ed$$Kwuo!e(gc*n~k_IACtuW+F_clgr#fwzMRr8basPR__CwxZM)fD0q z$Pvw@$RtCCZf^kBah|5PK|>+@u%b^V;Ct{}OQd;us%s5o{m_jc`&Sa0x1@B@oc+AeA9 z`uGFi$iLa-A#F&yJDHfMBvT=_L`-yIa@<2v|?)M#kylm3bT}~^m z4AKk?bek+eGJ?05!g55oH=P)tj{@*Bxa}wtjw3!%30P1^y|xh+diN)}R+*1g0l2ih zXT9I&C^C?)gst~T_7bOZzT=wc0Xx$uP@o5f$Bjxa8x1Mgq!&C7KYdf77PDqbBxW{D z5D#s^brLz`Sy`bFd$K2Z6u3Lkq^vMcnqXk-ZnWfEowPN#&Dh&jOSeLLfNUl}GwCok z-7jcLB53QBzU3yA0`mG3{=*o{(wt(rOE(O}F6tS{|!ZKq@khsrjHx}DV5+FB&H^U*lal{T?&Rl*!NkiQD(!xG_H6>Bea6j4!Es%aq?KfgZHu1g@bIeB5Tm1<*<*))GOxfO zM*c@NpF1DOlp8j9Ca@WN|4M>|nsY}eG;f{>bDqi(d8H#t-gcto6@NWZDpR*Z44bw`nHH7`ygeOcQtkBIMRrc&b? zA>CcJE_!fW$>1}`m2{WV_~)ycuLd->50&N~i#wB{JF>W1hfvDkcgDavq)4_#zqAh6 zgk-;)ro9^?v^w8jHS_SN#!>w#rAQ2)jq($EOKq!>Y2U85;%#eiUzJGO?Yf$m*2Lyf zA1!^(g>+qG**FikOTRcx-@|?peTq6vSRd~uT<;0{XmC1iM-DHUbNQHi>40<7##MKh zI0N*7DF3XIq)k2AoesQJrTz+~M72&t3Qqx-e4t{Rs({LGOT z<;~gFx&1ZcK9O50qn%&-hu`{!-%TO9&=blTHh;X+y9Zm+llh|H66nLn!ZcYpCUTC9 zW9i2_bnyN0oJY)zjuSy6$x__N1(cm&ZMF=OWns2adX`?VqQtv!r;FhQ$#oo*X2*M# zr&qpO2#IQnJ&9w6zMJHU{+sP~(XJ*>e^a0KZUUd*7uz?EZ;@Wz}S-9qWG$;KGUA?_Xr^%Hg|$KIlOtBYaGu+yz6iYYC2GEj;?C-g$|~^W0w0TlZ*!X zHNnPi_!x_i4603&m^ls(^oq@^iN! z8YAm1-Xm$*bgkf>p54-H$0fV1C9+#kDnY%-zck z`1w)0-vqcPa;YQrPa>z%7dh zn|~>NYC7pKod@C-JB+Bl*T3df@cmqVSvWi|j0X*$7C#}@PfVu7%m;kE*D02G7^(W`<&*T5`_rT?VonRgG}*TyO0-hM z$WSqqJ2!XKsUK{QkoE-cl8jtxQNoXw=S)te1L|HJ5zvOxNm25NaK4N$)@Nw8u{;lq zW?J3BqdhA+#U_JLFvWAAQCi2~-~9o4Z$5MibHjcdwfot=r1N*!+iJT?z;;_gp}X8s zZj^UfWxMpVgyQ{`ayF4c)2R~9y)$&l)%Qyc&hASj8;Lv0ZF6apTq~4xYt(4}^wiR5 zEQCxXQSdjKbNe#c#f2Gv_AwUwa_X-xBoZ@A98O5a_^3#GLY=&z4VDzmfV@95TsJ`q>aW z9B;T#6kR22SDqm2EVwB)S1IFTTt~K+wmGI7T{sy!x;Hb@7m$|2{X#43vT4ZH)oP4y z?q-RvksdQZ+q7!wU}{-*7Qe|C%t8shTgiV|9`T@M!sl9Bkz&2{#K_l#*qk#$%AQ}5 zw2QM_ErrRJ4H7>;`~8cgu)G4((WXSf|A+qUM!W|9%IKE93+XG}iHZZ-~&DC#j9G7T_Er`fGN zpxa%4^uy~-7+bgU3K44QG`f4_$JT??-?v#rx%k7oL|8*hZQAD8p5XLiuUq$MO%;dd zn`Tuu{o*xCqAE9Pmn@snhXxN!-)d;?FBA(iypouVdtT;6i83Rr?0ei;sio`ex&^0A zcAJB4r?M|FTKbcOf<^$RZE8(U>>9z>H=1l+riLH|>9AqO>TZ}P^T;riXcGs&1xJ$4 zWZC?RU{2r*IutG2qMndj z-gbhWR`)@pYw`A(f^iX{-*MzFX)&;GuwbE%^!stDfi^{&Jm~n&Wpq2(ZhiiXmr)-{ z<@vqRON(w3>(jg`r!QvNq_;x^60{6?v9r2_-*>*b%(OlW#IHDe9n5KDgKM;2GHHyRAzS zRjo6#MmN?$9l+8`srlKj6%t}bOKIka#z|3c)^x>8zRof0y^32~=^Zd11_D_B%&!{6 z<_XU94RG`&8Z6HicMF!Oe!lA-Wfm#6(~*v%PSzjc)7B`u6$d=IL_v@xe)XIIiMPCZ zZzE`6oH(gp9#xx7>_K0cS)w~pJMAoF%4*%vW~bL8 ziX1LGq}PNb6s?UNMW`S0Pw(nA`_Rxp)pLG!!)`aD+(XNs`xza(;p-;gxq~20+q~8P z{hD13L9weO&Nbtujwo2K)PT;Bq_xJ-*r_mSJYdDEY}zdk4{3?`c8dwiRg&c9&3x%f z@~LeAOynTfjStzviq^s2rax!jUqabO{xVwbuU)V7EIb0U!c)ZzX{i0V#(p2giN3Ll z8#0U7Hc>IV3=xa3vy3~1WpSqeIa$(f2$V3GMXY4G?lHu>{CW$%EZ|3>PC}bZ0_DDu z<6*tgggnS^+-GT3j+Z6@f5-tkWjH}Yl>YJdMBF`6wxYjk9H3EN1pwC6|I|1@*$cGq zF9T-$?54Q)F``WkQSV5+0!0WNJ6X2wrT2;DRuPHJ@=XTS5gaj6VcC=*oLspvGa(-N zR&lwy)`FJGyMtF=xxSd+;BXUM!-UmJXzrFpq>-HY{ z(tMS!vzdsFD3?c^@2c}C!2$kU2{kt09+_?v2tv-*Rr&3LEecalwA)&%( zxNnZag2KTv6w1Z&EWXs3iKHaQ3@Mr7(?V+|?u7uI|L=>E2T`RF(Xz*owl#w(>#p`07w@v3`70RhZs~HZ*?TThh0?=PL0E3&ck}k&MDO`B%TC5 zxc5VhAbA2O#KWuyS)nH=&htQEk`-N0kEe?&xJYHd*SClo=Z$sAl#8~e7_LhEm^k&< z=NVx~Ap?3op@S$jG3$4tJdL;bOK4i}h_M*I@EXj>IpB6Yk9_37w$*}xC6md(SPXJq zVTJc9s>!Qscep*Uaz6xl7DU0j<)hXyOhc)|^uAeW7>3@pmfpW9W9+7)_dfUw! zO+*^O(?Rdn-T&6xm~j@vmTx56EMT5=_{~ay&Y+()5Y`m&{e)Y)$4p^QmTHs^U&T(4 zo^ai(m?W)_8kX-=SbVL+@4bhiB;9myZV;~tM$wug z3GPJ6HtzF_-7_l}Vp+yakxMI65Mh7z4QV0Muhzsaz&o@Evx1-t2%^i>W;LQPX~qrd zKA5El?P0=_B8L9&Mw=R-n|fKS%Ug}8qJT#39wN*>Y@mjuMMv(=LSM%jphCyh-my6; zd0IugA%K^(6iUaw6`aWs`UD>v6;W&+$xG0>H!PoIcsX(&mdZd34=BZPX3bV@N7Yea zMnW_yxadfv7*1#YCV_|@Q@Cx21@}t$1Q&_L4`XOA*ecUx>ktSAkG(6u$I=7Z5hyJV zK=Os3x-Kj2R@(49Q%S3LtJ4_Vrm3H@@QsT)e4J=m6I=uD)~H~*t!73oe|u6H4EkTh ztxr#7sRfGe#nv4DLRZI6?&{NSTMy0O6kw05I}nU?%w3ZZwyD-3FkhH|8~tQKGrmJq zHhr0Algk$6T*Xt3Ur|za3Xu~`iRi$hrCWHMPVCpQB5Af(9%IaWlKLuqgJQMm>f8)b zV~1c%A)yQ4qwTAq2JhyC+1-HS0GPt4{EEbNU3^8!9y2=K{DAa|6w0h2Vbqx zFI+Esm_V}5bLF`h9!qj~x84Vat+VqJQc9dS?*v^W0&9F+jt7)H?qV>lBG-=`yKC5g zhcDdVNgH{wZKU?*Y7XoI*ZiqWpXLNIh+kjOQ3d|gVS-3~3{ zMK=C$Nn#~ftb5N`L<1WuaF?{ zVc~|r6;jZ=bNBvZ3pHXs2$&59@v>QKJ{Ia{XhNF0{lqLNS^^}!;1W?uuT|n@smiAA z|1LOyvI-rqSegxPVq7gDdA&POgMhM*GGVjKpB9iTHPD`@FCL@U%sm<1-s+4sL1p$M zu<3eOp2mgEdy?ZS2g8kRj^z8@H8=Kgw z4qfSAJdJ#bB$&Csu1N@yp+Yh#2XVJ#7IYs~4a#nP858Es&^v;VSn&mQxT(!+9sN9r z{e(GH+XX~O=g&}YtK>pzM)O|L!c)KZqlb%3`1W1AgxW~QX09J=W_20Unf7wdd9^In zR=a?X#Gs&{)$Q%Vzm}Kl>+AeKS_FisPz)dBX@Wi9#-@<9w-Tg2!58d4zBQ{}aH<^F zVPM@yh6(#V7Iwi8Yv49tDcy3c1Hz(bOzM`4&j@9L7JU=e@lK2#Aw=foPtbc5H1mF< z*a%SJyGABfV{?_=a|Zcn0S`MUOu<i@7*hx!!cGRVV9)8BvOlI8c~mYxp|&EWFJ-CLP+`WB)tIy z|9-+4hmgF9v{DHt=%Axv8=<+R4DBMoc7~X7T$2-MgKyGAw~ggY7Z)MLFQgS&YQx;Q zTZ0S_3(@Z^<)dknLeHtPL2*%x8p}|Vab(K-Y4D56WThUn@-6e!w7JBaKjrEg#gjyj%$RPz+9{*>KtvpblQnnqba3dW6AgpM1M@u8}? zoOkJovx{k^Hn1KouV|(YQI>gCt>X`7ibYA0Y7>~)`l`^ss8L?QZNNFneUTCbce;v+ z!%lj1Sc77SSO(Uo*mnrBEhnHW0{36R7-QPu52e|VUO;j?ycwNi;O}OZ)e}{XY-Lo{l zFV(P;#f3mi(#6I*k|Z$f$wsyKu({IFbb>zr#iRWfJz|uh+~-fPtJga}Ls$?ENDX*S zrE4WZc4oQE1cEqvLarA~xp@206d8Unp!@QKpzMT1Xr0$5r3Jf)7VXaD^s0=bNtrlc z=2^{oaj?_o=00nNmciC`;p(o%`nO7b1NmclRJSyO=3bhVs14WrlcjQkR|iDNc!7C3 zlu_k(>?lu63AlV;4j5!6fmNv;46X)8r?CA&h8%AJ=?%~;b?*9^q=!lDp$G#)RL=Db+19(?)n#H=m54}p#Si6HCh|oG? z8qEV~<6gRt&$s#&Kj(E*}QY!}puIx+6f1q)$%=Vv4 z2>TMzg4e@YOoPkNLk9*mky^0hof#YDRnT!~-4>+Rz)o+Z#+aSWu`l^`tWYRp-)-@1 z3;n8lxS%@=EzAD+n0I_3C^N8SAcCn^@P3Z}h_c8o`M@LSRov}sl=TPg$B-y)cCBX_md#T}EOJE0ZRrnG=g2JHIShu;87NLOb^pNLF_e0#azPCzsd!7iVsa6d}G>jgE6p z+m*S56QM^^=8CW^@k&VTwovFBMeRNac8lf$X>iLGulz<=Izzrin>*;eWxYlSkoS zLBm_EQN!SYiPe5x*iNpo+#Tmu=~bgjctO9Q_Afkbu~=RkvThm|x-I(4!E@y zA0#08-(}x=ofo1phI*?k&B$UZ{y5VOZ&zgPun6m9nox zw(jAomi-yc*gMj2c9N(x!7d|Q5 zO>B>s&Uixg*KK=Z)*}OUztJpS*%La;A+_tdC*W`nAsgNGr&s`FgVY>toSE7 zq6z+fEptq1TsyfSbUp(2@3=9Bo!Uk0p5fF)e2;F*dbd4(kt`(k5GWCnb{p6a3~Q&A zpRmH<69F7td%f3%X@2#F$D2!|`DgIUDr29~xUm3*Go4YtG9}URLyE!H)TeNvrQ_|` zKgx+iXgi$j^gi{N4^|2{G=NiXXb|rdbPhSwDovw~y`Vqdy|$sfLB z8~YY>_{M_3T%1Tu9SarNqrJ{)1*nctQzh8h+3(%Er;fO3A84QQh%cGH1<3(6sR=MT z{^&Yn? z{U|b8rVuolaI+j7braf0=oW#pW80*ltsFdPtUwup$-`qe84pxMpu3*v4QMP;bIGK2 z#YG(CKXnQ+vZkll>K3a54mvzMHS+qA)NSu*NG_JjyNhpFwk{I5EqZ7*6|DySPn2^a5m{M;m=S z#7Jh;CgQ|zo&{W=B)Vl#3A;~@mb>rjZo3;VHV2LSsmcSPxDM*D6R;Id5%LERRcRa< za4_6Tyn)Tb-@W6zK#>Aq7Wg@_09#0ILkm!}@06&3q@&y(K<)Xetw$a!rqD_4gC9Qb z2V9`dHM#(>v9;K&IS|QY3jsM)q5>ubW%dEb8w75fzdEM(&TB!j^BA}wX`f9scvXV; z`R%LhF&vvtgdD|y6V&R_h!ZdjDF9loB)$~!0BF{mGPjKr{U7czwE*}Y$%LOo2Dl~+ z0L$?@b5;5O`8?4_>m%+Nc!LCRMCb}MOY}eYd}37qQp%c73-vDH9i_ndNG^3UCotob4vgra%LPo3LS@4HdpsjGNtb)!UvAOoTbH9Z{ruwYLB!hVV+Yr8+0g z&OML}#mpNo2TGM~_dCnk{v|+5nG3kNBQX^GMKg5R1#XI|0>i2(!+@#Rrq4dm@x&Ck zqa@7SKQR;DLO9H3p<}azPmekQbjG%O!!hj(@|hxV%EvoX+}mQnmZ8G>47e$2*D}Cz zbTOoAW891r9-eq>6=|Ey-2l3Y4RCu-au1IHV6FhYvmnQk@p1rQ1ivL4T0UnxRzSSg z;lJ37lp+)_*M>P6ec-JYr`AqV$7Tri{dpZ$h}RD07Q}$)k@%VRr;!!&e^#Q9OzVI# z_7^e*0c??ZMcV^4|M|{@9%exBzqt)SgXXipD5(p~fq=*zOgi|O`qUApr^mpOx z{~j{`pZYKwpeY0_n?#I1)24FyQqTXOgZp>BB>Y>r0_GQl%mBYmWDymx$`&jV#I!;% zD_SG|7xFAqf=i%{%l{)Q0$je|P5|c6lg#@RGF2f9GK&AJ;{)LgNJgr4L=uBbh=3kR zEFOB^TnPdTD1c1${`!xW%1x3_r@}nP;~7IFowWab$#JF4*kC*Gu{__gH$WJhDktea z+MTckaeuxMymKro+~KdJj9Gy-PUZ0wVWh1Fn(<8jnW+#=fdq!|e=avFTQJDpSI`Q( z%K;)st0QP20YsT^KoeF7WUUW)y&rxXJdVMvuz!YGF}78mC50B?NFiCH%NxL~27Qc0 zZ{vg6FO69t9DDoR`q%5Msiv#!d5c6;)Tz<@k)f~kdVozR=ydzIZ@zt7z&OajqZts6 zpXj`B7{N5(lOr*g@6Y1N*BA-Gz#E;hWJY9eQPI)gm6n$N`}>e(?f~p^w49$w4$0yKOb>=3{bvEXQtW}@E_!fEga$$aZ2kh^(H(l)eQqrbbj(1h3r>_ zpFTWwFoF7C+%Ykv{(cQ|KSU{C{k8+he`!x`sX>(Wcjyw^{!XsQ&n$=L`0lF zT6LdyeBgTh;u45aNA-5!Y1cbHdNi6o0g;DU`IQ1`YfP8_BO4ll;0lYMML<&w1b6G% z0FrT^4m`)GvCUx8-2@{%CnNbjf$z%uG;$tTXK>2peWH7A)Idb=s&FFm@OOXPV7RdH z%y==;3=Lf`Qqb${342$+O`EAp??WpEc0)`xqgH zvL1_7AnIy+4rv!tXb&9PBX=dS zCeq@`?y1F}OX!t*-}(4NO~27C2L-F*r+WE0dvMVTC-C1oJUc_^L=eD%pk9=bum5Vq zq*M5cnOsh1r=xVTvqMYiW;zI8q|gbUY|qH^(j|5PCbulGm;AODl4BF%dtto9CgOWR zhPK-%cR=_s8;4@N&|;|~@X7wk#IV&sfiw}z2vkV{Q+F3RTtf&}#g>7w5J4Rj1nv|8 z$=5MHg5;2WNj$@=xL=k(WvPU!v>eKNk*za-<(}o(qdb0FO3)Dz@w?ITW{HRcXh6jo z&p_1sq()vuxABMzF0^$bCo2^%D@6W}Se@ww5)FoUyN}jiBdu8qx!Z`ChzJZbL16Rp z87Nu^$;E9&ZdCGM}^ z2+3&+x`s2wg3~MC6+tuN?CN@oSz%^@>J-@s$b1EftDpX!%x0g*cN`yGT^t@B&n`Y4 z^pwOudHuxIsFnIO&F5Kt(^F*}3Le$|uHR0;pRwQTBvaCibidWY299YE^&rzu!0|ow zidpDnZ%&UFBe4ru4m<%}wUm#d@HoMreBm3GKOy6LZ`v6lnaF0S?ois>R%(iNnWYUF zPrvgcbRRS1sL(*KKKMElXTqA;7y0rNTmi8?Rz!3~)oAxZ)Jx-XK2KtGWgf^WUc9Fh z=Cv6O1h*|`o^J$cWoDJ>-N?{g1#LUcQbFeXG!cap4hn)UYngzrRrv1-##c1AwY9OT zliZ15c}u{J?zYDri{Z)@PY_&&A|M62(X*_EvxkAG<>iPIR+ar&q{y<@aIyhi#d#VQ z;mFx(20Rf#0)WEV#6jeqX^l(!z9mo{kW}%DiJ(SzFU*! zA>Oug$RJw$0z#8ip%d!f>`&)hYqWcW?OCy+@@jzTzS$gvvQ_NnWqi;r8G^2U75aPw zRIe2oKGie3rL;#Uw7hOneIVbXhJDaL~4f zc0w%_A~u3v=d#wPrql@uByK2Al18vlfCv@HjcG$l8@`5+rmNO7pKSKBi{~yb8joSm zqDN#tPZJ_0X5Q<{-XZwONb&nEUjFu(!I$ID&Z?iezHMfN-{#ehK{%Y2WkCBh-w6$G z;)02VPF<(LT_Y4R%XR3=#tOiP`^HRdvAR*vUg8b60A|5AQ16W_C3xN*{;bMxdmv;J zH1uTx=vQ8C&nQ5&rtrbVas-v>tB}W@qL`Q%5Y~bS z0K^q_w4gh#vxQaZoVk)Jn# zYq$1-SHis3Gbs?~<2CXl;Yvj$j&U>)Z6UM)f&|V%E zmI|c3Bg6lL0D%5`WjH{nr&Mm%@fBOg*_rKoKak#N&~awAkeX#haq%#Fx80u%QXnu@x{ffhj@c^FDrbNk{g%r>;Kjt9*_tXVpiQ2U(~Sw<=ZPJ zZK?WQaX~e8vV6|8|4jsPl5g^nc(af1UVq%;-Bld^B5Qdt$?|7<#6-nK zBH^<6^)T?=W2F~w-+iTBfJGt%viWMgd!w1MnL{1gO*&f4df$+KUqBxMDp0hD=&~wB zznaU(RGrv%K(_9pcLTA)6Zai~ow~##RXrEVAF9&Akf+h!MGHk(AM^DyoYz(AU4rpXcjT=A^>4jHAGANi{22Tou5=m;mMDrIgQJuV%i(NJ*;)qusAkIl`WA z7bsv^Uh^#nb|Ov`w`TyZlM_7+r@CJ82M$h+9+*otPqRihhmzjl(R9|w(un&P&9em|vplN@2C_C& z6=<(Y^xxhiX0{qBkyuww7jX2Yc)NZ|FYJe3Rv3} zDgC&&kOoXczUlJXs=`lqfkj(bs`T8P{iv4_1M*kt=t_`d+t z1!o(y%B7(!=_-Iy1G%@q7hr$*E`KB;BQCIalbd_^A7ILT$v-oiqt_x!u)VwlMu}re|M~aPz^y?8Sd#p^;rWC28yR^I=CLpT z_sJq$RF45sNaepCHayoHLqkI%MAY`57v?pX5fF#3V?86uD)J@ecL@{8Q^_kPv{^+RXU_TP6N+C-7Wxtlty! zznc7h4X%KG?Tzr{8BkcfA#WYef`x|-xL(0wO;^7-$DIHVO&X`&tzi8Ckp-zObX!~i zZh1pdr0hlT#pzLZd5R5*nFlje@B+6*v3$rIc>W5=F1P|-J;!=blamUAE`}qxz#I@n zPMG5im+!Tm$EXJ&1%l)aDZ3%I3$+auK$0$Vt(?3G4SirkdkHQwWN!{=!;5DQ0;+MZ z*rp}Bz(Q>|P34u~+DrnR=ZNj8fC)&sbtPbq>ea4L=MVM)WM`WQw?hF+Q#ic1=JUb) zQp1>glh=_&om(?oj9X@Z^zPdu31JEZjxz-w2mgjrbDJaNXt>oFGPlv}QY7Y4CT#YD zVtZ&?Qg}-OP7b5Gk!>>bwHWH`bl2MUG)T$6PJ>d(_?101quO9-D{M1N$@e4PF@ks; zvkDC@1 z_%EpCsF1OXO-lY?^z_0~Lp1C&{}|Hbc40KcqO_^a#UfM}cNGOy@Uw^>q}tNR5I%OWq3S#uf-^Vbj^>JSZ&=1mMEO|y^3N2a&?Lop zY8VkG+ziaSSQeqcSWo7cW@DON@@jsRH&tq3>31Z}_kV`P-W6{2At7Vq7wxP?-$!bB_nz-U6oB(s?rr&? z-||B(>D4jzs&&IJ(_88(X0Qj-F4NV(*AUnkeS57RW$;(Gs|FB7|e0+L?G8I55)bi&Wlt*=6Wo$6EK@{-6Q+JEH)sr^0^23qfvpi z$^uxoMXI@PfIYPKaJTF=Y@z8E7<2jjYEA*xvP!#-zzXOyOn;DD>>NBL86Yjz&-0)# z)E@`5)`O3xPU}1bba2ZxIG6w}J5Iqot-03Mo5-H)Zu$$%vX;;c5ie&x*n`~TG}{MJ zNP|6KuC`3yo%`WY^Z5|W<@tW+11`MANYc@YF(A6==K(+X;Os(a5IBKyS94pI7&VC< zrClUo4IzHn6qwf)R}kx$@GQuluL(~a)gyU}!p`SsN3)FbC7U*H1L4azzdZKDNfXaU zpetrJi$_C?-P#uECaKeBsU9p54`9dFAX?y#QOOWfg0yj`7a|Q(3>sVxxO|S%PM;a? z_QDXk-6T5-@!~SRS#4N>pJ(yZ10bnl$oA(_lr^VyvNa$3t)bE-;Rz=t=#rGkbDY@K zwLWf7k*f#L^;EhNki)lI6Fy+?%79TWaEkN_wly$l=EB;SW<|U~0I!QftNG1x73Hus z5GOg+!`G;b`A~UKn4|#ZkmTYiZ?{8z_oLfyu&V?{jMc$$+QGGkFMPHOm3$0wG=M;V zn8t7+)|KH+%e(*Bmb>%wclxf_?+h;7ZqEmVdA{>~k$W-dnUnVoeN=wsXvO~rA4s21 zD~1R~#{?p|k((uwgTlbwu-kEqG)t)ewNSMNdkwnV>(<6h9p>lIy^RTHr6}9E&69Ha4!W6=$R%@7hqU+dxJ=4)*mInRR?T1%EK}wki z7g;UZkEmeX+~NT@snf&T{;mf7`u2ldn&stkA}R~@w;=Dx_>>%&$tO+!}P`z@tZ4M3W>kfBn*XUvGL~p)(ANo;9l)^!XXdGqq+G}E=W6ACAS_?s8u@987uOhmvj5wcpU(EJpr^l zaLYczufez;9cb82PO^Ml56GmEa^oI$&XsElg9ujtjL05k%LQ1MbKUvuhBFaP!{exXSL?|Bx)%q9I){_bxF6= zCk=gY_WY{uF?hZPC|eTzPiC&Rc>>o1(k%Nj>{jZCVB^*taWj>%uZ|1-20d%PO76Uf zDIpl@6|7wKv2`t4Zdh0BV(u(iSTo8+@=Qv1a!6z7T<_wgDioTpM)xC!tV}v=apBFJ zMSDmy%;=f@JS*v+KJ$k=LuBMUX1Q5+grR-wCC6ecH?E8`3Z&!0?ITh4-?5UU;Okzc;5vmQ~@-^po4 z>^&DZu6<}%I9=oiBhTXfH!&tuIuGCb5|>yF=;NDOjzWX&v_=2>-~;+wD8$O@Au~c0)_H|qdraYU9z*8@&Xr^ z^$5eLnojYKg5#nUD)9Bc&SI}`(e1k*${T*T*X8X5)$f`53_z{{Nqci@3e>U8Oi?qjdcG4tPD6 LRhB7{G7kPfdsbKE literal 0 HcmV?d00001 diff --git a/follow-recommendations-service/README.md b/follow-recommendations-service/README.md new file mode 100644 index 0000000000..1640184a53 --- /dev/null +++ b/follow-recommendations-service/README.md @@ -0,0 +1,40 @@ +# Follow Recommendations Service + +## Introduction to the Follow Recommendations Service (FRS) +The Follow Recommendations Service (FRS) is a robust recommendation engine designed to provide users with personalized suggestions for accounts to follow. At present, FRS supports Who-To-Follow (WTF) module recommendations across a variety of Twitter product interfaces. Additionally, by suggesting tweet authors, FRS also delivers FutureGraph tweet recommendations, which consist of tweets from accounts that users may be interested in following in the future. + +## Design +The system is tailored to accommodate diverse use cases, such as Post New-User-Experience (NUX), advertisements, FutureGraph tweets, and more. Each use case features a unique display location identifier. To view all display locations, refer to the following path: `follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/DisplayLocation.scala`. + +Recommendation steps are customized according to each display location. Common and high-level steps are encapsulated within the "RecommendationFlow," which includes operations like candidate generation, ranker selection, filtering, transformation, and beyond. To explore all flows, refer to this path: `follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows`. + +For each product (corresponding to a display location), one or multiple flows can be selected to generate candidates based on code and configurations. To view all products, refer to the following path: `follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/home_timeline_tweet_recs`. + +The FRS overview diagram is depicted below: + +![FRS_architecture.png](FRS_architecture.png) + + +### Candidate Generation +During this step, FRS utilizes various user signals and algorithms to identify candidates from all Twitter accounts. The candidate source folder is located at `follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/`, with a README file provided within each candidate source folder. + +### Filtering +In this phase, FRS applies different filtering logic after generating account candidates to improve quality and health. Filtering may occur before and/or after the ranking step, with heavier filtering logic (e.g., higher latency) typically applied after the ranking step. The filters' folder is located at `follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates`. + +### Ranking +During this step, FRS employs both Machine Learning (ML) and heuristic rule-based candidate ranking. For the ML ranker, ML features are fetched beforehand (i.e., feature hydration), +and a DataRecord (the Twitter-standard Machine Learning data format used to represent feature data, labels, and predictions when training or serving) is constructed for each pair. +These pairs are then sent to a separate ML prediction service, which houses the ML model trained offline. +The ML prediction service returns a prediction score, representing the probability that a user will follow and engage with the candidate. +This score is a weighted sum of p(follow|recommendation) and p(positive engagement|follow), and FRS uses this score to rank the candidates. + +The rankers' folder is located at `follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers`. + +### Transform +In this phase, the sequence of candidates undergoes necessary transformations, such as deduplication, attaching social proof (i.e., "followed by XX user"), adding tracking tokens, and more. +The transformers' folder can be found at `follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/transforms`. + +### Truncation +During this final step, FRS trims the candidate pool to a specified size. This process ensures that only the most relevant and engaging candidates are presented to users while maintaining an optimal user experience. + +By implementing these comprehensive steps and adapting to various use cases, the Follow Recommendations Service (FRS) effectively curates tailored suggestions for Twitter users, enhancing their overall experience and promoting meaningful connections within the platform. diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base/BUILD b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base/BUILD new file mode 100644 index 0000000000..d02506a857 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base/BUILD @@ -0,0 +1,18 @@ +scala_library( + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/com/google/guava", + "configapi/configapi-core/src/main/scala/com/twitter/timelines/configapi", + "finagle/finagle-core/src/main", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models", + "hermit/hermit-core/src/main/scala/com/twitter/hermit/model", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/candidate_source", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/recommendation", + "stitch/stitch-core", + ], + exports = [ + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/recommendation", + ], +) diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base/CandidateSourceRegistry.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base/CandidateSourceRegistry.scala new file mode 100644 index 0000000000..ee9cfbbe5b --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base/CandidateSourceRegistry.scala @@ -0,0 +1,36 @@ +package com.twitter.follow_recommendations.common.base + +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.follow_recommendations.common.base.EnrichedCandidateSource.toEnriched +import com.twitter.product_mixer.core.functional_component.candidate_source.CandidateSource +import com.twitter.product_mixer.core.model.common.identifier.CandidateSourceIdentifier + +// a helper structure to register and select candidate sources based on identifiers +trait CandidateSourceRegistry[Target, Candidate] { + + val statsReceiver: StatsReceiver + + def sources: Set[CandidateSource[Target, Candidate]] + + final lazy val candidateSources: Map[ + CandidateSourceIdentifier, + CandidateSource[Target, Candidate] + ] = { + val map = sources.map { c => + c.identifier -> c.observe(statsReceiver) + }.toMap + + if (map.size != sources.size) { + throw new IllegalArgumentException("Duplicate Candidate Source Identifiers") + } + + map + } + + def select( + identifiers: Set[CandidateSourceIdentifier] + ): Set[CandidateSource[Target, Candidate]] = { + // fails loud if the candidate source is not registered + identifiers.map(candidateSources(_)) + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base/EnrichedCandidateSource.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base/EnrichedCandidateSource.scala new file mode 100644 index 0000000000..9d8507528a --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base/EnrichedCandidateSource.scala @@ -0,0 +1,164 @@ +package com.twitter.follow_recommendations.common.base + +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.product_mixer.core.functional_component.candidate_source.CandidateSource +import com.twitter.stitch.Stitch +import com.twitter.util.Duration +import com.twitter.util.TimeoutException +import scala.language.implicitConversions + +class EnrichedCandidateSource[Target, Candidate](original: CandidateSource[Target, Candidate]) { + + /** + * Gate the candidate source based on the Predicate of target. + * It returns results only if the predicate returns Valid. + * + * @param predicate + * @return + */ + def gate(predicate: Predicate[Target]): CandidateSource[Target, Candidate] = { + throw new UnsupportedOperationException() + } + + def observe(statsReceiver: StatsReceiver): CandidateSource[Target, Candidate] = { + val originalIdentifier = original.identifier + val stats = statsReceiver.scope(originalIdentifier.name) + new CandidateSource[Target, Candidate] { + val identifier = originalIdentifier + override def apply(target: Target): Stitch[Seq[Candidate]] = { + StatsUtil.profileStitchSeqResults[Candidate](original(target), stats) + } + } + } + + /** + * Map target type into new target type (1 to optional mapping) + */ + def stitchMapKey[Target2]( + targetMapper: Target2 => Stitch[Option[Target]] + ): CandidateSource[Target2, Candidate] = { + val targetsMapper: Target2 => Stitch[Seq[Target]] = { target => + targetMapper(target).map(_.toSeq) + } + stitchMapKeys(targetsMapper) + } + + /** + * Map target type into new target type (1 to many mapping) + */ + def stitchMapKeys[Target2]( + targetMapper: Target2 => Stitch[Seq[Target]] + ): CandidateSource[Target2, Candidate] = { + new CandidateSource[Target2, Candidate] { + val identifier = original.identifier + override def apply(target: Target2): Stitch[Seq[Candidate]] = { + for { + mappedTargets <- targetMapper(target) + results <- Stitch.traverse(mappedTargets)(original(_)) + } yield results.flatten + } + } + } + + /** + * Map target type into new target type (1 to many mapping) + */ + def mapKeys[Target2]( + targetMapper: Target2 => Seq[Target] + ): CandidateSource[Target2, Candidate] = { + val stitchMapper: Target2 => Stitch[Seq[Target]] = { target => + Stitch.value(targetMapper(target)) + } + stitchMapKeys(stitchMapper) + } + + /** + * Map candidate types to new type based on candidateMapper + */ + def mapValues[Candidate2]( + candidateMapper: Candidate => Stitch[Option[Candidate2]] + ): CandidateSource[Target, Candidate2] = { + + new CandidateSource[Target, Candidate2] { + val identifier = original.identifier + override def apply(target: Target): Stitch[Seq[Candidate2]] = { + original(target).flatMap { candidates => + val results = Stitch.traverse(candidates)(candidateMapper(_)) + results.map(_.flatten) + } + } + } + } + + /** + * Map candidate types to new type based on candidateMapper + */ + def mapValue[Candidate2]( + candidateMapper: Candidate => Candidate2 + ): CandidateSource[Target, Candidate2] = { + val stitchMapper: Candidate => Stitch[Option[Candidate2]] = { c => + Stitch.value(Some(candidateMapper(c))) + } + mapValues(stitchMapper) + } + + /** + * This method wraps the candidate source in a designated timeout so that a single candidate + * source does not result in a timeout for the entire flow + */ + def within( + candidateTimeout: Duration, + statsReceiver: StatsReceiver + ): CandidateSource[Target, Candidate] = { + val originalIdentifier = original.identifier + val timeoutCounter = + statsReceiver.counter(originalIdentifier.name, "timeout") + + new CandidateSource[Target, Candidate] { + val identifier = originalIdentifier + override def apply(target: Target): Stitch[Seq[Candidate]] = { + original + .apply(target) + .within(candidateTimeout)(com.twitter.finagle.util.DefaultTimer) + .rescue { + case _: TimeoutException => + timeoutCounter.incr() + Stitch.Nil + } + } + } + } + + def failOpenWithin( + candidateTimeout: Duration, + statsReceiver: StatsReceiver + ): CandidateSource[Target, Candidate] = { + val originalIdentifier = original.identifier + val timeoutCounter = + statsReceiver.counter(originalIdentifier.name, "timeout") + + new CandidateSource[Target, Candidate] { + val identifier = originalIdentifier + override def apply(target: Target): Stitch[Seq[Candidate]] = { + original + .apply(target) + .within(candidateTimeout)(com.twitter.finagle.util.DefaultTimer) + .handle { + case _: TimeoutException => + timeoutCounter.incr() + Seq.empty + case e: Exception => + statsReceiver + .scope("candidate_source_error").scope(originalIdentifier.name).counter( + e.getClass.getSimpleName).incr + Seq.empty + } + } + } + } +} + +object EnrichedCandidateSource { + implicit def toEnriched[K, V](original: CandidateSource[K, V]): EnrichedCandidateSource[K, V] = + new EnrichedCandidateSource(original) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base/ParamPredicate.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base/ParamPredicate.scala new file mode 100644 index 0000000000..f457527bc2 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base/ParamPredicate.scala @@ -0,0 +1,17 @@ +package com.twitter.follow_recommendations.common.base + +import com.twitter.follow_recommendations.common.models.FilterReason.ParamReason +import com.twitter.stitch.Stitch +import com.twitter.timelines.configapi.HasParams +import com.twitter.timelines.configapi.Param + +case class ParamPredicate[Request <: HasParams](param: Param[Boolean]) extends Predicate[Request] { + + def apply(request: Request): Stitch[PredicateResult] = { + if (request.params(param)) { + Stitch.value(PredicateResult.Valid) + } else { + Stitch.value(PredicateResult.Invalid(Set(ParamReason(param.statName)))) + } + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base/Predicate.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base/Predicate.scala new file mode 100644 index 0000000000..e5b40ed829 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base/Predicate.scala @@ -0,0 +1,282 @@ +package com.twitter.follow_recommendations.common.base + +import com.twitter.finagle.stats.NullStatsReceiver +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.follow_recommendations.common.models.FilterReason +import com.twitter.stitch.Arrow +import com.twitter.stitch.Stitch + +trait Predicate[-Q] { + + def apply(item: Q): Stitch[PredicateResult] + def arrow: Arrow[Q, PredicateResult] = Arrow.apply(apply) + + def map[K](mapper: K => Q): Predicate[K] = Predicate(arrow.contramap(mapper)) + + /** + * check the predicate results for a batch of items for convenience. + * + * mark it as final to avoid potential abuse usage + */ + final def batch(items: Seq[Q]): Stitch[Seq[PredicateResult]] = { + this.arrow.traverse(items) + } + + /** + * Syntax sugar for functions which take in 2 inputs as a tuple. + */ + def apply[Q1, Q2](item1: Q1, item2: Q2)(implicit ev: ((Q1, Q2)) => Q): Stitch[PredicateResult] = { + apply((item1, item2)) + } + + /** + * Runs the predicates in sequence. The returned predicate will return true iff both the predicates return true. + * ie. it is an AND operation + * + * We short-circuit the evaluation, ie we don't evaluate the 2nd predicate if the 1st is false + * + * @param p predicate to run in sequence + * + * @return a new predicate object that represents the logical AND of both predicates + */ + def andThen[Q1 <: Q](p: Predicate[Q1]): Predicate[Q1] = { + Predicate({ query: Q1 => + apply(query).flatMap { + case PredicateResult.Valid => p(query) + case PredicateResult.Invalid(reasons) => Stitch.value(PredicateResult.Invalid(reasons)) + } + }) + } + + /** + * Creates a predicate which runs the current & given predicate in sequence. + * The returned predicate will return true if either current or given predicate returns true. + * That is, given predicate will be only run if current predicate returns false. + * + * @param p predicate to run in sequence + * + * @return new predicate object that represents the logical OR of both predicates. + * if both are invalid, the reason would be the set of all invalid reasons. + */ + def or[Q1 <: Q](p: Predicate[Q1]): Predicate[Q1] = { + Predicate({ query: Q1 => + apply(query).flatMap { + case PredicateResult.Valid => Stitch.value(PredicateResult.Valid) + case PredicateResult.Invalid(reasons) => + p(query).flatMap { + case PredicateResult.Valid => Stitch.value(PredicateResult.Valid) + case PredicateResult.Invalid(newReasons) => + Stitch.value(PredicateResult.Invalid(reasons ++ newReasons)) + } + } + }) + } + + /* + * Runs the predicate only if the provided predicate is valid, otherwise returns valid. + * */ + def gate[Q1 <: Q](gatingPredicate: Predicate[Q1]): Predicate[Q1] = { + Predicate { query: Q1 => + gatingPredicate(query).flatMap { result => + if (result == PredicateResult.Valid) { + apply(query) + } else { + Stitch.value(PredicateResult.Valid) + } + } + } + } + + def observe(statsReceiver: StatsReceiver): Predicate[Q] = Predicate( + StatsUtil.profilePredicateResult(this.arrow, statsReceiver)) + + def convertToFailOpenWithResultType(resultType: PredicateResult): Predicate[Q] = { + Predicate { query: Q => + apply(query).handle { + case _: Exception => + resultType + } + + } + } + +} + +class TruePredicate[Q] extends Predicate[Q] { + override def apply(item: Q): Stitch[PredicateResult] = Predicate.AlwaysTrueStitch +} + +class FalsePredicate[Q](reason: FilterReason) extends Predicate[Q] { + val InvalidResult = Stitch.value(PredicateResult.Invalid(Set(reason))) + override def apply(item: Q): Stitch[PredicateResult] = InvalidResult +} + +object Predicate { + + val AlwaysTrueStitch = Stitch.value(PredicateResult.Valid) + + val NumBatchesStat = "num_batches_stats" + val NumBatchesCount = "num_batches" + + def apply[Q](func: Q => Stitch[PredicateResult]): Predicate[Q] = new Predicate[Q] { + override def apply(item: Q): Stitch[PredicateResult] = func(item) + + override val arrow: Arrow[Q, PredicateResult] = Arrow(func) + } + + def apply[Q](outerArrow: Arrow[Q, PredicateResult]): Predicate[Q] = new Predicate[Q] { + override def apply(item: Q): Stitch[PredicateResult] = arrow(item) + + override val arrow: Arrow[Q, PredicateResult] = outerArrow + } + + /** + * Given some items, this function + * 1. chunks them up in groups + * 2. lazily applies a predicate on each group + * 3. filters based on the predicate + * 4. takes first numToTake items. + * + * If numToTake is satisfied, then any later predicates are not called. + * + * @param items items of type Q + * @param predicate predicate that determines whether an item is acceptable + * @param batchSize batch size to call the predicate with + * @param numToTake max number of items to return + * @param stats stats receiver + * @tparam Q type of item + * + * @return a future of K items + */ + def batchFilterTake[Q]( + items: Seq[Q], + predicate: Predicate[Q], + batchSize: Int, + numToTake: Int, + stats: StatsReceiver + ): Stitch[Seq[Q]] = { + + def take( + input: Iterator[Stitch[Seq[Q]]], + prev: Seq[Q], + takeSize: Int, + numOfBatch: Int + ): Stitch[(Seq[Q], Int)] = { + if (input.hasNext) { + val currFut = input.next() + currFut.flatMap { curr => + val taken = curr.take(takeSize) + val combined = prev ++ taken + if (taken.size < takeSize) + take(input, combined, takeSize - taken.size, numOfBatch + 1) + else Stitch.value((combined, numOfBatch + 1)) + } + } else { + Stitch.value((prev, numOfBatch)) + } + } + + val batchedItems = items.view.grouped(batchSize) + val batchedFutures = batchedItems.map { batch => + Stitch.traverse(batch)(predicate.apply).map { conds => + (batch.zip(conds)).withFilter(_._2.value).map(_._1) + } + } + take(batchedFutures, Nil, numToTake, 0).map { + case (filtered: Seq[Q], numOfBatch: Int) => + stats.stat(NumBatchesStat).add(numOfBatch) + stats.counter(NumBatchesCount).incr(numOfBatch) + filtered + } + } + + /** + * filter a list of items based on the predicate + * + * @param items a list of items + * @param predicate predicate of the item + * @tparam Q item type + * @return the list of items that satisfy the predicate + */ + def filter[Q](items: Seq[Q], predicate: Predicate[Q]): Stitch[Seq[Q]] = { + predicate.batch(items).map { results => + items.zip(results).collect { + case (item, PredicateResult.Valid) => item + } + } + } + + /** + * filter a list of items based on the predicate given the target + * + * @param target target item + * @param items a list of items + * @param predicate predicate of the (target, item) pair + * @tparam Q item type + * @return the list of items that satisfy the predicate given the target + */ + def filter[T, Q](target: T, items: Seq[Q], predicate: Predicate[(T, Q)]): Stitch[Seq[Q]] = { + predicate.batch(items.map(i => (target, i))).map { results => + items.zip(results).collect { + case (item, PredicateResult.Valid) => item + } + } + } + + /** + * Returns a predicate, where an element is true iff it that element is true for all input predicates. + * ie. it is an AND operation + * + * This is done concurrently. + * + * @param predicates list of predicates + * @tparam Q Type parameter + * + * @return new predicate object that is the logical "and" of the input predicates + */ + def andConcurrently[Q](predicates: Seq[Predicate[Q]]): Predicate[Q] = { + Predicate { query: Q => + Stitch.traverse(predicates)(p => p(query)).map { predicateResults => + val allInvalid = predicateResults + .collect { + case PredicateResult.Invalid(reason) => + reason + } + if (allInvalid.isEmpty) { + PredicateResult.Valid + } else { + val allInvalidReasons = allInvalid.reduce(_ ++ _) + PredicateResult.Invalid(allInvalidReasons) + } + } + } + } +} + +/** + * applies the underlying predicate when the param is on. + */ +abstract class GatedPredicateBase[Q]( + underlyingPredicate: Predicate[Q], + stats: StatsReceiver = NullStatsReceiver) + extends Predicate[Q] { + def gate(item: Q): Boolean + + val underlyingPredicateTotal = stats.counter("underlying_total") + val underlyingPredicateValid = stats.counter("underlying_valid") + val underlyingPredicateInvalid = stats.counter("underlying_invalid") + val notGatedCounter = stats.counter("not_gated") + + val ValidStitch: Stitch[PredicateResult.Valid.type] = Stitch.value(PredicateResult.Valid) + + override def apply(item: Q): Stitch[PredicateResult] = { + if (gate(item)) { + underlyingPredicateTotal.incr() + underlyingPredicate(item) + } else { + notGatedCounter.incr() + ValidStitch + } + } + +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base/PredicateResult.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base/PredicateResult.scala new file mode 100644 index 0000000000..002e902753 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base/PredicateResult.scala @@ -0,0 +1,18 @@ +package com.twitter.follow_recommendations.common.base + +import com.twitter.follow_recommendations.common.models.FilterReason + +sealed trait PredicateResult { + def value: Boolean +} + +object PredicateResult { + + case object Valid extends PredicateResult { + override val value = true + } + + case class Invalid(reasons: Set[FilterReason] = Set.empty[FilterReason]) extends PredicateResult { + override val value = false + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base/Ranker.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base/Ranker.scala new file mode 100644 index 0000000000..27eb50457e --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base/Ranker.scala @@ -0,0 +1,90 @@ +package com.twitter.follow_recommendations.common.base + +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.stitch.Stitch +import com.twitter.util.Duration +import com.twitter.util.TimeoutException + +/** + * Ranker is a special kind of transform that would only change the order of a list of items. + * If a single item is given, it "may" attach additional scoring information to the item. + * + * @tparam Target target to recommend the candidates + * @tparam Candidate candidate type to rank + */ +trait Ranker[Target, Candidate] extends Transform[Target, Candidate] { ranker => + + def rank(target: Target, candidates: Seq[Candidate]): Stitch[Seq[Candidate]] + + override def transform(target: Target, candidates: Seq[Candidate]): Stitch[Seq[Candidate]] = { + rank(target, candidates) + } + + override def observe(statsReceiver: StatsReceiver): Ranker[Target, Candidate] = { + val originalRanker = this + new Ranker[Target, Candidate] { + override def rank(target: Target, items: Seq[Candidate]): Stitch[Seq[Candidate]] = { + statsReceiver.counter(Transform.InputCandidatesCount).incr(items.size) + statsReceiver.stat(Transform.InputCandidatesStat).add(items.size) + StatsUtil.profileStitchSeqResults(originalRanker.rank(target, items), statsReceiver) + } + } + } + + def reverse: Ranker[Target, Candidate] = new Ranker[Target, Candidate] { + def rank(target: Target, candidates: Seq[Candidate]): Stitch[Seq[Candidate]] = + ranker.rank(target, candidates).map(_.reverse) + } + + def andThen(other: Ranker[Target, Candidate]): Ranker[Target, Candidate] = { + val original = this + new Ranker[Target, Candidate] { + def rank(target: Target, candidates: Seq[Candidate]): Stitch[Seq[Candidate]] = { + original.rank(target, candidates).flatMap { results => other.rank(target, results) } + } + } + } + + /** + * This method wraps the Ranker in a designated timeout. + * If the ranker timeouts, it would return the original candidates directly, + * instead of failing the whole recommendation flow + */ + def within(timeout: Duration, statsReceiver: StatsReceiver): Ranker[Target, Candidate] = { + val timeoutCounter = statsReceiver.counter("timeout") + val original = this + new Ranker[Target, Candidate] { + override def rank(target: Target, candidates: Seq[Candidate]): Stitch[Seq[Candidate]] = { + original + .rank(target, candidates) + .within(timeout)(com.twitter.finagle.util.DefaultTimer) + .rescue { + case _: TimeoutException => + timeoutCounter.incr() + Stitch.value(candidates) + } + } + } + } +} + +object Ranker { + + def chain[Target, Candidate]( + transformer: Transform[Target, Candidate], + ranker: Ranker[Target, Candidate] + ): Ranker[Target, Candidate] = { + new Ranker[Target, Candidate] { + def rank(target: Target, candidates: Seq[Candidate]): Stitch[Seq[Candidate]] = { + transformer + .transform(target, candidates) + .flatMap { results => ranker.rank(target, results) } + } + } + } +} + +class IdentityRanker[Target, Candidate] extends Ranker[Target, Candidate] { + def rank(target: Target, candidates: Seq[Candidate]): Stitch[Seq[Candidate]] = + Stitch.value(candidates) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base/RecommendationFlow.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base/RecommendationFlow.scala new file mode 100644 index 0000000000..6bddc9751d --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base/RecommendationFlow.scala @@ -0,0 +1,250 @@ +package com.twitter.follow_recommendations.common.base + +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.product_mixer.core.functional_component.candidate_source.CandidateSource +import com.twitter.product_mixer.core.model.common.UniversalNoun +import com.twitter.product_mixer.core.model.common.identifier.RecommendationPipelineIdentifier +import com.twitter.product_mixer.core.pipeline.recommendation.RecommendationPipelineResult +import com.twitter.product_mixer.core.quality_factor.QualityFactorObserver +import com.twitter.stitch.Stitch + +/** + * configs for results generated from the recommendation flow + * + * @param desiredCandidateCount num of desired candidates to return + * @param batchForCandidatesCheck batch size for candidates check + */ +case class RecommendationResultsConfig(desiredCandidateCount: Int, batchForCandidatesCheck: Int) + +trait BaseRecommendationFlow[Target, Candidate <: UniversalNoun[Long]] { + val identifier = RecommendationPipelineIdentifier("RecommendationFlow") + + def process( + pipelineRequest: Target + ): Stitch[RecommendationPipelineResult[Candidate, Seq[Candidate]]] + + def mapKey[Target2](fn: Target2 => Target): BaseRecommendationFlow[Target2, Candidate] = { + val original = this + new BaseRecommendationFlow[Target2, Candidate] { + override def process( + pipelineRequest: Target2 + ): Stitch[RecommendationPipelineResult[Candidate, Seq[Candidate]]] = + original.process(fn(pipelineRequest)) + } + } +} + +/** + * Defines a typical recommendation flow to fetch, filter, rank and transform candidates. + * + * 1. targetEligibility: determine the eligibility of target request + * 2. candidateSources: fetch candidates from candidate sources based on target type + * 3. preRankerCandidateFilter: light filtering of candidates + * 4. ranker: ranking of candidates (could be composed of multiple stages, light ranking, heavy ranking and etc) + * 5. postRankerTransform: deduping, grouping, rule based promotion / demotions and etc + * 6. validateCandidates: heavy filters to determine the eligibility of the candidates. + * will only be applied to candidates that we expect to return. + * 7. transformResults: transform the individual candidates into desired format (e.g. hydrate social proof) + * + * Note that the actual implementations may not need to implement all the steps if not needed + * (could just leave to IdentityRanker if ranking is not needed). + * + * Theoretically, the actual implementation could override the above flow to add + * more steps (e.g. add a transform step before ranking). + * But it is recommended to add the additional steps into this base flow if the step proves + * to have significant justification, or merge it into an existing step if it is a minor change. + * + * @tparam Target type of target request + * @tparam Candidate type of candidate to return + */ +trait RecommendationFlow[Target, Candidate <: UniversalNoun[Long]] + extends BaseRecommendationFlow[Target, Candidate] + with SideEffectsUtil[Target, Candidate] { + + /** + * optionally update or enrich the request before executing the flows + */ + protected def updateTarget(target: Target): Stitch[Target] = Stitch.value(target) + + /** + * check if the target is eligible for the flow + */ + protected def targetEligibility: Predicate[Target] + + /** + * define the candidate sources that should be used for the given target + */ + protected def candidateSources(target: Target): Seq[CandidateSource[Target, Candidate]] + + /** + * filter invalid candidates before the ranking phase. + */ + protected def preRankerCandidateFilter: Predicate[(Target, Candidate)] + + /** + * rank the candidates + */ + protected def selectRanker(target: Target): Ranker[Target, Candidate] + + /** + * transform the candidates after ranking (e.g. dedupping, grouping and etc) + */ + protected def postRankerTransform: Transform[Target, Candidate] + + /** + * filter invalid candidates before returning the results. + * + * Some heavy filters e.g. SGS filter could be applied in this step + */ + protected def validateCandidates: Predicate[(Target, Candidate)] + + /** + * transform the candidates into results and return + */ + protected def transformResults: Transform[Target, Candidate] + + /** + * configuration for recommendation results + */ + protected def resultsConfig(target: Target): RecommendationResultsConfig + + /** + * track the quality factor the recommendation pipeline + */ + protected def qualityFactorObserver: Option[QualityFactorObserver] = None + + def statsReceiver: StatsReceiver + + /** + * high level monitoring for the whole flow + * (make sure to add monitoring for each individual component by yourself) + * + * additional candidates: count, stats, non_empty_count + * target eligibility: latency, success, failures, request, count, valid_count, invalid_count, invalid_reasons + * candidate generation: latency, success, failures, request, count, non_empty_count, results_stat + * pre ranker filter: latency, success, failures, request, count, non_empty_count, results_stat + * ranker: latency, success, failures, request, count, non_empty_count, results_stat + * post ranker: latency, success, failures, request, count, non_empty_count, results_stat + * filter and take: latency, success, failures, request, count, non_empty_count, results_stat, batch count + * transform results: latency, success, failures, request, count, non_empty_count, results_stat + */ + import RecommendationFlow._ + lazy val additionalCandidatesStats = statsReceiver.scope(AdditionalCandidatesStats) + lazy val targetEligibilityStats = statsReceiver.scope(TargetEligibilityStats) + lazy val candidateGenerationStats = statsReceiver.scope(CandidateGenerationStats) + lazy val preRankerFilterStats = statsReceiver.scope(PreRankerFilterStats) + lazy val rankerStats = statsReceiver.scope(RankerStats) + lazy val postRankerTransformStats = statsReceiver.scope(PostRankerTransformStats) + lazy val filterAndTakeStats = statsReceiver.scope(FilterAndTakeStats) + lazy val transformResultsStats = statsReceiver.scope(TransformResultsStats) + + lazy val overallStats = statsReceiver.scope(OverallStats) + + import StatsUtil._ + + override def process( + pipelineRequest: Target + ): Stitch[RecommendationPipelineResult[Candidate, Seq[Candidate]]] = { + + observeStitchQualityFactor( + profileStitchSeqResults( + updateTarget(pipelineRequest).flatMap { target => + profilePredicateResult(targetEligibility(target), targetEligibilityStats).flatMap { + case PredicateResult.Valid => processValidTarget(target, Seq.empty) + case PredicateResult.Invalid(_) => Stitch.Nil + } + }, + overallStats + ).map { candidates => + RecommendationPipelineResult.empty.withResult(candidates) + }, + qualityFactorObserver, + overallStats + ) + } + + protected def processValidTarget( + target: Target, + additionalCandidates: Seq[Candidate] + ): Stitch[Seq[Candidate]] = { + + /** + * A basic recommendation flow looks like this: + * + * 1. fetch candidates from candidate sources + * 2. blend candidates with existing candidates + * 3. filter the candidates (light filters) before ranking + * 4. ranking + * 5. filter and truncate the candidates using postRankerCandidateFilter + * 6. transform the candidates based on product requirement + */ + val candidateSourcesToFetch = candidateSources(target) + for { + candidates <- profileStitchSeqResults( + Stitch.traverse(candidateSourcesToFetch)(_(target)).map(_.flatten), + candidateGenerationStats + ) + mergedCandidates = + profileSeqResults(additionalCandidates, additionalCandidatesStats) ++ + candidates + filteredCandidates <- profileStitchSeqResults( + Predicate.filter(target, mergedCandidates, preRankerCandidateFilter), + preRankerFilterStats + ) + rankedCandidates <- profileStitchSeqResults( + selectRanker(target).rank(target, filteredCandidates), + rankerStats + ) + transformed <- profileStitchSeqResults( + postRankerTransform.transform(target, rankedCandidates), + postRankerTransformStats + ) + truncated <- profileStitchSeqResults( + take(target, transformed, resultsConfig(target)), + filterAndTakeStats + ) + results <- profileStitchSeqResults( + transformResults.transform(target, truncated), + transformResultsStats + ) + _ <- applySideEffects( + target, + candidateSourcesToFetch, + candidates, + mergedCandidates, + filteredCandidates, + rankedCandidates, + transformed, + truncated, + results) + } yield results + } + + private[this] def take( + target: Target, + candidates: Seq[Candidate], + config: RecommendationResultsConfig + ): Stitch[Seq[Candidate]] = { + Predicate + .batchFilterTake( + candidates.map(c => (target, c)), + validateCandidates, + config.batchForCandidatesCheck, + config.desiredCandidateCount, + statsReceiver + ).map(_.map(_._2)) + } +} + +object RecommendationFlow { + + val AdditionalCandidatesStats = "additional_candidates" + val TargetEligibilityStats = "target_eligibility" + val CandidateGenerationStats = "candidate_generation" + val PreRankerFilterStats = "pre_ranker_filter" + val RankerStats = "ranker" + val PostRankerTransformStats = "post_ranker_transform" + val FilterAndTakeStats = "filter_and_take" + val TransformResultsStats = "transform_results" + val OverallStats = "overall" +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base/SideEffectsUtil.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base/SideEffectsUtil.scala new file mode 100644 index 0000000000..2c922f580b --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base/SideEffectsUtil.scala @@ -0,0 +1,24 @@ +package com.twitter.follow_recommendations.common.base + +import com.twitter.product_mixer.core.functional_component.candidate_source.CandidateSource +import com.twitter.stitch.Stitch + +/** + * SideEffectsUtil applies side effects to the intermediate candidate results from a recommendation flow pipeline. + * + * @tparam Target target to recommend the candidates + * @tparam Candidate candidate type to rank + */ +trait SideEffectsUtil[Target, Candidate] { + def applySideEffects( + target: Target, + candidateSources: Seq[CandidateSource[Target, Candidate]], + candidatesFromCandidateSources: Seq[Candidate], + mergedCandidates: Seq[Candidate], + filteredCandidates: Seq[Candidate], + rankedCandidates: Seq[Candidate], + transformedCandidates: Seq[Candidate], + truncatedCandidates: Seq[Candidate], + results: Seq[Candidate] + ): Stitch[Unit] = Stitch.Unit +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base/StatsUtil.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base/StatsUtil.scala new file mode 100644 index 0000000000..eb99a909e5 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base/StatsUtil.scala @@ -0,0 +1,272 @@ +package com.twitter.follow_recommendations.common.base +import com.twitter.finagle.stats.Stat +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.product_mixer.core.quality_factor.QualityFactorObserver +import com.twitter.stitch.Arrow +import com.twitter.stitch.Stitch +import com.twitter.util.Stopwatch +import java.util.concurrent.TimeUnit +import scala.util.control.NonFatal + +object StatsUtil { + val LatencyName = "latency_ms" + val RequestName = "requests" + val SuccessName = "success" + val FailureName = "failures" + val ResultsName = "results" + val ResultsStat = "results_stat" + val EmptyResultsName = "empty" + val NonEmptyResultsName = "non_empty" + val ValidCount = "valid" + val InvalidCount = "invalid" + val InvalidHasReasons = "has_reasons" + val Reasons = "reasons" + val QualityFactorStat = "quality_factor_stat" + val QualityFactorCounts = "quality_factor_counts" + + /** + * Helper function for timing a stitch, returning the original stitch. + */ + def profileStitch[T](stitch: Stitch[T], stat: StatsReceiver): Stitch[T] = { + + Stitch + .time(stitch) + .map { + case (response, stitchRunDuration) => + stat.counter(RequestName).incr() + stat.stat(LatencyName).add(stitchRunDuration.inMilliseconds) + response + .onSuccess { _ => stat.counter(SuccessName).incr() } + .onFailure { e => + stat.counter(FailureName).incr() + stat.scope(FailureName).counter(getCleanClassName(e)).incr() + } + } + .lowerFromTry + } + + /** + * Helper function for timing an arrow, returning the original arrow. + */ + def profileArrow[T, U](arrow: Arrow[T, U], stat: StatsReceiver): Arrow[T, U] = { + + Arrow + .time(arrow) + .map { + case (response, stitchRunDuration) => + stat.counter(RequestName).incr() + stat.stat(LatencyName).add(stitchRunDuration.inMilliseconds) + response + .onSuccess { _ => stat.counter(SuccessName).incr() } + .onFailure { e => + stat.counter(FailureName).incr() + stat.scope(FailureName).counter(getCleanClassName(e)).incr() + } + } + .lowerFromTry + } + + /** + * Helper function to count and track the distribution of results + */ + def profileResults[T](results: T, stat: StatsReceiver, size: T => Int): T = { + val numResults = size(results) + stat.counter(ResultsName).incr(numResults) + if (numResults == 0) { + stat.counter(EmptyResultsName).incr() + results + } else { + stat.stat(ResultsStat).add(numResults) + stat.counter(NonEmptyResultsName).incr() + results + } + } + + /** + * Helper function to count and track the distribution of a list of results + */ + def profileSeqResults[T](results: Seq[T], stat: StatsReceiver): Seq[T] = { + profileResults[Seq[T]](results, stat, _.size) + } + + /** + * Helper function for timing a stitch and count the number of results, returning the original stitch. + */ + def profileStitchResults[T](stitch: Stitch[T], stat: StatsReceiver, size: T => Int): Stitch[T] = { + profileStitch(stitch, stat).onSuccess { results => profileResults(results, stat, size) } + } + + /** + * Helper function for timing an arrow and count the number of results, returning the original arrow. + */ + def profileArrowResults[T, U]( + arrow: Arrow[T, U], + stat: StatsReceiver, + size: U => Int + ): Arrow[T, U] = { + profileArrow(arrow, stat).onSuccess { results => profileResults(results, stat, size) } + } + + /** + * Helper function for timing a stitch and count a seq of results, returning the original stitch. + */ + def profileStitchSeqResults[T](stitch: Stitch[Seq[T]], stat: StatsReceiver): Stitch[Seq[T]] = { + profileStitchResults[Seq[T]](stitch, stat, _.size) + } + + /** + * Helper function for timing a stitch and count optional results, returning the original stitch. + */ + def profileStitchOptionalResults[T]( + stitch: Stitch[Option[T]], + stat: StatsReceiver + ): Stitch[Option[T]] = { + profileStitchResults[Option[T]](stitch, stat, _.size) + } + + /** + * Helper function for timing a stitch and count a map of results, returning the original stitch. + */ + def profileStitchMapResults[K, V]( + stitch: Stitch[Map[K, V]], + stat: StatsReceiver + ): Stitch[Map[K, V]] = { + profileStitchResults[Map[K, V]](stitch, stat, _.size) + } + + def getCleanClassName(obj: Object): String = + obj.getClass.getSimpleName.stripSuffix("$") + + /** + * Helper function for timing a stitch and count a list of PredicateResult + */ + def profilePredicateResults( + predicateResult: Stitch[Seq[PredicateResult]], + statsReceiver: StatsReceiver + ): Stitch[Seq[PredicateResult]] = { + profileStitch[Seq[PredicateResult]]( + predicateResult, + statsReceiver + ).onSuccess { + _.map { + case PredicateResult.Valid => + statsReceiver.counter(ValidCount).incr() + case PredicateResult.Invalid(reasons) => + statsReceiver.counter(InvalidCount).incr() + reasons.map { filterReason => + statsReceiver.counter(InvalidHasReasons).incr() + statsReceiver.scope(Reasons).counter(filterReason.reason).incr() + } + } + } + } + + /** + * Helper function for timing a stitch and count individual PredicateResult + */ + def profilePredicateResult( + predicateResult: Stitch[PredicateResult], + statsReceiver: StatsReceiver + ): Stitch[PredicateResult] = { + profilePredicateResults( + predicateResult.map(Seq(_)), + statsReceiver + ).map(_.head) + } + + /** + * Helper function for timing an arrow and count a list of PredicateResult + */ + def profilePredicateResults[Q]( + predicateResult: Arrow[Q, Seq[PredicateResult]], + statsReceiver: StatsReceiver + ): Arrow[Q, Seq[PredicateResult]] = { + profileArrow[Q, Seq[PredicateResult]]( + predicateResult, + statsReceiver + ).onSuccess { + _.map { + case PredicateResult.Valid => + statsReceiver.counter(ValidCount).incr() + case PredicateResult.Invalid(reasons) => + statsReceiver.counter(InvalidCount).incr() + reasons.map { filterReason => + statsReceiver.counter(InvalidHasReasons).incr() + statsReceiver.scope(Reasons).counter(filterReason.reason).incr() + } + } + } + } + + /** + * Helper function for timing an arrow and count individual PredicateResult + */ + def profilePredicateResult[Q]( + predicateResult: Arrow[Q, PredicateResult], + statsReceiver: StatsReceiver + ): Arrow[Q, PredicateResult] = { + profilePredicateResults( + predicateResult.map(Seq(_)), + statsReceiver + ).map(_.head) + } + + /** + * Helper function for timing a stitch code block + */ + def profileStitchSeqResults[T]( + stats: StatsReceiver + )( + block: => Stitch[Seq[T]] + ): Stitch[Seq[T]] = { + stats.counter(RequestName).incr() + profileStitch(stats.stat(LatencyName), TimeUnit.MILLISECONDS) { + block onSuccess { r => + if (r.isEmpty) stats.counter(EmptyResultsName).incr() + stats.stat(ResultsStat).add(r.size) + } onFailure { e => + { + stats.counter(FailureName).incr() + stats.scope(FailureName).counter(e.getClass.getName).incr() + } + } + } + } + + /** + * Time a given asynchronous `f` using the given `unit`. + */ + def profileStitch[A](stat: Stat, unit: TimeUnit)(f: => Stitch[A]): Stitch[A] = { + val start = Stopwatch.timeNanos() + try { + f.respond { _ => stat.add(unit.convert(Stopwatch.timeNanos() - start, TimeUnit.NANOSECONDS)) } + } catch { + case NonFatal(e) => + stat.add(unit.convert(Stopwatch.timeNanos() - start, TimeUnit.NANOSECONDS)) + Stitch.exception(e) + } + } + + def observeStitchQualityFactor[T]( + stitch: Stitch[T], + qualityFactorObserverOption: Option[QualityFactorObserver], + statsReceiver: StatsReceiver + ): Stitch[T] = { + qualityFactorObserverOption + .map { observer => + Stitch + .time(stitch) + .map { + case (response, stitchRunDuration) => + observer(response, stitchRunDuration) + val qfVal = observer.qualityFactor.currentValue.floatValue() * 10000 + statsReceiver.counter(QualityFactorCounts).incr() + statsReceiver + .stat(QualityFactorStat) + .add(qfVal) + response + } + .lowerFromTry + }.getOrElse(stitch) + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base/Transform.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base/Transform.scala new file mode 100644 index 0000000000..c870db2f6e --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base/Transform.scala @@ -0,0 +1,85 @@ +package com.twitter.follow_recommendations.common.base + +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.stitch.Stitch +import com.twitter.timelines.configapi.HasParams +import com.twitter.timelines.configapi.Param + +/** + * transform a or a list of candidate for target T + * + * @tparam T target type + * @tparam C candidate type + */ +trait Transform[-T, C] { + + // you need to implement at least one of the two methods here. + def transformItem(target: T, item: C): Stitch[C] = { + transform(target, Seq(item)).map(_.head) + } + + def transform(target: T, items: Seq[C]): Stitch[Seq[C]] + + def mapTarget[T2](mapper: T2 => T): Transform[T2, C] = { + val original = this + new Transform[T2, C] { + override def transformItem(target: T2, item: C): Stitch[C] = { + original.transformItem(mapper(target), item) + } + override def transform(target: T2, items: Seq[C]): Stitch[Seq[C]] = { + original.transform(mapper(target), items) + } + } + } + + /** + * sequential composition. we execute this' transform first, followed by the other's transform + */ + def andThen[T1 <: T](other: Transform[T1, C]): Transform[T1, C] = { + val original = this + new Transform[T1, C] { + override def transformItem(target: T1, item: C): Stitch[C] = + original.transformItem(target, item).flatMap(other.transformItem(target, _)) + override def transform(target: T1, items: Seq[C]): Stitch[Seq[C]] = + original.transform(target, items).flatMap(other.transform(target, _)) + } + } + + def observe(statsReceiver: StatsReceiver): Transform[T, C] = { + val originalTransform = this + new Transform[T, C] { + override def transform(target: T, items: Seq[C]): Stitch[Seq[C]] = { + statsReceiver.counter(Transform.InputCandidatesCount).incr(items.size) + statsReceiver.stat(Transform.InputCandidatesStat).add(items.size) + StatsUtil.profileStitchSeqResults(originalTransform.transform(target, items), statsReceiver) + } + + override def transformItem(target: T, item: C): Stitch[C] = { + statsReceiver.counter(Transform.InputCandidatesCount).incr() + StatsUtil.profileStitch(originalTransform.transformItem(target, item), statsReceiver) + } + } + } +} + +trait GatedTransform[T <: HasParams, C] extends Transform[T, C] { + def gated(param: Param[Boolean]): Transform[T, C] = { + val original = this + (target: T, items: Seq[C]) => { + if (target.params(param)) { + original.transform(target, items) + } else { + Stitch.value(items) + } + } + } +} + +object Transform { + val InputCandidatesCount = "input_candidates" + val InputCandidatesStat = "input_candidates_stat" +} + +class IdentityTransform[T, C] extends Transform[T, C] { + override def transform(target: T, items: Seq[C]): Stitch[Seq[C]] = Stitch.value(items) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/addressbook/AddressBookParams.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/addressbook/AddressBookParams.scala new file mode 100644 index 0000000000..f93a60d1bf --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/addressbook/AddressBookParams.scala @@ -0,0 +1,9 @@ +package com.twitter.follow_recommendations.common.candidate_sources.addressbook + +import com.twitter.timelines.configapi.FSParam + +object AddressBookParams { + // Used by display locations that want only to read from the ABV2 Client and ignore Manhattan + // Currently the only display location that does this is the ABUploadInjection DisplayLocation + object ReadFromABV2Only extends FSParam[Boolean]("addressbook_read_only_from_abv2", false) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/addressbook/BUILD b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/addressbook/BUILD new file mode 100644 index 0000000000..ddbabad191 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/addressbook/BUILD @@ -0,0 +1,27 @@ +scala_library( + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/com/google/inject:guice", + "3rdparty/jvm/com/google/inject/extensions:guice-assistedinject", + "3rdparty/jvm/net/codingwell:scala-guice", + "3rdparty/jvm/org/slf4j:slf4j-api", + "finatra/inject/inject-core/src/main/scala", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/base", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/addressbook", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/email_storage_service", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/gizmoduck", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/phone_storage_service", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/strato", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/constants", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/deciders", + "src/thrift/com/twitter/hermit/candidate:hermit-candidate-scala", + "src/thrift/com/twitter/hermit/usercontacts:hermit-usercontacts-scala", + "strato/config/columns/onboarding/userrecs:userrecs-strato-client", + "strato/src/main/scala/com/twitter/strato/client", + "util/util-slf4j-api/src/main/scala", + ], +) diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/addressbook/ForwardEmailBookSource.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/addressbook/ForwardEmailBookSource.scala new file mode 100644 index 0000000000..d291459ce5 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/addressbook/ForwardEmailBookSource.scala @@ -0,0 +1,74 @@ +package com.twitter.follow_recommendations.common.candidate_sources.addressbook + +import com.twitter.finagle.stats.NullStatsReceiver +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.follow_recommendations.common.candidate_sources.addressbook.AddressBookParams.ReadFromABV2Only +import com.twitter.follow_recommendations.common.clients.addressbook.AddressbookClient +import com.twitter.follow_recommendations.common.clients.addressbook.models.EdgeType +import com.twitter.follow_recommendations.common.clients.addressbook.models.RecordIdentifier +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.follow_recommendations.common.utils.RescueWithStatsUtils.rescueWithStats +import com.twitter.hermit.model.Algorithm +import com.twitter.product_mixer.core.functional_component.candidate_source.CandidateSource +import com.twitter.product_mixer.core.model.common.identifier.CandidateSourceIdentifier +import com.twitter.product_mixer.core.model.marshalling.request.HasClientContext +import com.twitter.stitch.Stitch +import com.twitter.strato.generated.client.onboarding.userrecs.ForwardEmailBookClientColumn +import com.twitter.timelines.configapi.HasParams +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class ForwardEmailBookSource @Inject() ( + forwardEmailBookClientColumn: ForwardEmailBookClientColumn, + addressBookClient: AddressbookClient, + statsReceiver: StatsReceiver = NullStatsReceiver) + extends CandidateSource[HasParams with HasClientContext, CandidateUser] { + + override val identifier: CandidateSourceIdentifier = + ForwardEmailBookSource.Identifier + private val stats: StatsReceiver = statsReceiver.scope(this.getClass.getSimpleName) + + /** + * Generate a list of candidates for the target + */ + override def apply( + target: HasParams with HasClientContext + ): Stitch[Seq[CandidateUser]] = { + val candidateUsers: Stitch[Seq[Long]] = target.getOptionalUserId + .map { userId => + rescueWithStats( + addressBookClient.getUsers( + userId = userId, + identifiers = + Seq(RecordIdentifier(userId = Some(userId), email = None, phoneNumber = None)), + batchSize = AddressbookClient.AddressBook2BatchSize, + edgeType = ForwardEmailBookSource.DefaultEdgeType, + fetcherOption = + if (target.params.apply(ReadFromABV2Only)) None + else Some(forwardEmailBookClientColumn.fetcher), + queryOption = AddressbookClient + .createQueryOption( + edgeType = ForwardEmailBookSource.DefaultEdgeType, + isPhone = ForwardEmailBookSource.IsPhone) + ), + stats, + "AddressBookClient" + ) + }.getOrElse(Stitch.Nil) + + candidateUsers + .map( + _.take(ForwardEmailBookSource.NumEmailBookEntries) + .map(CandidateUser(_, score = Some(CandidateUser.DefaultCandidateScore)) + .withCandidateSource(identifier))) + } +} + +object ForwardEmailBookSource { + val Identifier: CandidateSourceIdentifier = CandidateSourceIdentifier( + Algorithm.ForwardEmailBook.toString) + val NumEmailBookEntries: Int = 1000 + val IsPhone = false + val DefaultEdgeType: EdgeType = EdgeType.Forward +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/addressbook/ForwardPhoneBookSource.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/addressbook/ForwardPhoneBookSource.scala new file mode 100644 index 0000000000..bb1f61f056 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/addressbook/ForwardPhoneBookSource.scala @@ -0,0 +1,72 @@ +package com.twitter.follow_recommendations.common.candidate_sources.addressbook + +import com.twitter.finagle.stats.NullStatsReceiver +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.follow_recommendations.common.candidate_sources.addressbook.AddressBookParams.ReadFromABV2Only +import com.twitter.follow_recommendations.common.clients.addressbook.AddressbookClient +import com.twitter.follow_recommendations.common.clients.addressbook.models.EdgeType +import com.twitter.follow_recommendations.common.clients.addressbook.models.RecordIdentifier +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.follow_recommendations.common.utils.RescueWithStatsUtils.rescueWithStats +import com.twitter.hermit.model.Algorithm +import com.twitter.product_mixer.core.functional_component.candidate_source.CandidateSource +import com.twitter.product_mixer.core.model.common.identifier.CandidateSourceIdentifier +import com.twitter.product_mixer.core.model.marshalling.request.HasClientContext +import com.twitter.stitch.Stitch +import com.twitter.strato.generated.client.onboarding.userrecs.ForwardPhoneContactsClientColumn +import com.twitter.timelines.configapi.HasParams +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class ForwardPhoneBookSource @Inject() ( + forwardPhoneContactsClientColumn: ForwardPhoneContactsClientColumn, + addressBookClient: AddressbookClient, + statsReceiver: StatsReceiver = NullStatsReceiver) + extends CandidateSource[HasParams with HasClientContext, CandidateUser] { + + override val identifier: CandidateSourceIdentifier = + ForwardPhoneBookSource.Identifier + private val stats: StatsReceiver = statsReceiver.scope(this.getClass.getSimpleName) + + /** + * Generate a list of candidates for the target + */ + override def apply(target: HasParams with HasClientContext): Stitch[Seq[CandidateUser]] = { + val candidateUsers: Stitch[Seq[Long]] = target.getOptionalUserId + .map { userId => + rescueWithStats( + addressBookClient.getUsers( + userId, + identifiers = + Seq(RecordIdentifier(userId = Some(userId), email = None, phoneNumber = None)), + batchSize = AddressbookClient.AddressBook2BatchSize, + edgeType = ForwardPhoneBookSource.DefaultEdgeType, + fetcherOption = + if (target.params.apply(ReadFromABV2Only)) None + else Some(forwardPhoneContactsClientColumn.fetcher), + queryOption = AddressbookClient + .createQueryOption( + edgeType = ForwardPhoneBookSource.DefaultEdgeType, + isPhone = ForwardPhoneBookSource.IsPhone) + ), + stats, + "AddressBookClient" + ) + }.getOrElse(Stitch.Nil) + + candidateUsers + .map( + _.take(ForwardPhoneBookSource.NumPhoneBookEntries) + .map(CandidateUser(_, score = Some(CandidateUser.DefaultCandidateScore)) + .withCandidateSource(identifier))) + } +} + +object ForwardPhoneBookSource { + val Identifier: CandidateSourceIdentifier = CandidateSourceIdentifier( + Algorithm.ForwardPhoneBook.toString) + val NumPhoneBookEntries: Int = 1000 + val IsPhone = true + val DefaultEdgeType: EdgeType = EdgeType.Forward +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/addressbook/README.md b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/addressbook/README.md new file mode 100644 index 0000000000..37b04a6387 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/addressbook/README.md @@ -0,0 +1,4 @@ +# Address Book Candidate Source +Provides the accounts of a given user's forward and reverse phone and email book contacts. +It is only available when the user has synced their address book with the service. + diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/addressbook/ReverseEmailBookSource.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/addressbook/ReverseEmailBookSource.scala new file mode 100644 index 0000000000..6e89f8c24d --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/addressbook/ReverseEmailBookSource.scala @@ -0,0 +1,78 @@ +package com.twitter.follow_recommendations.common.candidate_sources.addressbook + +import com.twitter.cds.contact_consent_state.thriftscala.PurposeOfProcessing +import com.twitter.finagle.stats.NullStatsReceiver +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.follow_recommendations.common.clients.addressbook.AddressbookClient +import com.twitter.follow_recommendations.common.clients.addressbook.models.EdgeType +import com.twitter.follow_recommendations.common.clients.addressbook.models.RecordIdentifier +import com.twitter.follow_recommendations.common.clients.email_storage_service.EmailStorageServiceClient +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.follow_recommendations.common.utils.RescueWithStatsUtils.rescueOptionalWithStats +import com.twitter.follow_recommendations.common.utils.RescueWithStatsUtils.rescueWithStats +import com.twitter.hermit.model.Algorithm +import com.twitter.product_mixer.core.functional_component.candidate_source.CandidateSource +import com.twitter.product_mixer.core.model.common.identifier.CandidateSourceIdentifier +import com.twitter.product_mixer.core.model.marshalling.request.HasClientContext +import com.twitter.stitch.Stitch +import com.twitter.strato.generated.client.onboarding.userrecs.ReverseEmailContactsClientColumn +import com.twitter.timelines.configapi.HasParams +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class ReverseEmailBookSource @Inject() ( + reverseEmailContactsClientColumn: ReverseEmailContactsClientColumn, + essClient: EmailStorageServiceClient, + addressBookClient: AddressbookClient, + statsReceiver: StatsReceiver = NullStatsReceiver) + extends CandidateSource[HasParams with HasClientContext, CandidateUser] { + override val identifier: CandidateSourceIdentifier = ReverseEmailBookSource.Identifier + private val rescueStats = statsReceiver.scope("ReverseEmailBookSource") + + /** + * Generate a list of candidates for the target + */ + override def apply(target: HasParams with HasClientContext): Stitch[Seq[CandidateUser]] = { + val reverseCandidatesFromEmail = target.getOptionalUserId + .map { userId => + val verifiedEmailStitchOpt = + rescueOptionalWithStats( + essClient.getVerifiedEmail(userId, PurposeOfProcessing.ContentRecommendations), + rescueStats, + "getVerifiedEmail") + verifiedEmailStitchOpt.flatMap { emailOpt => + rescueWithStats( + addressBookClient.getUsers( + userId = userId, + identifiers = emailOpt + .map(email => + RecordIdentifier(userId = None, email = Some(email), phoneNumber = None)).toSeq, + batchSize = ReverseEmailBookSource.NumEmailBookEntries, + edgeType = ReverseEmailBookSource.DefaultEdgeType, + fetcherOption = + if (target.params(AddressBookParams.ReadFromABV2Only)) None + else Some(reverseEmailContactsClientColumn.fetcher) + ), + rescueStats, + "AddressBookClient" + ) + } + }.getOrElse(Stitch.Nil) + + reverseCandidatesFromEmail.map( + _.take(ReverseEmailBookSource.NumEmailBookEntries) + .map( + CandidateUser(_, score = Some(CandidateUser.DefaultCandidateScore)) + .withCandidateSource(identifier)) + ) + } +} + +object ReverseEmailBookSource { + val Identifier: CandidateSourceIdentifier = CandidateSourceIdentifier( + Algorithm.ReverseEmailBookIbis.toString) + val NumEmailBookEntries: Int = 500 + val IsPhone = false + val DefaultEdgeType: EdgeType = EdgeType.Reverse +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/addressbook/ReversePhoneBookSource.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/addressbook/ReversePhoneBookSource.scala new file mode 100644 index 0000000000..4dbe5a6179 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/addressbook/ReversePhoneBookSource.scala @@ -0,0 +1,77 @@ +package com.twitter.follow_recommendations.common.candidate_sources.addressbook + +import com.twitter.cds.contact_consent_state.thriftscala.PurposeOfProcessing +import com.twitter.finagle.stats.NullStatsReceiver +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.follow_recommendations.common.clients.addressbook.AddressbookClient +import com.twitter.follow_recommendations.common.clients.addressbook.models.EdgeType +import com.twitter.follow_recommendations.common.clients.addressbook.models.RecordIdentifier +import com.twitter.follow_recommendations.common.clients.phone_storage_service.PhoneStorageServiceClient +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.follow_recommendations.common.utils.RescueWithStatsUtils.rescueWithStats +import com.twitter.hermit.model.Algorithm +import com.twitter.product_mixer.core.functional_component.candidate_source.CandidateSource +import com.twitter.product_mixer.core.model.common.identifier.CandidateSourceIdentifier +import com.twitter.product_mixer.core.model.marshalling.request.HasClientContext +import com.twitter.stitch.Stitch +import com.twitter.strato.generated.client.onboarding.userrecs.ReversePhoneContactsClientColumn +import com.twitter.timelines.configapi.HasParams +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class ReversePhoneBookSource @Inject() ( + reversePhoneContactsClientColumn: ReversePhoneContactsClientColumn, + pssClient: PhoneStorageServiceClient, + addressBookClient: AddressbookClient, + statsReceiver: StatsReceiver = NullStatsReceiver) + extends CandidateSource[HasParams with HasClientContext, CandidateUser] { + + override val identifier: CandidateSourceIdentifier = ReversePhoneBookSource.Identifier + private val stats: StatsReceiver = statsReceiver.scope(this.getClass.getSimpleName) + + /** + * Generate a list of candidates for the target + */ + override def apply(target: HasParams with HasClientContext): Stitch[Seq[CandidateUser]] = { + val reverseCandidatesFromPhones: Stitch[Seq[Long]] = target.getOptionalUserId + .map { userId => + pssClient + .getPhoneNumbers(userId, PurposeOfProcessing.ContentRecommendations) + .flatMap { phoneNumbers => + rescueWithStats( + addressBookClient.getUsers( + userId = userId, + identifiers = phoneNumbers.map(phoneNumber => + RecordIdentifier(userId = None, email = None, phoneNumber = Some(phoneNumber))), + batchSize = ReversePhoneBookSource.NumPhoneBookEntries, + edgeType = ReversePhoneBookSource.DefaultEdgeType, + fetcherOption = + if (target.params(AddressBookParams.ReadFromABV2Only)) None + else Some(reversePhoneContactsClientColumn.fetcher), + queryOption = AddressbookClient.createQueryOption( + edgeType = ReversePhoneBookSource.DefaultEdgeType, + isPhone = ReversePhoneBookSource.IsPhone) + ), + stats, + "AddressBookClient" + ) + } + }.getOrElse(Stitch.Nil) + + reverseCandidatesFromPhones.map( + _.take(ReversePhoneBookSource.NumPhoneBookEntries) + .map( + CandidateUser(_, score = Some(CandidateUser.DefaultCandidateScore)) + .withCandidateSource(identifier)) + ) + } +} + +object ReversePhoneBookSource { + val Identifier: CandidateSourceIdentifier = CandidateSourceIdentifier( + Algorithm.ReversePhoneBook.toString) + val NumPhoneBookEntries: Int = 500 + val IsPhone = true + val DefaultEdgeType: EdgeType = EdgeType.Reverse +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/base/BUILD b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/base/BUILD new file mode 100644 index 0000000000..275137bf92 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/base/BUILD @@ -0,0 +1,23 @@ +scala_library( + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/com/google/inject:guice", + "3rdparty/jvm/com/google/inject/extensions:guice-assistedinject", + "3rdparty/jvm/net/codingwell:scala-guice", + "3rdparty/jvm/org/slf4j:slf4j-api", + "content-recommender/thrift/src/main/thrift:thrift-scala", + "escherbird/src/scala/com/twitter/escherbird/util/stitchcache", + "finatra/inject/inject-core/src/main/scala", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/strato", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/transforms/modify_social_proof", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/common", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/candidate_source", + "src/scala/com/twitter/onboarding/relevance/features/ymbii", + "strato/src/main/scala/com/twitter/strato/client", + "util/util-slf4j-api/src/main/scala", + ], +) diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/base/CachedCandidateSource.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/base/CachedCandidateSource.scala new file mode 100644 index 0000000000..c0196c3048 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/base/CachedCandidateSource.scala @@ -0,0 +1,26 @@ +package com.twitter.follow_recommendations.common.candidate_sources.base + +import com.twitter.escherbird.util.stitchcache.StitchCache +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.product_mixer.core.functional_component.candidate_source.CandidateSource +import com.twitter.product_mixer.core.model.common.identifier.CandidateSourceIdentifier +import com.twitter.stitch.Stitch +import com.twitter.util.Duration + +class CachedCandidateSource[K <: Object, V <: Object]( + candidateSource: CandidateSource[K, V], + maxCacheSize: Int, + cacheTTL: Duration, + statsReceiver: StatsReceiver, + override val identifier: CandidateSourceIdentifier) + extends CandidateSource[K, V] { + + private val cache = StitchCache[K, Seq[V]]( + maxCacheSize = maxCacheSize, + ttl = cacheTTL, + statsReceiver = statsReceiver.scope(identifier.name, "cache"), + underlyingCall = (k: K) => candidateSource(k) + ) + + override def apply(target: K): Stitch[Seq[V]] = cache.readThrough(target) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/base/ExperimentalCandidateSource.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/base/ExperimentalCandidateSource.scala new file mode 100644 index 0000000000..9e2b0e6e9d --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/base/ExperimentalCandidateSource.scala @@ -0,0 +1,66 @@ +package com.twitter.follow_recommendations.common.candidate_sources.base +import com.twitter.product_mixer.core.functional_component.candidate_source.CandidateSource +import com.twitter.stitch.Stitch +import com.twitter.timelines.configapi.HasParams +import com.twitter.timelines.configapi.Param +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.product_mixer.core.model.common.identifier.CandidateSourceIdentifier + +/** + * A wrapper of CandidateSource to make it easier to do experimentation + * on new candidate generation algorithms + * + * @param baseSource base candidate source + * @param darkreadAlgorithmParam controls whether or not to darkread candidates (fetch them even if they will not be included) + * @param keepCandidatesParam controls whether or not to keep candidates from the base source + * @param resultCountThresholdParam controls how many results the source must return to bucket the user and return results (greater-than-or-equal-to) + * @tparam T request type. it must extend HasParams + * @tparam V value type + */ +class ExperimentalCandidateSource[T <: HasParams, V]( + baseSource: CandidateSource[T, V], + darkreadAlgorithmParam: Param[Boolean], + keepCandidatesParam: Param[Boolean], + resultCountThresholdParam: Param[Int], + baseStatsReceiver: StatsReceiver) + extends CandidateSource[T, V] { + + override val identifier: CandidateSourceIdentifier = baseSource.identifier + private[base] val statsReceiver = + baseStatsReceiver.scope(s"Experimental/${identifier.name}") + private[base] val requestsCounter = statsReceiver.counter("requests") + private[base] val resultCountGreaterThanThresholdCounter = + statsReceiver.counter("with_results_at_or_above_count_threshold") + private[base] val keepResultsCounter = statsReceiver.counter("keep_results") + private[base] val discardResultsCounter = statsReceiver.counter("discard_results") + + override def apply(request: T): Stitch[Seq[V]] = { + if (request.params(darkreadAlgorithmParam)) { + requestsCounter.incr() + fetchFromCandidateSourceAndProcessResults(request) + } else { + Stitch.Nil + } + } + + private def fetchFromCandidateSourceAndProcessResults(request: T): Stitch[Seq[V]] = { + baseSource(request).map { results => + if (results.length >= request.params(resultCountThresholdParam)) { + processResults(results, request.params(keepCandidatesParam)) + } else { + Nil + } + } + } + + private def processResults(results: Seq[V], keepResults: Boolean): Seq[V] = { + resultCountGreaterThanThresholdCounter.incr() + if (keepResults) { + keepResultsCounter.incr() + results + } else { + discardResultsCounter.incr() + Nil + } + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/base/RealGraphExpansionRepository.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/base/RealGraphExpansionRepository.scala new file mode 100644 index 0000000000..8add11fa64 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/base/RealGraphExpansionRepository.scala @@ -0,0 +1,208 @@ +package com.twitter.follow_recommendations.common.candidate_sources.base + +import com.twitter.conversions.DurationOps._ +import com.twitter.finagle.stats.NullStatsReceiver +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.finagle.util.DefaultTimer +import com.twitter.follow_recommendations.common.candidate_sources.base.RealGraphExpansionRepository.DefaultScore +import com.twitter.follow_recommendations.common.candidate_sources.base.RealGraphExpansionRepository.MaxNumIntermediateNodesToKeep +import com.twitter.follow_recommendations.common.candidate_sources.base.RealGraphExpansionRepository.FirstDegreeCandidatesTimeout +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.follow_recommendations.common.models._ +import com.twitter.onboarding.relevance.features.ymbii.ExpansionCandidateScores +import com.twitter.onboarding.relevance.features.ymbii.RawYMBIICandidateFeatures +import com.twitter.onboarding.relevance.store.thriftscala.CandidatesFollowedV1 +import com.twitter.product_mixer.core.functional_component.candidate_source.CandidateSource +import com.twitter.product_mixer.core.model.common.identifier.CandidateSourceIdentifier +import com.twitter.stitch.Stitch +import com.twitter.strato.client.Fetcher +import com.twitter.util.Duration +import scala.collection.immutable +import scala.util.control.NonFatal + +private final case class InterestExpansionCandidate( + userID: Long, + score: Double, + features: RawYMBIICandidateFeatures) + +abstract class RealGraphExpansionRepository[Request]( + realgraphExpansionStore: Fetcher[ + Long, + Unit, + CandidatesFollowedV1 + ], + override val identifier: CandidateSourceIdentifier, + statsReceiver: StatsReceiver = NullStatsReceiver, + maxUnderlyingCandidatesToQuery: Int = 50, + maxCandidatesToReturn: Int = 40, + overrideUnderlyingTimeout: Option[Duration] = None, + appendSocialProof: Boolean = false) + extends CandidateSource[ + Request, + CandidateUser + ] { + + val underlyingCandidateSource: Seq[ + CandidateSource[ + Request, + CandidateUser + ] + ] + + private val stats = statsReceiver.scope(this.getClass.getSimpleName).scope(identifier.name) + private val underlyingCandidateSourceFailureStats = + stats.scope("underlying_candidate_source_failure") + + def apply( + request: Request, + ): Stitch[Seq[CandidateUser]] = { + + val candidatesFromUnderlyingSourcesStitch: Seq[Stitch[Seq[CandidateUser]]] = + underlyingCandidateSource.map { candidateSource => + candidateSource + .apply(request) + .within(overrideUnderlyingTimeout.getOrElse(FirstDegreeCandidatesTimeout))( + DefaultTimer + ) + .handle { + case NonFatal(e) => + underlyingCandidateSourceFailureStats + .counter(candidateSource.identifier.name, e.getClass.getSimpleName).incr() + Seq.empty + } + } + + for { + underlyingCandidatesFromEachAlgo <- Stitch.collect(candidatesFromUnderlyingSourcesStitch) + // The first algorithm in the list has the highest priority. Depending on if its not + // populated, fall back to other algorithms. Once a particular algorithm is chosen, only + // take the top few candidates from the underlying store for expansion. + underlyingCandidatesTuple = + underlyingCandidatesFromEachAlgo + .zip(underlyingCandidateSource) + .find(_._1.nonEmpty) + + underlyingAlgorithmUsed: Option[CandidateSourceIdentifier] = underlyingCandidatesTuple.map { + case (_, candidateSource) => candidateSource.identifier + } + + // Take maxUnderlyingCandidatesToQuery to query realgraphExpansionStore + underlyingCandidates = + underlyingCandidatesTuple + .map { + case (candidates, candidateSource) => + stats + .scope("underlyingAlgorithmUsedScope").counter( + candidateSource.identifier.name).incr() + candidates + } + .getOrElse(Seq.empty) + .sortBy(_.score.getOrElse(DefaultScore))(Ordering.Double.reverse) + .take(maxUnderlyingCandidatesToQuery) + + underlyingCandidateMap: Map[Long, Double] = underlyingCandidates.map { candidate => + (candidate.id, candidate.score.getOrElse(DefaultScore)) + }.toMap + + expansionCandidates <- + Stitch + .traverse(underlyingCandidateMap.keySet.toSeq) { candidateId => + Stitch.join( + Stitch.value(candidateId), + realgraphExpansionStore.fetch(candidateId).map(_.v)) + + }.map(_.toMap) + + rerankedCandidates: Seq[InterestExpansionCandidate] = + rerankCandidateExpansions(underlyingCandidateMap, expansionCandidates) + + rerankedCandidatesFiltered = rerankedCandidates.take(maxCandidatesToReturn) + + } yield { + rerankedCandidatesFiltered.map { candidate => + val socialProofReason = if (appendSocialProof) { + val socialProofIds = candidate.features.expansionCandidateScores + .map(_.intermediateCandidateId) + Some( + Reason(Some( + AccountProof(followProof = Some(FollowProof(socialProofIds, socialProofIds.size)))))) + } else { + None + } + CandidateUser( + id = candidate.userID, + score = Some(candidate.score), + reason = socialProofReason, + userCandidateSourceDetails = Some( + UserCandidateSourceDetails( + primaryCandidateSource = Some(identifier), + candidateSourceFeatures = Map(identifier -> Seq(candidate.features)) + )) + ).addAddressBookMetadataIfAvailable(underlyingAlgorithmUsed.toSeq) + } + } + } + + /** + * Expands underlying candidates, returning them in sorted order. + * + * @param underlyingCandidatesMap A map from underlying candidate id to score + * @param expansionCandidateMap A map from underlying candidate id to optional expansion candidates + * @return A sorted sequence of expansion candidates and associated scores + */ + private def rerankCandidateExpansions( + underlyingCandidatesMap: Map[Long, Double], + expansionCandidateMap: Map[Long, Option[CandidatesFollowedV1]] + ): Seq[InterestExpansionCandidate] = { + + // extract features + val candidates: Seq[(Long, ExpansionCandidateScores)] = for { + (underlyingCandidateId, underlyingCandidateScore) <- underlyingCandidatesMap.toSeq + expansionCandidates = + expansionCandidateMap + .get(underlyingCandidateId) + .flatten + .map(_.candidatesFollowed) + .getOrElse(Seq.empty) + expansionCandidate <- expansionCandidates + } yield expansionCandidate.candidateID -> ExpansionCandidateScores( + underlyingCandidateId, + Some(underlyingCandidateScore), + Some(expansionCandidate.score) + ) + + // merge intermediate nodes for the same candidate + val dedupedCandidates: Seq[(Long, Seq[ExpansionCandidateScores])] = + candidates.groupBy(_._1).mapValues(_.map(_._2).sortBy(_.intermediateCandidateId)).toSeq + + // score the candidate + val candidatesWithTotalScore: Seq[((Long, Seq[ExpansionCandidateScores]), Double)] = + dedupedCandidates.map { candidate: (Long, Seq[ExpansionCandidateScores]) => + ( + candidate, + candidate._2.map { ieScore: ExpansionCandidateScores => + ieScore.scoreFromUserToIntermediateCandidate.getOrElse(DefaultScore) * + ieScore.scoreFromIntermediateToExpansionCandidate.getOrElse(DefaultScore) + }.sum) + } + + // sort candidate by score + for { + ((candidate, edges), score) <- candidatesWithTotalScore.sortBy(_._2)(Ordering[Double].reverse) + } yield InterestExpansionCandidate( + candidate, + score, + RawYMBIICandidateFeatures( + edges.size, + edges.take(MaxNumIntermediateNodesToKeep).to[immutable.Seq]) + ) + } + +} + +object RealGraphExpansionRepository { + private val FirstDegreeCandidatesTimeout: Duration = 250.milliseconds + private val MaxNumIntermediateNodesToKeep = 20 + private val DefaultScore = 0.0d + +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/base/SimilarUserExpanderParams.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/base/SimilarUserExpanderParams.scala new file mode 100644 index 0000000000..a4a0c9784d --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/base/SimilarUserExpanderParams.scala @@ -0,0 +1,31 @@ +package com.twitter.follow_recommendations.common.candidate_sources.base + +import com.twitter.timelines.configapi.FSBoundedParam +import com.twitter.timelines.configapi.FSParam + +object SimilarUserExpanderParams { + + case object EnableNonDirectFollowExpansion + extends FSParam[Boolean]("similar_user_enable_non_direct_follow_expansion", true) + + case object EnableSimsExpandSeedAccountsSort + extends FSParam[Boolean]("similar_user_enable_sims_expander_seed_account_sort", false) + + case object DefaultExpansionInputCount + extends FSBoundedParam[Int]( + name = "similar_user_default_expansion_input_count", + default = Integer.MAX_VALUE, + min = 0, + max = Integer.MAX_VALUE) + + case object DefaultFinalCandidatesReturnedCount + extends FSBoundedParam[Int]( + name = "similar_user_default_final_candidates_returned_count", + default = Integer.MAX_VALUE, + min = 0, + max = Integer.MAX_VALUE) + + case object DefaultEnableImplicitEngagedExpansion + extends FSParam[Boolean]("similar_user_enable_implicit_engaged_expansion", true) + +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/base/SimilarUserExpanderRepository.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/base/SimilarUserExpanderRepository.scala new file mode 100644 index 0000000000..336902b342 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/base/SimilarUserExpanderRepository.scala @@ -0,0 +1,313 @@ +package com.twitter.follow_recommendations.common.candidate_sources.base + +import com.twitter.follow_recommendations.common.candidate_sources.base.SimilarUserExpanderParams.DefaultEnableImplicitEngagedExpansion +import com.twitter.follow_recommendations.common.candidate_sources.base.SimilarUserExpanderParams.DefaultExpansionInputCount +import com.twitter.follow_recommendations.common.candidate_sources.base.SimilarUserExpanderParams.DefaultFinalCandidatesReturnedCount +import com.twitter.follow_recommendations.common.candidate_sources.base.SimilarUserExpanderParams.EnableNonDirectFollowExpansion +import com.twitter.follow_recommendations.common.candidate_sources.base.SimilarUserExpanderParams.EnableSimsExpandSeedAccountsSort +import com.twitter.follow_recommendations.common.candidate_sources.base.SimilarUserExpanderRepository.DefaultCandidateBuilder +import com.twitter.follow_recommendations.common.candidate_sources.base.SimilarUserExpanderRepository.DefaultScore +import com.twitter.follow_recommendations.common.models.AccountProof +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.follow_recommendations.common.models.EngagementType +import com.twitter.follow_recommendations.common.models.FollowProof +import com.twitter.follow_recommendations.common.models.Reason +import com.twitter.follow_recommendations.common.models.SimilarToProof +import com.twitter.follow_recommendations.common.models.UserCandidateSourceDetails +import com.twitter.hermit.candidate.thriftscala.Candidates +import com.twitter.product_mixer.core.functional_component.candidate_source.CandidateSource +import com.twitter.product_mixer.core.model.common.identifier.CandidateSourceIdentifier +import com.twitter.stitch.Stitch +import com.twitter.strato.client.Fetcher +import com.twitter.timelines.configapi.FSBoundedParam +import com.twitter.timelines.configapi.FSParam +import com.twitter.timelines.configapi.HasParams +import com.twitter.timelines.configapi.Params + +case class SecondDegreeCandidate(userId: Long, score: Double, socialProof: Option[Seq[Long]]) + +abstract class SimilarUserExpanderRepository[-Request <: HasParams]( + override val identifier: CandidateSourceIdentifier, + similarToCandidatesFetcher: Fetcher[ + Long, + Unit, + Candidates + ], + expansionInputSizeParam: FSBoundedParam[Int] = DefaultExpansionInputCount, + candidatesReturnedSizeParam: FSBoundedParam[Int] = DefaultFinalCandidatesReturnedCount, + enableImplicitEngagedExpansion: FSParam[Boolean] = DefaultEnableImplicitEngagedExpansion, + thresholdToAvoidExpansion: Int = 30, + maxExpansionPerCandidate: Option[Int] = None, + includingOriginalCandidates: Boolean = false, + scorer: (Double, Double) => Double = SimilarUserExpanderRepository.DefaultScorer, + aggregator: (Seq[Double]) => Double = ScoreAggregator.Max, + candidateBuilder: (Long, CandidateSourceIdentifier, Double, CandidateUser) => CandidateUser = + DefaultCandidateBuilder) + extends TwoHopExpansionCandidateSource[ + Request, + CandidateUser, + SecondDegreeCandidate, + CandidateUser + ] { + + val originalCandidateSource: CandidateSource[Request, CandidateUser] + val backupOriginalCandidateSource: Option[CandidateSource[Request, CandidateUser]] = None + + override def firstDegreeNodes(request: Request): Stitch[Seq[CandidateUser]] = { + + val originalCandidatesStitch: Stitch[Seq[CandidateUser]] = + originalCandidateSource(request) + + val backupCandidatesStitch: Stitch[Seq[CandidateUser]] = + if (request.params(EnableNonDirectFollowExpansion)) { + backupOriginalCandidateSource.map(_.apply(request)).getOrElse(Stitch.Nil) + } else { + Stitch.Nil + } + + val firstDegreeCandidatesCombinedStitch: Stitch[Seq[CandidateUser]] = + Stitch + .join(originalCandidatesStitch, backupCandidatesStitch).map { + case (firstDegreeOrigCandidates, backupFirstDegreeCandidates) => + if (request.params(EnableSimsExpandSeedAccountsSort)) { + firstDegreeOrigCandidates ++ backupFirstDegreeCandidates sortBy { + -_.score.getOrElse(DefaultScore) + } + } else { + firstDegreeOrigCandidates ++ backupFirstDegreeCandidates + } + } + + val candidatesAfterImplicitEngagementsRemovalStitch: Stitch[Seq[CandidateUser]] = + getCandidatesAfterImplicitEngagementFiltering( + request.params, + firstDegreeCandidatesCombinedStitch) + + val firstDegreeCandidatesCombinedTrimmed = candidatesAfterImplicitEngagementsRemovalStitch.map { + candidates: Seq[CandidateUser] => + candidates.take(request.params(expansionInputSizeParam)) + } + + firstDegreeCandidatesCombinedTrimmed.map { firstDegreeResults: Seq[CandidateUser] => + if (firstDegreeResults.nonEmpty && firstDegreeResults.size < thresholdToAvoidExpansion) { + firstDegreeResults + .groupBy(_.id).mapValues( + _.maxBy(_.score) + ).values.toSeq + } else { + Nil + } + } + + } + + override def secondaryDegreeNodes( + request: Request, + firstDegreeCandidate: CandidateUser + ): Stitch[Seq[SecondDegreeCandidate]] = { + similarToCandidatesFetcher.fetch(firstDegreeCandidate.id).map(_.v).map { candidateListOption => + candidateListOption + .map { candidatesList => + candidatesList.candidates.map(candidate => + SecondDegreeCandidate(candidate.userId, candidate.score, candidate.socialProof)) + }.getOrElse(Nil) + } + + } + + override def aggregateAndScore( + req: Request, + firstDegreeToSecondDegreeNodesMap: Map[CandidateUser, Seq[SecondDegreeCandidate]] + ): Stitch[Seq[CandidateUser]] = { + + val similarExpanderResults = firstDegreeToSecondDegreeNodesMap.flatMap { + case (firstDegreeCandidate, seqOfSecondDegreeCandidates) => + val sourceScore = firstDegreeCandidate.score.getOrElse(DefaultScore) + val results: Seq[CandidateUser] = seqOfSecondDegreeCandidates.map { secondDegreeCandidate => + val score = scorer(sourceScore, secondDegreeCandidate.score) + candidateBuilder(secondDegreeCandidate.userId, identifier, score, firstDegreeCandidate) + } + maxExpansionPerCandidate match { + case None => results + case Some(limit) => results.sortBy(-_.score.getOrElse(DefaultScore)).take(limit) + } + }.toSeq + + val allCandidates = { + if (includingOriginalCandidates) + firstDegreeToSecondDegreeNodesMap.keySet.toSeq + else + Nil + } ++ similarExpanderResults + + val groupedCandidates: Seq[CandidateUser] = allCandidates + .groupBy(_.id) + .flatMap { + case (_, candidates) => + val finalScore = aggregator(candidates.map(_.score.getOrElse(DefaultScore))) + val candidateSourceDetailsCombined = aggregateCandidateSourceDetails(candidates) + val accountSocialProofcombined = aggregateAccountSocialProof(candidates) + + candidates.headOption.map( + _.copy( + score = Some(finalScore), + reason = accountSocialProofcombined, + userCandidateSourceDetails = candidateSourceDetailsCombined) + .withCandidateSource(identifier)) + } + .toSeq + + Stitch.value( + groupedCandidates + .sortBy { -_.score.getOrElse(DefaultScore) }.take(req.params(candidatesReturnedSizeParam)) + ) + } + + def aggregateCandidateSourceDetails( + candidates: Seq[CandidateUser] + ): Option[UserCandidateSourceDetails] = { + candidates + .map { candidate => + candidate.userCandidateSourceDetails.map(_.candidateSourceScores).getOrElse(Map.empty) + }.reduceLeftOption { (scoreMap1, scoreMap2) => + scoreMap1 ++ scoreMap2 + }.map { + UserCandidateSourceDetails(primaryCandidateSource = None, _) + } + + } + + def aggregateAccountSocialProof(candidates: Seq[CandidateUser]): Option[Reason] = { + candidates + .map { candidate => + ( + candidate.reason + .flatMap(_.accountProof.flatMap(_.similarToProof.map(_.similarTo))).getOrElse(Nil), + candidate.reason + .flatMap(_.accountProof.flatMap(_.followProof.map(_.followedBy))).getOrElse(Nil), + candidate.reason + .flatMap(_.accountProof.flatMap(_.followProof.map(_.numIds))).getOrElse(0) + ) + }.reduceLeftOption { (accountProofOne, accountProofTwo) => + ( + // merge similarToIds + accountProofOne._1 ++ accountProofTwo._1, + // merge followedByIds + accountProofOne._2 ++ accountProofTwo._2, + // add numIds + accountProofOne._3 + accountProofTwo._3) + }.map { proofs => + Reason(accountProof = Some( + AccountProof( + similarToProof = Some(SimilarToProof(proofs._1)), + followProof = if (proofs._2.nonEmpty) Some(FollowProof(proofs._2, proofs._3)) else None + ))) + } + } + + def getCandidatesAfterImplicitEngagementFiltering( + params: Params, + firstDegreeCandidatesStitch: Stitch[Seq[CandidateUser]] + ): Stitch[Seq[CandidateUser]] = { + + if (!params(enableImplicitEngagedExpansion)) { + + /** + * Remove candidates whose engagement types only contain implicit engagements + * (e.g. Profile View, Tweet Click) and only expand those candidates who contain explicit + * engagements. + */ + firstDegreeCandidatesStitch.map { candidates => + candidates.filter { cand => + cand.engagements.exists(engage => + engage == EngagementType.Like || engage == EngagementType.Retweet || engage == EngagementType.Mention) + } + } + } else { + firstDegreeCandidatesStitch + } + } + +} + +object SimilarUserExpanderRepository { + val DefaultScorer: (Double, Double) => Double = (sourceScore: Double, similarScore: Double) => + similarScore + val MultiplyScorer: (Double, Double) => Double = (sourceScore: Double, similarScore: Double) => + sourceScore * similarScore + val SourceScorer: (Double, Double) => Double = (sourceScore: Double, similarScore: Double) => + sourceScore + + val DefaultScore = 0.0d + + val DefaultCandidateBuilder: ( + Long, + CandidateSourceIdentifier, + Double, + CandidateUser + ) => CandidateUser = + ( + userId: Long, + _: CandidateSourceIdentifier, + score: Double, + candidate: CandidateUser + ) => { + val originalCandidateSourceDetails = + candidate.userCandidateSourceDetails.flatMap { candSourceDetails => + candSourceDetails.primaryCandidateSource.map { primaryCandidateSource => + UserCandidateSourceDetails( + primaryCandidateSource = None, + candidateSourceScores = Map(primaryCandidateSource -> candidate.score)) + } + } + CandidateUser( + id = userId, + score = Some(score), + userCandidateSourceDetails = originalCandidateSourceDetails, + reason = + Some(Reason(Some(AccountProof(similarToProof = Some(SimilarToProof(Seq(candidate.id))))))) + ) + } + + val FollowClusterCandidateBuilder: ( + Long, + CandidateSourceIdentifier, + Double, + CandidateUser + ) => CandidateUser = + (userId: Long, _: CandidateSourceIdentifier, score: Double, candidate: CandidateUser) => { + val originalCandidateSourceDetails = + candidate.userCandidateSourceDetails.flatMap { candSourceDetails => + candSourceDetails.primaryCandidateSource.map { primaryCandidateSource => + UserCandidateSourceDetails( + primaryCandidateSource = None, + candidateSourceScores = Map(primaryCandidateSource -> candidate.score)) + } + } + + val originalFollowCluster = candidate.reason + .flatMap(_.accountProof.flatMap(_.followProof.map(_.followedBy))) + + CandidateUser( + id = userId, + score = Some(score), + userCandidateSourceDetails = originalCandidateSourceDetails, + reason = Some( + Reason( + Some( + AccountProof( + similarToProof = Some(SimilarToProof(Seq(candidate.id))), + followProof = originalFollowCluster.map(follows => + FollowProof(follows, follows.size))))) + ) + ) + } +} + +object ScoreAggregator { + // aggregate the same candidates with same id by taking the one with largest score + val Max: Seq[Double] => Double = (candidateScores: Seq[Double]) => { candidateScores.max } + + // aggregate the same candidates with same id by taking the sum of the scores + val Sum: Seq[Double] => Double = (candidateScores: Seq[Double]) => { candidateScores.sum } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/base/SocialProofEnforcedCandidateSource.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/base/SocialProofEnforcedCandidateSource.scala new file mode 100644 index 0000000000..f650c9f6d3 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/base/SocialProofEnforcedCandidateSource.scala @@ -0,0 +1,86 @@ +package com.twitter.follow_recommendations.common.candidate_sources.base + +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.follow_recommendations.common.transforms.modify_social_proof.ModifySocialProof +import com.twitter.product_mixer.core.functional_component.candidate_source.CandidateSource +import com.twitter.product_mixer.core.model.common.identifier.CandidateSourceIdentifier +import com.twitter.product_mixer.core.model.marshalling.request.HasClientContext +import com.twitter.stitch.Stitch +import com.twitter.timelines.configapi.HasParams +import com.twitter.util.Duration + +abstract class SocialProofEnforcedCandidateSource( + candidateSource: CandidateSource[HasClientContext with HasParams, CandidateUser], + modifySocialProof: ModifySocialProof, + minNumSocialProofsRequired: Int, + override val identifier: CandidateSourceIdentifier, + baseStatsReceiver: StatsReceiver) + extends CandidateSource[HasClientContext with HasParams, CandidateUser] { + + val statsReceiver = baseStatsReceiver.scope(identifier.name) + + override def apply(target: HasClientContext with HasParams): Stitch[Seq[CandidateUser]] = { + val mustCallSgs: Boolean = target.params(SocialProofEnforcedCandidateSourceParams.MustCallSgs) + val callSgsCachedColumn: Boolean = + target.params(SocialProofEnforcedCandidateSourceParams.CallSgsCachedColumn) + val QueryIntersectionIdsNum: Int = + target.params(SocialProofEnforcedCandidateSourceParams.QueryIntersectionIdsNum) + val MaxNumCandidatesToAnnotate: Int = + target.params(SocialProofEnforcedCandidateSourceParams.MaxNumCandidatesToAnnotate) + val gfsIntersectionIdsNum: Int = + target.params(SocialProofEnforcedCandidateSourceParams.GfsIntersectionIdsNum) + val sgsIntersectionIdsNum: Int = + target.params(SocialProofEnforcedCandidateSourceParams.SgsIntersectionIdsNum) + val gfsLagDuration: Duration = + target.params(SocialProofEnforcedCandidateSourceParams.GfsLagDurationInDays) + + candidateSource(target) + .flatMap { candidates => + val candidatesWithoutEnoughSocialProof = candidates + .collect { + case candidate if !candidate.followedBy.exists(_.size >= minNumSocialProofsRequired) => + candidate + } + statsReceiver + .stat("candidates_with_no_social_proofs").add(candidatesWithoutEnoughSocialProof.size) + val candidatesToAnnotate = + candidatesWithoutEnoughSocialProof.take(MaxNumCandidatesToAnnotate) + statsReceiver.stat("candidates_to_annotate").add(candidatesToAnnotate.size) + + val annotatedCandidatesMapStitch = target.getOptionalUserId + .map { userId => + modifySocialProof + .hydrateSocialProof( + userId, + candidatesToAnnotate, + Some(QueryIntersectionIdsNum), + mustCallSgs, + callSgsCachedColumn, + gfsLagDuration = gfsLagDuration, + gfsIntersectionIds = gfsIntersectionIdsNum, + sgsIntersectionIds = sgsIntersectionIdsNum + ).map { annotatedCandidates => + annotatedCandidates + .map(annotatedCandidate => (annotatedCandidate.id, annotatedCandidate)).toMap + } + }.getOrElse(Stitch.value(Map.empty[Long, CandidateUser])) + + annotatedCandidatesMapStitch.map { annotatedCandidatesMap => + candidates + .flatMap { candidate => + if (candidate.followedBy.exists(_.size >= minNumSocialProofsRequired)) { + Some(candidate) + } else { + annotatedCandidatesMap.get(candidate.id).collect { + case annotatedCandidate + if annotatedCandidate.followedBy.exists( + _.size >= minNumSocialProofsRequired) => + annotatedCandidate + } + } + }.map(_.withCandidateSource(identifier)) + } + } + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/base/SocialProofEnforcedCandidateSourceFSConfig.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/base/SocialProofEnforcedCandidateSourceFSConfig.scala new file mode 100644 index 0000000000..a74164f28f --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/base/SocialProofEnforcedCandidateSourceFSConfig.scala @@ -0,0 +1,30 @@ +package com.twitter.follow_recommendations.common.candidate_sources.base + +import com.twitter.follow_recommendations.configapi.common.FeatureSwitchConfig +import com.twitter.timelines.configapi.FSBoundedParam +import com.twitter.timelines.configapi.FSName +import com.twitter.timelines.configapi.HasDurationConversion +import com.twitter.timelines.configapi.Param +import com.twitter.util.Duration +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class SocialProofEnforcedCandidateSourceFSConfig @Inject() () extends FeatureSwitchConfig { + override val booleanFSParams: Seq[Param[Boolean] with FSName] = + Seq( + SocialProofEnforcedCandidateSourceParams.MustCallSgs, + SocialProofEnforcedCandidateSourceParams.CallSgsCachedColumn, + ) + override val intFSParams: Seq[FSBoundedParam[Int]] = + Seq( + SocialProofEnforcedCandidateSourceParams.QueryIntersectionIdsNum, + SocialProofEnforcedCandidateSourceParams.MaxNumCandidatesToAnnotate, + SocialProofEnforcedCandidateSourceParams.GfsIntersectionIdsNum, + SocialProofEnforcedCandidateSourceParams.SgsIntersectionIdsNum, + ) + + override val durationFSParams: Seq[FSBoundedParam[Duration] with HasDurationConversion] = Seq( + SocialProofEnforcedCandidateSourceParams.GfsLagDurationInDays + ) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/base/SocialProofEnforcedCandidateSourceParams.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/base/SocialProofEnforcedCandidateSourceParams.scala new file mode 100644 index 0000000000..36e50e59f2 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/base/SocialProofEnforcedCandidateSourceParams.scala @@ -0,0 +1,56 @@ +package com.twitter.follow_recommendations.common.candidate_sources.base + +import com.twitter.conversions.DurationOps._ +import com.twitter.timelines.configapi.DurationConversion +import com.twitter.timelines.configapi.FSBoundedParam +import com.twitter.timelines.configapi.FSParam +import com.twitter.timelines.configapi.HasDurationConversion +import com.twitter.util.Duration + +object SocialProofEnforcedCandidateSourceParams { + case object MustCallSgs + extends FSParam[Boolean]("social_proof_enforced_candidate_source_must_call_sgs", true) + + case object CallSgsCachedColumn + extends FSParam[Boolean]( + "social_proof_enforced_candidate_source_call_sgs_cached_column", + false) + + case object QueryIntersectionIdsNum + extends FSBoundedParam[Int]( + name = "social_proof_enforced_candidate_source_query_intersection_ids_num", + default = 3, + min = 0, + max = Integer.MAX_VALUE) + + case object MaxNumCandidatesToAnnotate + extends FSBoundedParam[Int]( + name = "social_proof_enforced_candidate_source_max_num_candidates_to_annotate", + default = 50, + min = 0, + max = Integer.MAX_VALUE) + + case object GfsIntersectionIdsNum + extends FSBoundedParam[Int]( + name = "social_proof_enforced_candidate_source_gfs_intersection_ids_num", + default = 3, + min = 0, + max = Integer.MAX_VALUE) + + case object SgsIntersectionIdsNum + extends FSBoundedParam[Int]( + name = "social_proof_enforced_candidate_source_sgs_intersection_ids_num", + default = 10, + min = 0, + max = Integer.MAX_VALUE) + + case object GfsLagDurationInDays + extends FSBoundedParam[Duration]( + name = "social_proof_enforced_candidate_source_gfs_lag_duration_in_days", + default = 14.days, + min = 1.days, + max = 60.days) + with HasDurationConversion { + override val durationConversion: DurationConversion = DurationConversion.FromDays + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/base/StratoFetcherSource.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/base/StratoFetcherSource.scala new file mode 100644 index 0000000000..ea6fa57fa3 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/base/StratoFetcherSource.scala @@ -0,0 +1,27 @@ +package com.twitter.follow_recommendations.common.candidate_sources.base + +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.product_mixer.core.functional_component.candidate_source.CandidateSource +import com.twitter.product_mixer.core.model.common.identifier.CandidateSourceIdentifier +import com.twitter.stitch.Stitch +import com.twitter.strato.client.Fetcher + +abstract class StratoFetcherSource[K, U, V]( + fetcher: Fetcher[K, U, V], + view: U, + override val identifier: CandidateSourceIdentifier) + extends CandidateSource[K, CandidateUser] { + + def map(user: K, v: V): Seq[CandidateUser] + + override def apply(target: K): Stitch[Seq[CandidateUser]] = { + fetcher + .fetch(target, view) + .map { result => + result.v + .map { candidates => map(target, candidates) } + .getOrElse(Nil) + .map(_.withCandidateSource(identifier)) + } + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/base/StratoFetcherWithUnitViewSource.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/base/StratoFetcherWithUnitViewSource.scala new file mode 100644 index 0000000000..1f1572ee6c --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/base/StratoFetcherWithUnitViewSource.scala @@ -0,0 +1,9 @@ +package com.twitter.follow_recommendations.common.candidate_sources.base + +import com.twitter.product_mixer.core.model.common.identifier.CandidateSourceIdentifier +import com.twitter.strato.client.Fetcher + +abstract class StratoFetcherWithUnitViewSource[K, V]( + fetcher: Fetcher[K, Unit, V], + override val identifier: CandidateSourceIdentifier) + extends StratoFetcherSource[K, Unit, V](fetcher, Unit, identifier) diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/base/TweetAuthorsCandidateSource.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/base/TweetAuthorsCandidateSource.scala new file mode 100644 index 0000000000..541b248372 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/base/TweetAuthorsCandidateSource.scala @@ -0,0 +1,71 @@ +package com.twitter.follow_recommendations.common.candidate_sources.base + +import com.twitter.follow_recommendations.common.models.TweetCandidate +import com.twitter.product_mixer.core.functional_component.candidate_source.CandidateSource +import com.twitter.stitch.Stitch + +/** + * base trait for tweet authors based algorithms, e.g. topical tweet authors, twistly, ... + * + * @tparam Target target type + * @tparam Candidate output candidate types + */ +trait TweetAuthorsCandidateSource[-Target, +Candidate] extends CandidateSource[Target, Candidate] { + + /** + * fetch Tweet candidates + */ + def getTweetCandidates(target: Target): Stitch[Seq[TweetCandidate]] + + /** + * fetch authorId + */ + def getTweetAuthorId(tweetCandidate: TweetCandidate): Stitch[Option[Long]] + + /** + * wrap candidate ID and TweetAuthorProof in Candidate + */ + def toCandidate(authorId: Long, tweetIds: Seq[Long], score: Option[Double]): Candidate + + /** + * aggregate scores, default to the first score + */ + def aggregator(scores: Seq[Double]): Double = + scores.headOption.getOrElse(TweetAuthorsCandidateSource.DefaultScore) + + /** + * aggregation method for a group of tweet candidates + */ + def aggregateAndScore( + target: Target, + tweetCandidates: Seq[TweetCandidate] + ): Seq[Candidate] + + /** + * generate a list of candidates for the target + */ + def build( + target: Target + ): Stitch[Seq[Candidate]] = { + // Fetch Tweet candidates and hydrate author IDs + val tweetCandidatesStitch = for { + tweetCandidates <- getTweetCandidates(target) + authorIds <- Stitch.collect(tweetCandidates.map(getTweetAuthorId(_))) + } yield { + for { + (authorIdOpt, tweetCandidate) <- authorIds.zip(tweetCandidates) + authorId <- authorIdOpt + } yield tweetCandidate.copy(authorId = authorId) + } + + // Aggregate and score, convert to candidate + tweetCandidatesStitch.map(aggregateAndScore(target, _)) + } + + def apply(target: Target): Stitch[Seq[Candidate]] = + build(target) +} + +object TweetAuthorsCandidateSource { + final val DefaultScore: Double = 0.0 +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/base/TwoHopExpansionCandidateSource.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/base/TwoHopExpansionCandidateSource.scala new file mode 100644 index 0000000000..40c699d222 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/base/TwoHopExpansionCandidateSource.scala @@ -0,0 +1,46 @@ +package com.twitter.follow_recommendations.common.candidate_sources.base + +import com.twitter.product_mixer.core.functional_component.candidate_source.CandidateSource +import com.twitter.stitch.Stitch + +/** + * base trait for two-hop expansion based algorithms, e.g. online_stp, phonebook_prediction, + * recent following sims, recent engagement sims, ... + * + * @tparam Target target type + * @tparam FirstDegree type of first degree nodes + * @tparam SecondaryDegree type of secondary degree nodes + * @tparam Candidate output candidate types + */ +trait TwoHopExpansionCandidateSource[-Target, FirstDegree, SecondaryDegree, +Candidate] + extends CandidateSource[Target, Candidate] { + + /** + * fetch first degree nodes given request + */ + def firstDegreeNodes(req: Target): Stitch[Seq[FirstDegree]] + + /** + * fetch secondary degree nodes given request and first degree nodes + */ + def secondaryDegreeNodes(req: Target, node: FirstDegree): Stitch[Seq[SecondaryDegree]] + + /** + * aggregate and score the candidates to generate final results + */ + def aggregateAndScore( + req: Target, + firstDegreeToSecondDegreeNodesMap: Map[FirstDegree, Seq[SecondaryDegree]] + ): Stitch[Seq[Candidate]] + + /** + * Generate a list of candidates for the target + */ + def apply(target: Target): Stitch[Seq[Candidate]] = { + for { + firstDegreeNodes <- firstDegreeNodes(target) + secondaryDegreeNodes <- Stitch.traverse(firstDegreeNodes)(secondaryDegreeNodes(target, _)) + aggregated <- aggregateAndScore(target, firstDegreeNodes.zip(secondaryDegreeNodes).toMap) + } yield aggregated + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/crowd_search_accounts/BUILD b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/crowd_search_accounts/BUILD new file mode 100644 index 0000000000..de3bc0a6b7 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/crowd_search_accounts/BUILD @@ -0,0 +1,22 @@ +scala_library( + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + strict_deps = True, + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/com/google/inject:guice", + "escherbird/src/scala/com/twitter/escherbird/util/stitchcache", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/base", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/geoduck", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/constants", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/common", + "hermit/hermit-core/src/main/scala/com/twitter/hermit/model", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/candidate_source", + "src/thrift/com/twitter/onboarding/relevance/crowd_search_accounts:crowd_search_accounts-scala", + "strato/config/columns/onboarding/userrecs:userrecs-strato-client", + "strato/src/main/scala/com/twitter/strato/client", + "util/util-core/src/main/scala/com/twitter/conversions", + ], +) diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/crowd_search_accounts/CrowdSearchAccountsFSConfig.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/crowd_search_accounts/CrowdSearchAccountsFSConfig.scala new file mode 100644 index 0000000000..520983b60e --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/crowd_search_accounts/CrowdSearchAccountsFSConfig.scala @@ -0,0 +1,18 @@ +package com.twitter.follow_recommendations.common.candidate_sources.crowd_search_accounts + +import com.twitter.follow_recommendations.configapi.common.FeatureSwitchConfig +import com.twitter.timelines.configapi.FSBoundedParam +import com.twitter.timelines.configapi.FSName +import com.twitter.timelines.configapi.Param +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class CrowdSearchAccountsFSConfig @Inject() () extends FeatureSwitchConfig { + override val booleanFSParams: Seq[Param[Boolean] with FSName] = Seq( + CrowdSearchAccountsParams.CandidateSourceEnabled, + ) + override val doubleFSParams: Seq[FSBoundedParam[Double]] = Seq( + CrowdSearchAccountsParams.CandidateSourceWeight, + ) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/crowd_search_accounts/CrowdSearchAccountsParams.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/crowd_search_accounts/CrowdSearchAccountsParams.scala new file mode 100644 index 0000000000..a167b7768d --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/crowd_search_accounts/CrowdSearchAccountsParams.scala @@ -0,0 +1,32 @@ +package com.twitter.follow_recommendations.common.candidate_sources.crowd_search_accounts + +import com.twitter.timelines.configapi.FSBoundedParam +import com.twitter.timelines.configapi.FSEnumSeqParam +import com.twitter.timelines.configapi.FSParam + +object CrowdSearchAccountsParams { + // whether or not to fetch CrowdSearchAccounts candidate sources + case object CandidateSourceEnabled + extends FSParam[Boolean]("crowd_search_accounts_candidate_source_enabled", false) + + /** + * Contains the logic key for account filtering and ranking. Currently we have 3 main logic keys + * - new_daily: filtering top searched accounts with max daily searches based on new users + * - new_weekly: filtering top searched accounts with max weekly searches based on new users + * - daily: filtering top searched accounts with max daily searches + * - weekly: filtering top searched accounts with max weekly searches + * Mapping of the Logic Id to Logic key is done via @enum AccountsFilteringAndRankingLogic + */ + case object AccountsFilteringAndRankingLogics + extends FSEnumSeqParam[AccountsFilteringAndRankingLogicId.type]( + name = "crowd_search_accounts_filtering_and_ranking_logic_ids", + default = Seq(AccountsFilteringAndRankingLogicId.SearchesWeekly), + enum = AccountsFilteringAndRankingLogicId) + + case object CandidateSourceWeight + extends FSBoundedParam[Double]( + "crowd_search_accounts_candidate_source_weight", + default = 1200, + min = 0.001, + max = 2000) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/crowd_search_accounts/CrowdSearchAccountsSource.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/crowd_search_accounts/CrowdSearchAccountsSource.scala new file mode 100644 index 0000000000..6d3e903a1a --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/crowd_search_accounts/CrowdSearchAccountsSource.scala @@ -0,0 +1,111 @@ +package com.twitter.follow_recommendations.common.candidate_sources.crowd_search_accounts + +import com.twitter.escherbird.util.stitchcache.StitchCache +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.follow_recommendations.common.candidate_sources.crowd_search_accounts.CrowdSearchAccountsParams.AccountsFilteringAndRankingLogics +import com.twitter.follow_recommendations.common.candidate_sources.crowd_search_accounts.CrowdSearchAccountsParams.CandidateSourceEnabled +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.follow_recommendations.common.models.HasGeohashAndCountryCode +import com.twitter.hermit.model.Algorithm +import com.twitter.onboarding.relevance.crowd_search_accounts.thriftscala.CrowdSearchAccounts +import com.twitter.product_mixer.core.functional_component.candidate_source.CandidateSource +import com.twitter.product_mixer.core.model.common.identifier.CandidateSourceIdentifier +import com.twitter.product_mixer.core.model.marshalling.request.HasClientContext +import com.twitter.stitch.Stitch +import com.twitter.strato.generated.client.onboarding.userrecs.CrowdSearchAccountsClientColumn +import com.twitter.timelines.configapi.HasParams +import com.twitter.util.Duration +import com.twitter.util.logging.Logging +import javax.inject.Inject +import javax.inject.Singleton + +object AccountsFilteringAndRankingLogicId extends Enumeration { + type AccountsFilteringAndRankingLogicId = Value + + val NewSearchesDaily: AccountsFilteringAndRankingLogicId = Value("new_searches_daily") + val NewSearchesWeekly: AccountsFilteringAndRankingLogicId = Value("new_searches_weekly") + val SearchesDaily: AccountsFilteringAndRankingLogicId = Value("searches_daily") + val SearchesWeekly: AccountsFilteringAndRankingLogicId = Value("searches_weekly") +} + +object CrowdSearchAccountsSource { + val MaxCacheSize = 500 + val CacheTTL: Duration = Duration.fromHours(24) + + type Target = HasParams with HasClientContext with HasGeohashAndCountryCode + + val Identifier: CandidateSourceIdentifier = CandidateSourceIdentifier( + Algorithm.CrowdSearchAccounts.toString) +} + +@Singleton +class CrowdSearchAccountsSource @Inject() ( + crowdSearchAccountsClientColumn: CrowdSearchAccountsClientColumn, + statsReceiver: StatsReceiver, +) extends CandidateSource[CrowdSearchAccountsSource.Target, CandidateUser] + with Logging { + + /** @see [[CandidateSourceIdentifier]] */ + override val identifier: CandidateSourceIdentifier = + CrowdSearchAccountsSource.Identifier + + private val stats = statsReceiver.scope(identifier.name) + private val requestsStats = stats.counter("requests") + private val noCountryCodeStats = stats.counter("no_country_code") + private val successStats = stats.counter("success") + private val errorStats = stats.counter("error") + + private val cache = StitchCache[String, Option[CrowdSearchAccounts]]( + maxCacheSize = CrowdSearchAccountsSource.MaxCacheSize, + ttl = CrowdSearchAccountsSource.CacheTTL, + statsReceiver = statsReceiver.scope(identifier.name, "cache"), + underlyingCall = (k: String) => { + crowdSearchAccountsClientColumn.fetcher + .fetch(k) + .map { result => result.v } + } + ) + + /** returns a Seq of ''potential'' content */ + override def apply( + target: CrowdSearchAccountsSource.Target + ): Stitch[Seq[CandidateUser]] = { + if (!target.params(CandidateSourceEnabled)) { + return Stitch.value(Seq[CandidateUser]()) + } + requestsStats.incr() + target.getCountryCode + .orElse(target.geohashAndCountryCode.flatMap(_.countryCode)).map { countryCode => + Stitch + .collect(target + .params(AccountsFilteringAndRankingLogics).map(logic => + cache.readThrough(countryCode.toUpperCase() + "-" + logic))) + .onSuccess(_ => { + successStats.incr() + }) + .onFailure(t => { + debug("candidate source failed identifier = %s".format(identifier), t) + errorStats.incr() + }) + .map(transformCrowdSearchAccountsToCandidateSource) + }.getOrElse { + noCountryCodeStats.incr() + Stitch.value(Seq[CandidateUser]()) + } + } + + private def transformCrowdSearchAccountsToCandidateSource( + crowdSearchAccounts: Seq[Option[CrowdSearchAccounts]] + ): Seq[CandidateUser] = { + crowdSearchAccounts + .flatMap(opt => + opt + .map(accounts => + accounts.accounts.map(account => + CandidateUser( + id = account.accountId, + score = Some(account.searchActivityScore), + ).withCandidateSource(identifier))) + .getOrElse(Seq[CandidateUser]())) + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/crowd_search_accounts/README.md b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/crowd_search_accounts/README.md new file mode 100644 index 0000000000..043279b447 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/crowd_search_accounts/README.md @@ -0,0 +1,4 @@ +# Crowd Search Candidate Source +Provides the most searched accounts within a specific country over the past 1 and 7 days. +* When we refer to "most searched accounts", we are referring to accounts that have been clicked on the most frequently by users after they see search results in both the typeahead and search results page. +* The results returned by the service have undergone health filters. diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/geo/BUILD b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/geo/BUILD new file mode 100644 index 0000000000..2493a4a93b --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/geo/BUILD @@ -0,0 +1,23 @@ +scala_library( + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/com/google/inject:guice", + "3rdparty/jvm/com/google/inject/extensions:guice-assistedinject", + "3rdparty/jvm/net/codingwell:scala-guice", + "3rdparty/jvm/org/slf4j:slf4j-api", + "finatra/inject/inject-core/src/main/scala", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/base", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/geoduck", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/strato", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/constants", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/common", + "src/thrift/com/twitter/hermit/pop_geo:hermit-pop-geo-scala", + "strato/config/columns/onboarding/userrecs:userrecs-strato-client", + "strato/src/main/scala/com/twitter/strato/client", + "util/util-slf4j-api/src/main/scala", + ], +) diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/geo/BasePopGeoHashSource.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/geo/BasePopGeoHashSource.scala new file mode 100644 index 0000000000..862046bbc6 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/geo/BasePopGeoHashSource.scala @@ -0,0 +1,74 @@ +package com.twitter.follow_recommendations.common.candidate_sources.geo + +import com.google.inject.Singleton +import com.twitter.finagle.stats.Counter +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.follow_recommendations.common.models.HasGeohashAndCountryCode +import com.twitter.product_mixer.core.functional_component.candidate_source.CandidateSource +import com.twitter.product_mixer.core.model.common.identifier.CandidateSourceIdentifier +import com.twitter.product_mixer.core.model.marshalling.request.HasClientContext +import com.twitter.stitch.Stitch +import com.twitter.timelines.configapi.HasParams +import javax.inject.Inject + +@Singleton +class BasePopGeohashSource @Inject() ( + popGeoSource: CandidateSource[String, CandidateUser], + statsReceiver: StatsReceiver) + extends CandidateSource[ + HasParams with HasClientContext with HasGeohashAndCountryCode, + CandidateUser + ] + with BasePopGeohashSourceConfig { + + val stats: StatsReceiver = statsReceiver + + // counter to check if we found a geohash value in the request + val foundGeohashCounter: Counter = stats.counter("found_geohash_value") + // counter to check if we are missing a geohash value in the request + val missingGeohashCounter: Counter = stats.counter("missing_geohash_value") + + /** @see [[CandidateSourceIdentifier]] */ + override val identifier: CandidateSourceIdentifier = CandidateSourceIdentifier( + "BasePopGeohashSource") + + override def apply( + target: HasParams with HasClientContext with HasGeohashAndCountryCode + ): Stitch[Seq[CandidateUser]] = { + if (!candidateSourceEnabled(target)) { + return Stitch.Nil + } + target.geohashAndCountryCode + .flatMap(_.geohash).map { geohash => + foundGeohashCounter.incr() + val keys = (minGeohashLength(target) to math.min(maxGeohashLength(target), geohash.length)) + .map("geohash_" + geohash.take(_)).reverse + if (returnResultFromAllPrecision(target)) { + Stitch + .collect(keys.map(popGeoSource.apply)).map( + _.flatten.map(_.withCandidateSource(identifier)) + ) + } else { + Stitch + .collect(keys.map(popGeoSource.apply)).map( + _.find(_.nonEmpty) + .getOrElse(Nil) + .take(maxResults(target)).map(_.withCandidateSource(identifier)) + ) + } + }.getOrElse { + missingGeohashCounter.incr() + Stitch.Nil + } + } +} + +trait BasePopGeohashSourceConfig { + type Target = HasParams with HasClientContext + def maxResults(target: Target): Int = 200 + def minGeohashLength(target: Target): Int = 2 + def maxGeohashLength(target: Target): Int = 4 + def returnResultFromAllPrecision(target: Target): Boolean = false + def candidateSourceEnabled(target: Target): Boolean = false +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/geo/PopCountryBackFillSource.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/geo/PopCountryBackFillSource.scala new file mode 100644 index 0000000000..e2a5f9baf2 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/geo/PopCountryBackFillSource.scala @@ -0,0 +1,33 @@ +package com.twitter.follow_recommendations.common.candidate_sources.geo + +import com.google.inject.Singleton +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.hermit.model.Algorithm +import com.twitter.product_mixer.core.functional_component.candidate_source.CandidateSource +import com.twitter.product_mixer.core.model.common.identifier.CandidateSourceIdentifier +import com.twitter.product_mixer.core.model.marshalling.request.HasClientContext +import com.twitter.stitch.Stitch +import com.twitter.timelines.configapi.HasParams +import javax.inject.Inject + +@Singleton +class PopCountryBackFillSource @Inject() (popGeoSource: PopGeoSource) + extends CandidateSource[HasClientContext with HasParams, CandidateUser] { + + override val identifier: CandidateSourceIdentifier = PopCountryBackFillSource.Identifier + + override def apply(target: HasClientContext with HasParams): Stitch[Seq[CandidateUser]] = { + target.getOptionalUserId + .map(_ => + popGeoSource(PopCountryBackFillSource.DefaultKey) + .map(_.take(PopCountryBackFillSource.MaxResults).map(_.withCandidateSource(identifier)))) + .getOrElse(Stitch.Nil) + } +} + +object PopCountryBackFillSource { + val Identifier: CandidateSourceIdentifier = + CandidateSourceIdentifier(Algorithm.PopCountryBackFill.toString) + val MaxResults = 40 + val DefaultKey = "country_US" +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/geo/PopCountrySource.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/geo/PopCountrySource.scala new file mode 100644 index 0000000000..de6377df6a --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/geo/PopCountrySource.scala @@ -0,0 +1,63 @@ +package com.twitter.follow_recommendations.common.candidate_sources.geo + +import com.google.inject.Singleton +import com.twitter.core_workflows.user_model.thriftscala.UserState +import com.twitter.finagle.stats.Counter +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.follow_recommendations.common.models.HasGeohashAndCountryCode +import com.twitter.follow_recommendations.common.models.HasUserState +import com.twitter.hermit.model.Algorithm +import com.twitter.product_mixer.core.functional_component.candidate_source.CandidateSource +import com.twitter.product_mixer.core.model.common.identifier.CandidateSourceIdentifier +import com.twitter.product_mixer.core.model.marshalling.request.HasClientContext +import com.twitter.stitch.Stitch +import com.twitter.timelines.configapi.HasParams +import javax.inject.Inject + +@Singleton +class PopCountrySource @Inject() ( + popGeoSource: PopGeoSource, + statsReceiver: StatsReceiver) + extends CandidateSource[ + HasClientContext with HasParams with HasUserState with HasGeohashAndCountryCode, + CandidateUser + ] { + + override val identifier: CandidateSourceIdentifier = PopCountrySource.Identifier + val stats: StatsReceiver = statsReceiver.scope("PopCountrySource") + + // counter to check if we found a country code value in the request + val foundCountryCodeCounter: Counter = stats.counter("found_country_code_value") + // counter to check if we are missing a country code value in the request + val missingCountryCodeCounter: Counter = stats.counter("missing_country_code_value") + + override def apply( + target: HasClientContext with HasParams with HasUserState with HasGeohashAndCountryCode + ): Stitch[Seq[CandidateUser]] = { + target.geohashAndCountryCode + .flatMap(_.countryCode).map { countryCode => + foundCountryCodeCounter.incr() + if (target.userState.exists(PopCountrySource.BlacklistedTargetUserStates.contains)) { + Stitch.Nil + } else { + popGeoSource("country_" + countryCode) + .map(_.take(PopCountrySource.MaxResults).map(_.withCandidateSource(identifier))) + } + }.getOrElse { + missingCountryCodeCounter.incr() + Stitch.Nil + } + } +} + +object PopCountrySource { + val Identifier: CandidateSourceIdentifier = CandidateSourceIdentifier( + Algorithm.PopCountry.toString) + val MaxResults = 40 + val BlacklistedTargetUserStates: Set[UserState] = Set( + UserState.HeavyTweeter, + UserState.HeavyNonTweeter, + UserState.MediumTweeter, + UserState.MediumNonTweeter) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/geo/PopGeoQualityFollowSource.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/geo/PopGeoQualityFollowSource.scala new file mode 100644 index 0000000000..ad473a282d --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/geo/PopGeoQualityFollowSource.scala @@ -0,0 +1,99 @@ +package com.twitter.follow_recommendations.common.candidate_sources.geo +import com.google.inject.Singleton +import com.twitter.escherbird.util.stitchcache.StitchCache +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.follow_recommendations.common.models.AccountProof +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.follow_recommendations.common.models.PopularInGeoProof +import com.twitter.follow_recommendations.common.models.Reason +import com.twitter.hermit.model.Algorithm +import com.twitter.hermit.pop_geo.thriftscala.PopUsersInPlace +import com.twitter.product_mixer.core.functional_component.candidate_source.CandidateSource +import com.twitter.product_mixer.core.model.common.identifier.CandidateSourceIdentifier +import com.twitter.stitch.Stitch +import com.twitter.strato.generated.client.onboarding.userrecs.UniquePopQualityFollowUsersInPlaceClientColumn +import com.twitter.util.Duration +import javax.inject.Inject + +@Singleton +class PopGeohashQualityFollowSource @Inject() ( + popGeoSource: PopGeoQualityFollowSource, + statsReceiver: StatsReceiver) + extends BasePopGeohashSource( + popGeoSource = popGeoSource, + statsReceiver = statsReceiver.scope("PopGeohashQualityFollowSource"), + ) { + override val identifier: CandidateSourceIdentifier = PopGeohashQualityFollowSource.Identifier + override def maxResults(target: Target): Int = { + target.params(PopGeoQualityFollowSourceParams.PopGeoSourceMaxResultsPerPrecision) + } + override def minGeohashLength(target: Target): Int = { + target.params(PopGeoQualityFollowSourceParams.PopGeoSourceGeoHashMinPrecision) + } + override def maxGeohashLength(target: Target): Int = { + target.params(PopGeoQualityFollowSourceParams.PopGeoSourceGeoHashMaxPrecision) + } + override def returnResultFromAllPrecision(target: Target): Boolean = { + target.params(PopGeoQualityFollowSourceParams.PopGeoSourceReturnFromAllPrecisions) + } + override def candidateSourceEnabled(target: Target): Boolean = { + target.params(PopGeoQualityFollowSourceParams.CandidateSourceEnabled) + } +} + +object PopGeohashQualityFollowSource { + val Identifier: CandidateSourceIdentifier = CandidateSourceIdentifier( + Algorithm.PopGeohashQualityFollow.toString) +} + +object PopGeoQualityFollowSource { + val MaxCacheSize = 20000 + val CacheTTL: Duration = Duration.fromHours(24) + val MaxResults = 200 +} + +@Singleton +class PopGeoQualityFollowSource @Inject() ( + popGeoQualityFollowClientColumn: UniquePopQualityFollowUsersInPlaceClientColumn, + statsReceiver: StatsReceiver, +) extends CandidateSource[String, CandidateUser] { + + /** @see [[CandidateSourceIdentifier]] */ + override val identifier: CandidateSourceIdentifier = CandidateSourceIdentifier( + "PopGeoQualityFollowSource") + + private val cache = StitchCache[String, Option[PopUsersInPlace]]( + maxCacheSize = PopGeoQualityFollowSource.MaxCacheSize, + ttl = PopGeoQualityFollowSource.CacheTTL, + statsReceiver = statsReceiver.scope(identifier.name, "cache"), + underlyingCall = (k: String) => { + popGeoQualityFollowClientColumn.fetcher + .fetch(k) + .map { result => result.v } + } + ) + + override def apply(target: String): Stitch[Seq[CandidateUser]] = { + val result: Stitch[Option[PopUsersInPlace]] = cache.readThrough(target) + result.map { pu => + pu.map { candidates => + candidates.popUsers.sortBy(-_.score).take(PopGeoQualityFollowSource.MaxResults).map { + candidate => + CandidateUser( + id = candidate.userId, + score = Some(candidate.score), + reason = Some( + Reason( + Some( + AccountProof( + popularInGeoProof = Some(PopularInGeoProof(location = candidates.place)) + ) + ) + ) + ) + ) + } + }.getOrElse(Nil) + } + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/geo/PopGeoQualityFollowSourceFSConfig.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/geo/PopGeoQualityFollowSourceFSConfig.scala new file mode 100644 index 0000000000..4d85775229 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/geo/PopGeoQualityFollowSourceFSConfig.scala @@ -0,0 +1,24 @@ +package com.twitter.follow_recommendations.common.candidate_sources.geo + +import com.twitter.follow_recommendations.configapi.common.FeatureSwitchConfig +import com.twitter.timelines.configapi.FSBoundedParam +import com.twitter.timelines.configapi.FSName +import com.twitter.timelines.configapi.FSParam +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class PopGeoQualityFollowSourceFSConfig @Inject() () extends FeatureSwitchConfig { + override val intFSParams: Seq[FSBoundedParam[Int] with FSName] = Seq( + PopGeoQualityFollowSourceParams.PopGeoSourceGeoHashMaxPrecision, + PopGeoQualityFollowSourceParams.PopGeoSourceGeoHashMinPrecision, + PopGeoQualityFollowSourceParams.PopGeoSourceMaxResultsPerPrecision + ) + override val doubleFSParams: Seq[FSBoundedParam[Double] with FSName] = Seq( + PopGeoQualityFollowSourceParams.CandidateSourceWeight + ) + override val booleanFSParams: Seq[FSParam[Boolean] with FSName] = Seq( + PopGeoQualityFollowSourceParams.CandidateSourceEnabled, + PopGeoQualityFollowSourceParams.PopGeoSourceReturnFromAllPrecisions + ) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/geo/PopGeoQualityFollowSourceParams.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/geo/PopGeoQualityFollowSourceParams.scala new file mode 100644 index 0000000000..dac92df521 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/geo/PopGeoQualityFollowSourceParams.scala @@ -0,0 +1,42 @@ +package com.twitter.follow_recommendations.common.candidate_sources.geo + +import com.twitter.timelines.configapi.FSBoundedParam +import com.twitter.timelines.configapi.FSParam + +object PopGeoQualityFollowSourceParams { + case object CandidateSourceEnabled + extends FSParam[Boolean]("pop_geo_quality_follow_source_enabled", false) + + case object PopGeoSourceGeoHashMinPrecision + extends FSBoundedParam[Int]( + "pop_geo_quality_follow_source_geo_hash_min_precision", + default = 2, + min = 0, + max = 10) + + case object PopGeoSourceGeoHashMaxPrecision + extends FSBoundedParam[Int]( + "pop_geo_quality_follow_source_geo_hash_max_precision", + default = 3, + min = 0, + max = 10) + + case object PopGeoSourceReturnFromAllPrecisions + extends FSParam[Boolean]( + "pop_geo_quality_follow_source_return_from_all_precisions", + default = false) + + case object PopGeoSourceMaxResultsPerPrecision + extends FSBoundedParam[Int]( + "pop_geo_quality_follow_source_max_results_per_precision", + default = 200, + min = 0, + max = 1000) + + case object CandidateSourceWeight + extends FSBoundedParam[Double]( + "pop_geo_quality_follow_source_weight", + default = 200, + min = 0.001, + max = 2000) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/geo/PopGeoSource.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/geo/PopGeoSource.scala new file mode 100644 index 0000000000..2db13ed69f --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/geo/PopGeoSource.scala @@ -0,0 +1,69 @@ +package com.twitter.follow_recommendations.common.candidate_sources.geo + +import com.google.inject.Singleton +import com.google.inject.name.Named +import com.twitter.conversions.DurationOps._ +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.follow_recommendations.common.candidate_sources.base.CachedCandidateSource +import com.twitter.follow_recommendations.common.candidate_sources.base.StratoFetcherWithUnitViewSource +import com.twitter.follow_recommendations.common.constants.GuiceNamedConstants +import com.twitter.follow_recommendations.common.models.AccountProof +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.follow_recommendations.common.models.PopularInGeoProof +import com.twitter.follow_recommendations.common.models.Reason +import com.twitter.hermit.pop_geo.thriftscala.PopUsersInPlace +import com.twitter.product_mixer.core.model.common.identifier.CandidateSourceIdentifier +import com.twitter.strato.client.Fetcher +import com.twitter.util.Duration +import javax.inject.Inject + +@Singleton +class BasePopGeoSource @Inject() ( + @Named(GuiceNamedConstants.POP_USERS_IN_PLACE_FETCHER) fetcher: Fetcher[ + String, + Unit, + PopUsersInPlace + ]) extends StratoFetcherWithUnitViewSource[String, PopUsersInPlace]( + fetcher, + BasePopGeoSource.Identifier) { + + override def map(target: String, candidates: PopUsersInPlace): Seq[CandidateUser] = + BasePopGeoSource.map(target, candidates) +} + +object BasePopGeoSource { + val Identifier: CandidateSourceIdentifier = CandidateSourceIdentifier("BasePopGeoSource") + val MaxResults = 200 + + def map(target: String, candidates: PopUsersInPlace): Seq[CandidateUser] = + candidates.popUsers.sortBy(-_.score).take(BasePopGeoSource.MaxResults).view.map { candidate => + CandidateUser( + id = candidate.userId, + score = Some(candidate.score), + reason = Some( + Reason( + Some( + AccountProof( + popularInGeoProof = Some(PopularInGeoProof(location = candidates.place)) + ) + ) + ) + ) + ) + } +} + +@Singleton +class PopGeoSource @Inject() (basePopGeoSource: BasePopGeoSource, statsReceiver: StatsReceiver) + extends CachedCandidateSource[String, CandidateUser]( + basePopGeoSource, + PopGeoSource.MaxCacheSize, + PopGeoSource.CacheTTL, + statsReceiver, + PopGeoSource.Identifier) + +object PopGeoSource { + val Identifier: CandidateSourceIdentifier = CandidateSourceIdentifier("PopGeoSource") + val MaxCacheSize = 20000 + val CacheTTL: Duration = 1.hours +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/geo/PopGeoSourceFSConfig.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/geo/PopGeoSourceFSConfig.scala new file mode 100644 index 0000000000..ea3f6ce387 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/geo/PopGeoSourceFSConfig.scala @@ -0,0 +1,20 @@ +package com.twitter.follow_recommendations.common.candidate_sources.geo + +import com.twitter.follow_recommendations.configapi.common.FeatureSwitchConfig +import com.twitter.timelines.configapi.FSBoundedParam +import com.twitter.timelines.configapi.FSName +import com.twitter.timelines.configapi.FSParam +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class PopGeoSourceFSConfig @Inject() () extends FeatureSwitchConfig { + override val intFSParams: Seq[FSBoundedParam[Int] with FSName] = Seq( + PopGeoSourceParams.PopGeoSourceGeoHashMaxPrecision, + PopGeoSourceParams.PopGeoSourceMaxResultsPerPrecision, + PopGeoSourceParams.PopGeoSourceGeoHashMinPrecision, + ) + override val booleanFSParams: Seq[FSParam[Boolean] with FSName] = Seq( + PopGeoSourceParams.PopGeoSourceReturnFromAllPrecisions, + ) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/geo/PopGeoSourceParams.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/geo/PopGeoSourceParams.scala new file mode 100644 index 0000000000..a63e320b4d --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/geo/PopGeoSourceParams.scala @@ -0,0 +1,30 @@ +package com.twitter.follow_recommendations.common.candidate_sources.geo + +import com.twitter.timelines.configapi.FSBoundedParam +import com.twitter.timelines.configapi.FSParam + +object PopGeoSourceParams { + case object PopGeoSourceGeoHashMinPrecision + extends FSBoundedParam[Int]( + "pop_geo_source_geo_hash_min_precision", + default = 2, + min = 0, + max = 10) + + case object PopGeoSourceGeoHashMaxPrecision + extends FSBoundedParam[Int]( + "pop_geo_source_geo_hash_max_precision", + default = 4, + min = 0, + max = 10) + + case object PopGeoSourceReturnFromAllPrecisions + extends FSParam[Boolean]("pop_geo_source_return_from_all_precisions", default = false) + + case object PopGeoSourceMaxResultsPerPrecision + extends FSBoundedParam[Int]( + "pop_geo_source_max_results_per_precision", + default = 200, + min = 0, + max = 1000) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/geo/PopGeohashSource.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/geo/PopGeohashSource.scala new file mode 100644 index 0000000000..9447b48670 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/geo/PopGeohashSource.scala @@ -0,0 +1,36 @@ +package com.twitter.follow_recommendations.common.candidate_sources.geo + +import com.google.inject.Singleton +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.hermit.model.Algorithm +import com.twitter.product_mixer.core.model.common.identifier.CandidateSourceIdentifier +import javax.inject.Inject + +@Singleton +class PopGeohashSource @Inject() ( + popGeoSource: PopGeoSource, + statsReceiver: StatsReceiver) + extends BasePopGeohashSource( + popGeoSource = popGeoSource, + statsReceiver = statsReceiver.scope("PopGeohashSource"), + ) { + override def candidateSourceEnabled(target: Target): Boolean = true + override val identifier: CandidateSourceIdentifier = PopGeohashSource.Identifier + override def minGeohashLength(target: Target): Int = { + target.params(PopGeoSourceParams.PopGeoSourceGeoHashMinPrecision) + } + override def maxResults(target: Target): Int = { + target.params(PopGeoSourceParams.PopGeoSourceMaxResultsPerPrecision) + } + override def maxGeohashLength(target: Target): Int = { + target.params(PopGeoSourceParams.PopGeoSourceGeoHashMaxPrecision) + } + override def returnResultFromAllPrecision(target: Target): Boolean = { + target.params(PopGeoSourceParams.PopGeoSourceReturnFromAllPrecisions) + } +} + +object PopGeohashSource { + val Identifier: CandidateSourceIdentifier = CandidateSourceIdentifier( + Algorithm.PopGeohash.toString) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/geo/README.md b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/geo/README.md new file mode 100644 index 0000000000..13c2f245fa --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/geo/README.md @@ -0,0 +1,4 @@ +# Pop Geo Candidate Source +Provides the most followed / quality followed accounts in a specific country and a geolocation within past 2 weeks. +* A "quality follow" refers to any follow that leads to visible engagement, such as favorites, mentions, retweets, direct messages, replies, and quote tweets. The engagement must be allowed in either direction, and must occur on the day of the follow or within one subsequent day. Additionally, there must be no unfollowing, blocking, muting, or reporting of the account in the same time period. +* The minimum geolocation precision used is ±20 km (12 mi), and precise user geolocation is not utilized. diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/ppmi_locale_follow/BUILD b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/ppmi_locale_follow/BUILD new file mode 100644 index 0000000000..0ff68490c8 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/ppmi_locale_follow/BUILD @@ -0,0 +1,23 @@ +scala_library( + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/com/google/inject:guice", + "3rdparty/jvm/com/google/inject/extensions:guice-assistedinject", + "3rdparty/jvm/net/codingwell:scala-guice", + "3rdparty/jvm/org/slf4j:slf4j-api", + "finatra/inject/inject-core/src/main/scala", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/base", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/strato", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/constants", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/common", + "src/thrift/com/twitter/hermit/candidate:hermit-candidate-scala", + "strato/config/columns/onboarding:onboarding-strato-client", + "strato/config/columns/onboarding/userrecs:userrecs-strato-client", + "strato/src/main/scala/com/twitter/strato/client", + "util/util-slf4j-api/src/main/scala", + ], +) diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/ppmi_locale_follow/PPMILocaleFollowSource.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/ppmi_locale_follow/PPMILocaleFollowSource.scala new file mode 100644 index 0000000000..e22fb465d5 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/ppmi_locale_follow/PPMILocaleFollowSource.scala @@ -0,0 +1,84 @@ +package com.twitter.follow_recommendations.common.candidate_sources.ppmi_locale_follow + +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.follow_recommendations.common.candidate_sources.ppmi_locale_follow.PPMILocaleFollowSourceParams.CandidateSourceEnabled +import com.twitter.follow_recommendations.common.candidate_sources.ppmi_locale_follow.PPMILocaleFollowSourceParams.LocaleToExcludeFromRecommendation +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.hermit.model.Algorithm +import com.twitter.product_mixer.core.functional_component.candidate_source.CandidateSource +import com.twitter.product_mixer.core.model.common.identifier.CandidateSourceIdentifier +import com.twitter.product_mixer.core.model.marshalling.request.HasClientContext +import com.twitter.stitch.Stitch + +import javax.inject.Inject +import javax.inject.Singleton +import com.twitter.strato.generated.client.onboarding.UserPreferredLanguagesOnUserClientColumn +import com.twitter.strato.generated.client.onboarding.userrecs.LocaleFollowPpmiClientColumn +import com.twitter.timelines.configapi.HasParams + +/** + * Fetches candidates based on the Positive Pointwise Mutual Information (PPMI) statistic + * for a set of locales + * */ +@Singleton +class PPMILocaleFollowSource @Inject() ( + userPreferredLanguagesOnUserClientColumn: UserPreferredLanguagesOnUserClientColumn, + localeFollowPpmiClientColumn: LocaleFollowPpmiClientColumn, + statsReceiver: StatsReceiver) + extends CandidateSource[HasClientContext with HasParams, CandidateUser] { + + override val identifier: CandidateSourceIdentifier = PPMILocaleFollowSource.Identifier + private val stats = statsReceiver.scope("PPMILocaleFollowSource") + + override def apply(target: HasClientContext with HasParams): Stitch[Seq[CandidateUser]] = { + (for { + countryCode <- target.getCountryCode + userId <- target.getOptionalUserId + } yield { + getPreferredLocales(userId, countryCode.toLowerCase()) + .flatMap { locale => + stats.addGauge("allLocale") { + locale.length + } + val filteredLocale = + locale.filter(!target.params(LocaleToExcludeFromRecommendation).contains(_)) + stats.addGauge("postFilterLocale") { + filteredLocale.length + } + if (target.params(CandidateSourceEnabled)) { + getPPMILocaleFollowCandidates(filteredLocale) + } else Stitch(Seq.empty) + } + .map(_.sortBy(_.score)(Ordering[Option[Double]].reverse) + .take(PPMILocaleFollowSource.DefaultMaxCandidatesToReturn)) + }).getOrElse(Stitch.Nil) + } + + private def getPPMILocaleFollowCandidates( + locales: Seq[String] + ): Stitch[Seq[CandidateUser]] = { + Stitch + .traverse(locales) { locale => + // Get PPMI candidates for each locale + localeFollowPpmiClientColumn.fetcher + .fetch(locale) + .map(_.v + .map(_.candidates).getOrElse(Nil).map { candidate => + CandidateUser(id = candidate.userId, score = Some(candidate.score)) + }.map(_.withCandidateSource(identifier))) + }.map(_.flatten) + } + + private def getPreferredLocales(userId: Long, countryCode: String): Stitch[Seq[String]] = { + userPreferredLanguagesOnUserClientColumn.fetcher + .fetch(userId) + .map(_.v.map(_.languages).getOrElse(Nil).map { lang => + s"$countryCode-$lang".toLowerCase + }) + } +} + +object PPMILocaleFollowSource { + val Identifier = CandidateSourceIdentifier(Algorithm.PPMILocaleFollow.toString) + val DefaultMaxCandidatesToReturn = 100 +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/ppmi_locale_follow/PPMILocaleFollowSourceFSConfig.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/ppmi_locale_follow/PPMILocaleFollowSourceFSConfig.scala new file mode 100644 index 0000000000..8a40ca92dd --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/ppmi_locale_follow/PPMILocaleFollowSourceFSConfig.scala @@ -0,0 +1,24 @@ +package com.twitter.follow_recommendations.common.candidate_sources.ppmi_locale_follow + +import com.twitter.follow_recommendations.configapi.common.FeatureSwitchConfig +import com.twitter.timelines.configapi.FSBoundedParam +import com.twitter.timelines.configapi.FSName +import com.twitter.timelines.configapi.Param + +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class PPMILocaleFollowSourceFSConfig @Inject() () extends FeatureSwitchConfig { + override val booleanFSParams: Seq[Param[Boolean] with FSName] = Seq( + PPMILocaleFollowSourceParams.CandidateSourceEnabled, + ) + + override val stringSeqFSParams: Seq[Param[Seq[String]] with FSName] = Seq( + PPMILocaleFollowSourceParams.LocaleToExcludeFromRecommendation, + ) + + override val doubleFSParams: Seq[FSBoundedParam[Double]] = Seq( + PPMILocaleFollowSourceParams.CandidateSourceWeight, + ) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/ppmi_locale_follow/PPMILocaleFollowSourceParams.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/ppmi_locale_follow/PPMILocaleFollowSourceParams.scala new file mode 100644 index 0000000000..18aac545d1 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/ppmi_locale_follow/PPMILocaleFollowSourceParams.scala @@ -0,0 +1,22 @@ +package com.twitter.follow_recommendations.common.candidate_sources.ppmi_locale_follow + +import com.twitter.timelines.configapi.FSBoundedParam +import com.twitter.timelines.configapi.FSParam + +class PPMILocaleFollowSourceParams {} +object PPMILocaleFollowSourceParams { + case object LocaleToExcludeFromRecommendation + extends FSParam[Seq[String]]( + "ppmilocale_follow_source_locales_to_exclude_from_recommendation", + default = Seq.empty) + + case object CandidateSourceEnabled + extends FSParam[Boolean]("ppmilocale_follow_source_enabled", true) + + case object CandidateSourceWeight + extends FSBoundedParam[Double]( + "ppmilocale_follow_source_candidate_source_weight", + default = 1, + min = 0.001, + max = 2000) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/ppmi_locale_follow/README.md b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/ppmi_locale_follow/README.md new file mode 100644 index 0000000000..48cbee1124 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/ppmi_locale_follow/README.md @@ -0,0 +1,6 @@ +# PPMI Locale Follow Candidate Source +Provides accounts based on PPMI ([Positive Pointwise Mutual Information](https://en.wikipedia.org/wiki/Pointwise_mutual_information#Positive_PMI)) using follow actions as a feature for a specific local (language + country) within a week. In simpler terms, it provides a list of the most followed accounts for a given country and language input, based on the PPMI algorithm. + +PPMI is a statistical measure of the association between two events. In this case, it measures the association between the follow actions and the accounts being followed. + +In summary, the service utilizes PPMI and follow actions to provide a list of the most followed accounts for a specific country and language input. diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/promoted_accounts/BUILD b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/promoted_accounts/BUILD new file mode 100644 index 0000000000..27caedd22b --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/promoted_accounts/BUILD @@ -0,0 +1,11 @@ +scala_library( + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/adserver", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/socialgraph", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models", + "src/thrift/com/twitter/socialgraph:thrift-scala", + ], +) diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/promoted_accounts/PromotedAccountsCandidateSource.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/promoted_accounts/PromotedAccountsCandidateSource.scala new file mode 100644 index 0000000000..ff2ad4cd95 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/promoted_accounts/PromotedAccountsCandidateSource.scala @@ -0,0 +1,111 @@ +package com.twitter.follow_recommendations.common.candidate_sources.promoted_accounts + +import com.twitter.adserver.thriftscala.AdServerException +import com.twitter.adserver.{thriftscala => adthrift} +import com.twitter.finagle.TimeoutException +import com.twitter.finagle.stats.Counter +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.follow_recommendations.common.clients.adserver.AdRequest +import com.twitter.follow_recommendations.common.clients.adserver.AdserverClient +import com.twitter.follow_recommendations.common.clients.socialgraph.SocialGraphClient +import com.twitter.follow_recommendations.common.models.FollowProof +import com.twitter.hermit.model.Algorithm +import com.twitter.inject.Logging +import com.twitter.product_mixer.core.functional_component.candidate_source.CandidateSource +import com.twitter.product_mixer.core.model.common.identifier.CandidateSourceIdentifier +import com.twitter.stitch.Stitch +import javax.inject.Inject +import javax.inject.Singleton + +case class PromotedCandidateUser( + id: Long, + position: Int, + adImpression: adthrift.AdImpression, + followProof: FollowProof, + primaryCandidateSource: Option[CandidateSourceIdentifier]) + +@Singleton +class PromotedAccountsCandidateSource @Inject() ( + adserverClient: AdserverClient, + sgsClient: SocialGraphClient, + statsReceiver: StatsReceiver) + extends CandidateSource[AdRequest, PromotedCandidateUser] + with Logging { + + override val identifier: CandidateSourceIdentifier = + PromotedAccountsCandidateSource.Identifier + + val stats: StatsReceiver = statsReceiver.scope(identifier.name) + val failureStat: StatsReceiver = stats.scope("failures") + val adServerExceptionsCounter: Counter = failureStat.counter("AdServerException") + val timeoutCounter: Counter = failureStat.counter("TimeoutException") + + def apply(request: AdRequest): Stitch[Seq[PromotedCandidateUser]] = { + adserverClient + .getAdImpressions(request) + .rescue { + case e: TimeoutException => + timeoutCounter.incr() + logger.warn("Timeout on Adserver", e) + Stitch.Nil + case e: AdServerException => + adServerExceptionsCounter.incr() + logger.warn("Failed to fetch ads", e) + Stitch.Nil + } + .flatMap { adImpressions: Seq[adthrift.AdImpression] => + profileNumResults(adImpressions.size, "results_from_ad_server") + val idToImpMap = (for { + imp <- adImpressions + promotedAccountId <- imp.promotedAccountId + } yield promotedAccountId -> imp).toMap + request.clientContext.userId + .map { userId => + sgsClient + .getIntersections( + userId, + adImpressions.filter(shouldShowSocialContext).flatMap(_.promotedAccountId), + PromotedAccountsCandidateSource.NumIntersections + ).map { promotedAccountWithIntersections => + idToImpMap.map { + case (promotedAccountId, imp) => + PromotedCandidateUser( + promotedAccountId, + imp.insertionPosition + .map(_.toInt).getOrElse( + getInsertionPositionDefaultValue(request.isTest.getOrElse(false)) + ), + imp, + promotedAccountWithIntersections + .getOrElse(promotedAccountId, FollowProof(Nil, 0)), + Some(identifier) + ) + }.toSeq + }.onSuccess(result => profileNumResults(result.size, "final_results")) + }.getOrElse(Stitch.Nil) + } + } + + private def shouldShowSocialContext(imp: adthrift.AdImpression): Boolean = + imp.experimentValues.exists { expValues => + expValues.get("display.display_style").contains("show_social_context") + } + + private def getInsertionPositionDefaultValue(isTest: Boolean): Int = { + if (isTest) 0 else -1 + } + + private def profileNumResults(resultsSize: Int, statName: String): Unit = { + if (resultsSize <= 5) { + stats.scope(statName).counter(resultsSize.toString).incr() + } else { + stats.scope(statName).counter("more_than_5").incr() + } + } +} + +object PromotedAccountsCandidateSource { + val Identifier: CandidateSourceIdentifier = CandidateSourceIdentifier( + Algorithm.PromotedAccount.toString) + val NumIntersections = 3 +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/promoted_accounts/README.md b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/promoted_accounts/README.md new file mode 100644 index 0000000000..1091e1d885 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/promoted_accounts/README.md @@ -0,0 +1,2 @@ +# Promoted Accounts Candidate Source +Promoted accounts returned from Ads server. diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/real_graph/BUILD b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/real_graph/BUILD new file mode 100644 index 0000000000..07c3c66653 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/real_graph/BUILD @@ -0,0 +1,24 @@ +scala_library( + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/com/google/inject:guice", + "3rdparty/jvm/com/google/inject/extensions:guice-assistedinject", + "3rdparty/jvm/net/codingwell:scala-guice", + "3rdparty/jvm/org/slf4j:slf4j-api", + "finatra/inject/inject-core/src/main/scala", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/base", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/real_time_real_graph", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/constants", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/stores", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/common", + "strato/config/columns/onboarding/realGraph:realGraph-strato-client", + "strato/config/columns/onboarding/userrecs:userrecs-strato-client", + "strato/config/columns/recommendations/twistly:twistly-strato-client", + "strato/src/main/scala/com/twitter/strato/client", + "util/util-slf4j-api/src/main/scala", + ], +) diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/real_graph/README.md b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/real_graph/README.md new file mode 100644 index 0000000000..4ba5a8c11d --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/real_graph/README.md @@ -0,0 +1,6 @@ +# RealGraph Candidate Source +Provides out-of-network RealGraph candidates for a given user. RealGraph is a user-user graph dataset that aims to measure the strength of the relationship between two users. + +RealGraph comprises two components: a real-time pipeline that tracks various counts and relationships between user-user edges (such as the number of favorites, replies, retweets, clicks, whether followed, muted, or blocked), and an offline pipeline of a larger set of such user-user edge counts and relationships. Currently, the top k in-network scores have been exported for use by various teams. + +The RealGraph dataset is used to predict user interactions at Twitter, and is based on the paper "[Realgraph: User interaction prediction at Twitter](http://www.ueo-workshop.com/wp-content/uploads/2014/04/sig-alternate.pdf)" by the UEO workshop at KDD'14. diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/real_graph/RealGraphOonFSConfig.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/real_graph/RealGraphOonFSConfig.scala new file mode 100644 index 0000000000..df9c8ac68d --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/real_graph/RealGraphOonFSConfig.scala @@ -0,0 +1,27 @@ +package com.twitter.follow_recommendations.common.candidate_sources.real_graph + +import com.twitter.follow_recommendations.configapi.common.FeatureSwitchConfig +import com.twitter.timelines.configapi.FSBoundedParam +import com.twitter.timelines.configapi.FSName +import com.twitter.timelines.configapi.Param +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class RealGraphOonFSConfig @Inject() () extends FeatureSwitchConfig { + override val booleanFSParams: Seq[Param[Boolean] with FSName] = + Seq( + RealGraphOonParams.IncludeRealGraphOonCandidates, + RealGraphOonParams.TryToReadRealGraphOonCandidates, + RealGraphOonParams.UseV2 + ) + override val doubleFSParams: Seq[FSBoundedParam[Double]] = + Seq( + RealGraphOonParams.ScoreThreshold + ) + override val intFSParams: Seq[FSBoundedParam[Int]] = + Seq( + RealGraphOonParams.RealGraphOonResultCountThreshold, + RealGraphOonParams.MaxResults, + ) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/real_graph/RealGraphOonParams.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/real_graph/RealGraphOonParams.scala new file mode 100644 index 0000000000..feaa1d7c74 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/real_graph/RealGraphOonParams.scala @@ -0,0 +1,47 @@ +package com.twitter.follow_recommendations.common.candidate_sources.real_graph + +import com.twitter.timelines.configapi.FSBoundedParam +import com.twitter.timelines.configapi.FSParam + +object RealGraphOonParams { + case object IncludeRealGraphOonCandidates + extends FSParam[Boolean]( + "real_graph_oon_include_candidates", + false + ) + case object TryToReadRealGraphOonCandidates + extends FSParam[Boolean]( + "real_graph_oon_try_to_read_candidates", + false + ) + case object RealGraphOonResultCountThreshold + extends FSBoundedParam[Int]( + "real_graph_oon_result_count_threshold", + default = 1, + min = 0, + max = Integer.MAX_VALUE + ) + + case object UseV2 + extends FSParam[Boolean]( + "real_graph_oon_use_v2", + false + ) + + case object ScoreThreshold + extends FSBoundedParam[Double]( + "real_graph_oon_score_threshold", + default = 0.26, + min = 0, + max = 1.0 + ) + + case object MaxResults + extends FSBoundedParam[Int]( + "real_graph_oon_max_results", + default = 200, + min = 0, + max = 1000 + ) + +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/real_graph/RealGraphOonV2Source.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/real_graph/RealGraphOonV2Source.scala new file mode 100644 index 0000000000..3c709770cb --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/real_graph/RealGraphOonV2Source.scala @@ -0,0 +1,58 @@ +package com.twitter.follow_recommendations.common.candidate_sources.real_graph + +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.product_mixer.core.functional_component.candidate_source.CandidateSource +import com.twitter.hermit.model.Algorithm +import com.twitter.product_mixer.core.model.common.identifier.CandidateSourceIdentifier +import com.twitter.product_mixer.core.model.marshalling.request.HasClientContext +import com.twitter.stitch.Stitch +import com.twitter.strato.generated.client.onboarding.realGraph.UserRealgraphOonV2ClientColumn +import com.twitter.timelines.configapi.HasParams +import com.twitter.wtf.candidate.thriftscala.CandidateSeq +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class RealGraphOonV2Source @Inject() ( + realGraphClientColumn: UserRealgraphOonV2ClientColumn) + extends CandidateSource[HasParams with HasClientContext, CandidateUser] { + + override val identifier: CandidateSourceIdentifier = + RealGraphOonV2Source.Identifier + + override def apply(request: HasParams with HasClientContext): Stitch[Seq[CandidateUser]] = { + request.getOptionalUserId + .map { userId => + realGraphClientColumn.fetcher + .fetch(userId) + .map { result => + result.v + .map { candidates => parseStratoResults(request, candidates) } + .getOrElse(Nil) + // returned candidates are sorted by score in descending order + .take(request.params(RealGraphOonParams.MaxResults)) + .map(_.withCandidateSource(identifier)) + } + }.getOrElse(Stitch(Seq.empty)) + } + + private def parseStratoResults( + request: HasParams with HasClientContext, + candidateSeqThrift: CandidateSeq + ): Seq[CandidateUser] = { + candidateSeqThrift.candidates.collect { + case candidate if candidate.score >= request.params(RealGraphOonParams.ScoreThreshold) => + CandidateUser( + candidate.userId, + Some(candidate.score) + ) + } + } + +} + +object RealGraphOonV2Source { + val Identifier: CandidateSourceIdentifier = CandidateSourceIdentifier( + Algorithm.RealGraphOonV2.toString + ) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/real_graph/RealGraphSource.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/real_graph/RealGraphSource.scala new file mode 100644 index 0000000000..7aa2aa8af3 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/real_graph/RealGraphSource.scala @@ -0,0 +1,40 @@ +package com.twitter.follow_recommendations.common.candidate_sources.real_graph + +import com.twitter.follow_recommendations.common.clients.real_time_real_graph.RealTimeRealGraphClient +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.hermit.model.Algorithm +import com.twitter.product_mixer.core.functional_component.candidate_source.CandidateSource +import com.twitter.product_mixer.core.model.common.identifier.CandidateSourceIdentifier +import com.twitter.product_mixer.core.model.marshalling.request.HasClientContext +import com.twitter.stitch.Stitch +import com.twitter.timelines.configapi.HasParams +import javax.inject.Inject +import javax.inject.Singleton + +/** + * This source gets the already followed edges from the real graph column as a candidate source. + */ +@Singleton +class RealGraphSource @Inject() ( + realGraph: RealTimeRealGraphClient) + extends CandidateSource[HasParams with HasClientContext, CandidateUser] { + override val identifier: CandidateSourceIdentifier = RealGraphSource.Identifier + + override def apply(request: HasParams with HasClientContext): Stitch[Seq[CandidateUser]] = { + request.getOptionalUserId + .map { userId => + realGraph.getRealGraphWeights(userId).map { scoreMap => + scoreMap.map { + case (candidateId, realGraphScore) => + CandidateUser(id = candidateId, score = Some(realGraphScore)) + .withCandidateSource(identifier) + }.toSeq + } + }.getOrElse(Stitch.Nil) + } +} + +object RealGraphSource { + val Identifier: CandidateSourceIdentifier = CandidateSourceIdentifier( + Algorithm.RealGraphFollowed.toString) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/recent_engagement/BUILD b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/recent_engagement/BUILD new file mode 100644 index 0000000000..b2764c42ec --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/recent_engagement/BUILD @@ -0,0 +1,29 @@ +scala_library( + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/com/google/inject:guice", + "3rdparty/jvm/com/google/inject/extensions:guice-assistedinject", + "3rdparty/jvm/net/codingwell:scala-guice", + "3rdparty/jvm/org/slf4j:slf4j-api", + "discovery-ds/src/main/thrift/com/twitter/dds/jobs/repeated_profile_visits:profile_visit-scala", + "finatra/inject/inject-core/src/main/scala", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/base", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims_expansion", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/real_time_real_graph", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/strato", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/constants", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/common", + "hermit/hermit-core/src/main/scala/com/twitter/hermit/model", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/candidate_source", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/common/identifier", + "src/thrift/com/twitter/experiments/general_metrics:general_metrics-scala", + "strato/config/columns/rux:rux-strato-client", + "strato/src/main/scala/com/twitter/strato/client", + "util/util-slf4j-api/src/main/scala", + ], +) diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/recent_engagement/README.md b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/recent_engagement/README.md new file mode 100644 index 0000000000..616a3f7ed0 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/recent_engagement/README.md @@ -0,0 +1,4 @@ +# Recent Engagement Candidate Source +Provides recently engaged accounts for a given user: +* Explicit engagements: like, retweet, reply +* Implicit engagements: profile visit diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/recent_engagement/RecentEngagementDirectFollowSource.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/recent_engagement/RecentEngagementDirectFollowSource.scala new file mode 100644 index 0000000000..edfaac5e69 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/recent_engagement/RecentEngagementDirectFollowSource.scala @@ -0,0 +1,38 @@ +package com.twitter.follow_recommendations.common.candidate_sources.recent_engagement + +import com.twitter.follow_recommendations.common.clients.real_time_real_graph.RealTimeRealGraphClient +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.hermit.model.Algorithm +import com.twitter.product_mixer.core.functional_component.candidate_source.CandidateSource +import com.twitter.product_mixer.core.model.common.identifier.CandidateSourceIdentifier +import com.twitter.stitch.Stitch +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class RecentEngagementDirectFollowSource @Inject() ( + realTimeRealGraphClient: RealTimeRealGraphClient) + extends CandidateSource[Long, CandidateUser] { + + val identifier: CandidateSourceIdentifier = + RecentEngagementDirectFollowSource.Identifier + + /** + * Generate a list of candidates for the target using RealtimeGraphClient + * and RecentEngagementStore. + */ + override def apply(targetUserId: Long): Stitch[Seq[CandidateUser]] = { + realTimeRealGraphClient + .getUsersRecentlyEngagedWith( + userId = targetUserId, + engagementScoreMap = RealTimeRealGraphClient.EngagementScoreMap, + includeDirectFollowCandidates = true, + includeNonDirectFollowCandidates = false + ) + .map(_.map(_.withCandidateSource(identifier)).sortBy(-_.score.getOrElse(0.0))) + } +} + +object RecentEngagementDirectFollowSource { + val Identifier = CandidateSourceIdentifier(Algorithm.RecentEngagementDirectFollow.toString) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/recent_engagement/RecentEngagementNonDirectFollowSource.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/recent_engagement/RecentEngagementNonDirectFollowSource.scala new file mode 100644 index 0000000000..46572da716 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/recent_engagement/RecentEngagementNonDirectFollowSource.scala @@ -0,0 +1,38 @@ +package com.twitter.follow_recommendations.common.candidate_sources.recent_engagement + +import com.twitter.follow_recommendations.common.clients.real_time_real_graph.RealTimeRealGraphClient +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.hermit.model.Algorithm +import com.twitter.product_mixer.core.functional_component.candidate_source.CandidateSource +import com.twitter.product_mixer.core.model.common.identifier.CandidateSourceIdentifier +import com.twitter.stitch.Stitch +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class RecentEngagementNonDirectFollowSource @Inject() ( + realTimeRealGraphClient: RealTimeRealGraphClient) + extends CandidateSource[Long, CandidateUser] { + + val identifier: CandidateSourceIdentifier = + RecentEngagementNonDirectFollowSource.Identifier + + /** + * Generate a list of candidates for the target using RealtimeGraphClient + * and RecentEngagementStore. + */ + override def apply(targetUserId: Long): Stitch[Seq[CandidateUser]] = { + realTimeRealGraphClient + .getUsersRecentlyEngagedWith( + userId = targetUserId, + engagementScoreMap = RealTimeRealGraphClient.EngagementScoreMap, + includeDirectFollowCandidates = false, + includeNonDirectFollowCandidates = true + ) + .map(_.map(_.withCandidateSource(identifier)).sortBy(-_.score.getOrElse(0.0))) + } +} + +object RecentEngagementNonDirectFollowSource { + val Identifier = CandidateSourceIdentifier(Algorithm.RecentEngagementNonDirectFollow.toString) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/recent_engagement/RepeatedProfileVisitsFSConfig.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/recent_engagement/RepeatedProfileVisitsFSConfig.scala new file mode 100644 index 0000000000..5aaccae2be --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/recent_engagement/RepeatedProfileVisitsFSConfig.scala @@ -0,0 +1,22 @@ +package com.twitter.follow_recommendations.common.candidate_sources.recent_engagement + +import com.twitter.follow_recommendations.configapi.common.FeatureSwitchConfig +import com.twitter.timelines.configapi.FSBoundedParam +import com.twitter.timelines.configapi.FSName +import com.twitter.timelines.configapi.Param +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class RepeatedProfileVisitsFSConfig @Inject() () extends FeatureSwitchConfig { + override val booleanFSParams: Seq[Param[Boolean] with FSName] = + Seq( + RepeatedProfileVisitsParams.IncludeCandidates, + RepeatedProfileVisitsParams.UseOnlineDataset, + ) + override val intFSParams: Seq[FSBoundedParam[Int]] = + Seq( + RepeatedProfileVisitsParams.RecommendationThreshold, + RepeatedProfileVisitsParams.BucketingThreshold, + ) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/recent_engagement/RepeatedProfileVisitsParams.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/recent_engagement/RepeatedProfileVisitsParams.scala new file mode 100644 index 0000000000..5402d38f43 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/recent_engagement/RepeatedProfileVisitsParams.scala @@ -0,0 +1,37 @@ +package com.twitter.follow_recommendations.common.candidate_sources.recent_engagement + +import com.twitter.timelines.configapi.FSBoundedParam +import com.twitter.timelines.configapi.FSParam + +object RepeatedProfileVisitsParams { + + // If RepeatedProfileVisitsSource is run and there are recommended candidates for the target user, whether or not + // to actually include such candidates in our output recommendations. This FS will be used to control bucketing of + // users into control vs treatment buckets. + case object IncludeCandidates + extends FSParam[Boolean](name = "repeated_profile_visits_include_candidates", default = false) + + // The threshold at or above which we will consider a profile to have been visited "frequently enough" to recommend + // the profile to the target user. + case object RecommendationThreshold + extends FSBoundedParam[Int]( + name = "repeated_profile_visits_recommendation_threshold", + default = 3, + min = 0, + max = Integer.MAX_VALUE) + + // The threshold at or above which we will consider a profile to have been visited "frequently enough" to recommend + // the profile to the target user. + case object BucketingThreshold + extends FSBoundedParam[Int]( + name = "repeated_profile_visits_bucketing_threshold", + default = 3, + min = 0, + max = Integer.MAX_VALUE) + + // Whether or not to use the online dataset (which has repeated profile visits information updated to within minutes) + // instead of the offline dataset (updated via offline jobs, which can have delays of hours to days). + case object UseOnlineDataset + extends FSParam[Boolean](name = "repeated_profile_visits_use_online_dataset", default = true) + +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/recent_engagement/RepeatedProfileVisitsSource.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/recent_engagement/RepeatedProfileVisitsSource.scala new file mode 100644 index 0000000000..c4b4aa3e76 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/recent_engagement/RepeatedProfileVisitsSource.scala @@ -0,0 +1,157 @@ +package com.twitter.follow_recommendations.common.candidate_sources.recent_engagement + +import com.google.inject.Inject +import com.google.inject.Singleton +import com.twitter.dds.jobs.repeated_profile_visits.thriftscala.ProfileVisitorInfo +import com.twitter.experiments.general_metrics.thriftscala.IdType +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.follow_recommendations.common.clients.real_time_real_graph.Engagement +import com.twitter.follow_recommendations.common.clients.real_time_real_graph.RealTimeRealGraphClient +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.timelines.configapi.HasParams +import com.twitter.timelines.configapi.Params +import com.twitter.hermit.model.Algorithm +import com.twitter.inject.Logging +import com.twitter.product_mixer.core.functional_component.candidate_source.CandidateSource +import com.twitter.product_mixer.core.model.common.identifier.CandidateSourceIdentifier +import com.twitter.product_mixer.core.model.marshalling.request.HasClientContext +import com.twitter.stitch.Stitch +import com.twitter.strato.generated.client.rux.RepeatedProfileVisitsAggregateClientColumn + +@Singleton +class RepeatedProfileVisitsSource @Inject() ( + repeatedProfileVisitsAggregateClientColumn: RepeatedProfileVisitsAggregateClientColumn, + realTimeRealGraphClient: RealTimeRealGraphClient, + statsReceiver: StatsReceiver) + extends CandidateSource[HasParams with HasClientContext, CandidateUser] + with Logging { + + val identifier: CandidateSourceIdentifier = + RepeatedProfileVisitsSource.Identifier + + val sourceStatsReceiver = statsReceiver.scope("repeated_profile_visits_source") + val offlineFetchErrorCounter = sourceStatsReceiver.counter("offline_fetch_error") + val offlineFetchSuccessCounter = sourceStatsReceiver.counter("offline_fetch_success") + val onlineFetchErrorCounter = sourceStatsReceiver.counter("online_fetch_error") + val onlineFetchSuccessCounter = sourceStatsReceiver.counter("online_fetch_success") + val noRepeatedProfileVisitsAboveBucketingThresholdCounter = + sourceStatsReceiver.counter("no_repeated_profile_visits_above_bucketing_threshold") + val hasRepeatedProfileVisitsAboveBucketingThresholdCounter = + sourceStatsReceiver.counter("has_repeated_profile_visits_above_bucketing_threshold") + val noRepeatedProfileVisitsAboveRecommendationsThresholdCounter = + sourceStatsReceiver.counter("no_repeated_profile_visits_above_recommendations_threshold") + val hasRepeatedProfileVisitsAboveRecommendationsThresholdCounter = + sourceStatsReceiver.counter("has_repeated_profile_visits_above_recommendations_threshold") + val includeCandidatesCounter = sourceStatsReceiver.counter("include_candidates") + val noIncludeCandidatesCounter = sourceStatsReceiver.counter("no_include_candidates") + + // Returns visited user -> visit count, via off dataset. + def applyWithOfflineDataset(targetUserId: Long): Stitch[Map[Long, Int]] = { + repeatedProfileVisitsAggregateClientColumn.fetcher + .fetch(ProfileVisitorInfo(id = targetUserId, idType = IdType.User)).map(_.v) + .handle { + case e: Throwable => + logger.error("Strato fetch for RepeatedProfileVisitsAggregateClientColumn failed: " + e) + offlineFetchErrorCounter.incr() + None + }.onSuccess { result => + offlineFetchSuccessCounter.incr() + }.map { resultOption => + resultOption + .flatMap { result => + result.profileVisitSet.map { profileVisitSet => + profileVisitSet + .filter(profileVisit => profileVisit.totalTargetVisitsInLast14Days.getOrElse(0) > 0) + .filter(profileVisit => !profileVisit.doesSourceIdFollowTargetId.getOrElse(false)) + .flatMap { profileVisit => + (profileVisit.targetId, profileVisit.totalTargetVisitsInLast14Days) match { + case (Some(targetId), Some(totalVisitsInLast14Days)) => + Some(targetId -> totalVisitsInLast14Days) + case _ => None + } + }.toMap[Long, Int] + } + }.getOrElse(Map.empty) + } + } + + // Returns visited user -> visit count, via online dataset. + def applyWithOnlineData(targetUserId: Long): Stitch[Map[Long, Int]] = { + val visitedUserToEngagementsStitch: Stitch[Map[Long, Seq[Engagement]]] = + realTimeRealGraphClient.getRecentProfileViewEngagements(targetUserId) + visitedUserToEngagementsStitch + .onFailure { f => + onlineFetchErrorCounter.incr() + }.onSuccess { result => + onlineFetchSuccessCounter.incr() + }.map { visitedUserToEngagements => + visitedUserToEngagements + .mapValues(engagements => engagements.size) + } + } + + def getRepeatedVisitedAccounts(params: Params, targetUserId: Long): Stitch[Map[Long, Int]] = { + var results: Stitch[Map[Long, Int]] = Stitch.value(Map.empty) + if (params.getBoolean(RepeatedProfileVisitsParams.UseOnlineDataset)) { + results = applyWithOnlineData(targetUserId) + } else { + results = applyWithOfflineDataset(targetUserId) + } + // Only keep users that had non-zero engagement counts. + results.map(_.filter(input => input._2 > 0)) + } + + def getRecommendations(params: Params, userId: Long): Stitch[Seq[CandidateUser]] = { + val recommendationThreshold = params.getInt(RepeatedProfileVisitsParams.RecommendationThreshold) + val bucketingThreshold = params.getInt(RepeatedProfileVisitsParams.BucketingThreshold) + + // Get the list of repeatedly visited profilts. Only keep accounts with >= bucketingThreshold visits. + val repeatedVisitedAccountsStitch: Stitch[Map[Long, Int]] = + getRepeatedVisitedAccounts(params, userId).map(_.filter(kv => kv._2 >= bucketingThreshold)) + + repeatedVisitedAccountsStitch.map { candidates => + // Now check if we should includeCandidates (e.g. whether user is in control bucket or treatment buckets). + if (candidates.isEmpty) { + // User has not visited any accounts above bucketing threshold. We will not bucket user into experiment. Just + // don't return no candidates. + noRepeatedProfileVisitsAboveBucketingThresholdCounter.incr() + Seq.empty + } else { + hasRepeatedProfileVisitsAboveBucketingThresholdCounter.incr() + if (!params.getBoolean(RepeatedProfileVisitsParams.IncludeCandidates)) { + // User has reached bucketing criteria. We check whether to include candidates (e.g. checking which bucket + // the user is in for the experiment). In this case the user is in a bucket to not include any candidates. + noIncludeCandidatesCounter.incr() + Seq.empty + } else { + includeCandidatesCounter.incr() + // We should include candidates. Include any candidates above recommendation thresholds. + val outputCandidatesSeq = candidates + .filter(kv => kv._2 >= recommendationThreshold).map { kv => + val user = kv._1 + val visitCount = kv._2 + CandidateUser(user, Some(visitCount.toDouble)) + .withCandidateSource(RepeatedProfileVisitsSource.Identifier) + }.toSeq + if (outputCandidatesSeq.isEmpty) { + noRepeatedProfileVisitsAboveRecommendationsThresholdCounter.incr() + } else { + hasRepeatedProfileVisitsAboveRecommendationsThresholdCounter.incr() + } + outputCandidatesSeq + } + } + } + } + + override def apply(request: HasParams with HasClientContext): Stitch[Seq[CandidateUser]] = { + request.getOptionalUserId + .map { userId => + getRecommendations(request.params, userId) + }.getOrElse(Stitch.Nil) + } +} + +object RepeatedProfileVisitsSource { + val Identifier = CandidateSourceIdentifier(Algorithm.RepeatedProfileVisits.toString) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/salsa/BUILD b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/salsa/BUILD new file mode 100644 index 0000000000..78a5729e24 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/salsa/BUILD @@ -0,0 +1,21 @@ +scala_library( + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/com/google/inject:guice", + "3rdparty/jvm/com/google/inject/extensions:guice-assistedinject", + "3rdparty/jvm/net/codingwell:scala-guice", + "3rdparty/jvm/org/slf4j:slf4j-api", + "finatra/inject/inject-core/src/main/scala", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/base", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/real_time_real_graph", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/constants", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models", + "src/thrift/com/twitter/onboarding/relevance/candidates:candidates-scala", + "strato/config/columns/onboarding/userrecs:userrecs-strato-client", + "strato/src/main/scala/com/twitter/strato/client", + "util/util-slf4j-api/src/main/scala", + ], +) diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/salsa/README.md b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/salsa/README.md new file mode 100644 index 0000000000..fb6d032d8e --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/salsa/README.md @@ -0,0 +1,10 @@ +# SALSA Candidate Source +Provides an account expansion based on the SALSA PYMK (People You May Know) algorithm for a given account. The algorithm focuses on the mutual follow and address book graph, making it highly effective at providing good mutual follow recommendations. + +The SALSA algorithm constructs a local graph and performs personalized random walks to identify the best recommendations for the user. The local graph represents the community of users that are most similar to or most relevant to the user, while the personalized random walk identifies the most popular interests among them. + +For each target user, the local graph is a bipartite graph with a left-hand side (LHS) and a right-hand side (RHS). The LHS is built from several sources, including the target user, forward and reverse address books, mutual follows, recent followings, and recent followers. We choose a specified number of top candidates from these sources for each target user with different weights assigned to each source to favor the corresponding source, and build the LHS using the target user and those top candidates. The RHS consists of two parts: the top candidates from the sources mentioned above for the target user and the mutual follows of the other entries in the LHS. + +The random walk starts from the target user in the LHS and adopts a restarting strategy to realize personalization. + +In summary, the SALSA Candidate Source provides an account expansion based on the SALSA PYMK algorithm, utilizing a bipartite graph with personalized random walks to identify the most relevant and interesting recommendations for the user. diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/salsa/RecentEngagementDirectFollowSalsaExpansionSource.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/salsa/RecentEngagementDirectFollowSalsaExpansionSource.scala new file mode 100644 index 0000000000..91fbba2bab --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/salsa/RecentEngagementDirectFollowSalsaExpansionSource.scala @@ -0,0 +1,40 @@ +package com.twitter.follow_recommendations.common.candidate_sources.salsa + +import com.twitter.follow_recommendations.common.clients.real_time_real_graph.RealTimeRealGraphClient +import com.twitter.hermit.model.Algorithm +import com.twitter.product_mixer.core.model.common.identifier.CandidateSourceIdentifier +import com.twitter.stitch.Stitch +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class RecentEngagementDirectFollowSalsaExpansionSource @Inject() ( + realTimeRealGraphClient: RealTimeRealGraphClient, + salsaExpander: SalsaExpander) + extends SalsaExpansionBasedCandidateSource[Long](salsaExpander) { + + override val identifier: CandidateSourceIdentifier = + RecentEngagementDirectFollowSalsaExpansionSource.Identifier + + override def firstDegreeNodes(target: Long): Stitch[Seq[Long]] = realTimeRealGraphClient + .getUsersRecentlyEngagedWith( + target, + RealTimeRealGraphClient.EngagementScoreMap, + includeDirectFollowCandidates = true, + includeNonDirectFollowCandidates = false + ).map { recentlyFollowed => + recentlyFollowed + .take(RecentEngagementDirectFollowSalsaExpansionSource.NumFirstDegreeNodesToRetrieve) + .map(_.id) + } + + override def maxResults(target: Long): Int = + RecentEngagementDirectFollowSalsaExpansionSource.OutputSize +} + +object RecentEngagementDirectFollowSalsaExpansionSource { + val Identifier: CandidateSourceIdentifier = CandidateSourceIdentifier( + Algorithm.RecentEngagementSarusOcCur.toString) + val NumFirstDegreeNodesToRetrieve = 10 + val OutputSize = 200 +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/salsa/SalsaExpander.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/salsa/SalsaExpander.scala new file mode 100644 index 0000000000..a9390826ec --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/salsa/SalsaExpander.scala @@ -0,0 +1,117 @@ +package com.twitter.follow_recommendations.common.candidate_sources.salsa + +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.strato.generated.client.onboarding.userrecs.SalsaFirstDegreeOnUserClientColumn +import com.twitter.strato.generated.client.onboarding.userrecs.SalsaSecondDegreeOnUserClientColumn +import com.twitter.follow_recommendations.common.models.AccountProof +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.follow_recommendations.common.models.FollowProof +import com.twitter.follow_recommendations.common.models.Reason +import com.twitter.stitch.Stitch +import com.twitter.wtf.candidate.thriftscala.Candidate +import javax.inject.Inject +import javax.inject.Singleton + +case class SalsaExpandedCandidate( + candidateId: Long, + numberOfConnections: Int, + totalScore: Double, + connectingUsers: Seq[Long]) { + def toCandidateUser: CandidateUser = + CandidateUser( + id = candidateId, + score = Some(totalScore), + reason = Some(Reason( + Some(AccountProof(followProof = Some(FollowProof(connectingUsers, connectingUsers.size)))))) + ) +} + +case class SimilarUserCandidate(candidateId: Long, score: Double, similarToCandidate: Long) + +/** + * Salsa expander uses pre-computed lists of candidates for each input user id and returns the highest scored candidates in the pre-computed lists as the expansion for the corresponding input id. + */ +@Singleton +class SalsaExpander @Inject() ( + statsReceiver: StatsReceiver, + firstDegreeClient: SalsaFirstDegreeOnUserClientColumn, + secondDegreeClient: SalsaSecondDegreeOnUserClientColumn, +) { + + val stats = statsReceiver.scope("salsa_expander") + + private def similarUsers( + input: Seq[Long], + neighbors: Seq[Option[Seq[Candidate]]] + ): Seq[SalsaExpandedCandidate] = { + input + .zip(neighbors).flatMap { + case (recId, Some(neighbors)) => + neighbors.map(neighbor => SimilarUserCandidate(neighbor.userId, neighbor.score, recId)) + case _ => Nil + }.groupBy(_.candidateId).map { + case (key, neighbors) => + val scores = neighbors.map(_.score) + val connectingUsers = neighbors + .sortBy(-_.score) + .take(SalsaExpander.MaxConnectingUsersToOutputPerExpandedCandidate) + .map(_.similarToCandidate) + + SalsaExpandedCandidate(key, scores.size, scores.sum, connectingUsers) + } + .filter( + _.numberOfConnections >= math + .min(SalsaExpander.MinConnectingUsersThreshold, input.size) + ) + .toSeq + } + + def apply( + firstDegreeInput: Seq[Long], + secondDegreeInput: Seq[Long], + maxNumOfCandidatesToReturn: Int + ): Stitch[Seq[CandidateUser]] = { + + val firstDegreeNeighborsStitch = + Stitch + .collect(firstDegreeInput.map(firstDegreeClient.fetcher + .fetch(_).map(_.v.map(_.candidates.take(SalsaExpander.MaxDirectNeighbors))))).onSuccess { + firstDegreeNeighbors => + stats.stat("first_degree_neighbors").add(firstDegreeNeighbors.flatten.size) + } + + val secondDegreeNeighborsStitch = + Stitch + .collect( + secondDegreeInput.map( + secondDegreeClient.fetcher + .fetch(_).map( + _.v.map(_.candidates.take(SalsaExpander.MaxIndirectNeighbors))))).onSuccess { + secondDegreeNeighbors => + stats.stat("second_degree_neighbors").add(secondDegreeNeighbors.flatten.size) + } + + val neighborStitches = + Stitch.join(firstDegreeNeighborsStitch, secondDegreeNeighborsStitch).map { + case (first, second) => first ++ second + } + + val similarUsersToInput = neighborStitches.map { neighbors => + similarUsers(firstDegreeInput ++ secondDegreeInput, neighbors) + } + + similarUsersToInput.map { + // Rank the candidate cot users by the combined weights from the connecting users. This is the default original implementation. It is unlikely to have weight ties and thus a second ranking function is not necessary. + _.sortBy(-_.totalScore) + .take(maxNumOfCandidatesToReturn) + .map(_.toCandidateUser) + } + } +} + +object SalsaExpander { + val MaxDirectNeighbors = 2000 + val MaxIndirectNeighbors = 2000 + val MinConnectingUsersThreshold = 2 + val MaxConnectingUsersToOutputPerExpandedCandidate = 3 +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/salsa/SalsaExpansionBasedCandidateSource.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/salsa/SalsaExpansionBasedCandidateSource.scala new file mode 100644 index 0000000000..b299b966da --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/salsa/SalsaExpansionBasedCandidateSource.scala @@ -0,0 +1,32 @@ +package com.twitter.follow_recommendations.common.candidate_sources.salsa + +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.product_mixer.core.functional_component.candidate_source.CandidateSource +import com.twitter.stitch.Stitch + +abstract class SalsaExpansionBasedCandidateSource[Target](salsaExpander: SalsaExpander) + extends CandidateSource[Target, CandidateUser] { + + // Define first/second degree as empty sequences in cases of subclasses + // that don't implement one or the other. + // Example: MagicRecs only uses first degree nodes, and can ignore implementing secondDegreeNodes + // + // This allows apply(target) to combine both in the base class + def firstDegreeNodes(target: Target): Stitch[Seq[Long]] = Stitch.value(Seq()) + + def secondDegreeNodes(target: Target): Stitch[Seq[Long]] = Stitch.value(Seq()) + + // max number output results + def maxResults(target: Target): Int + + override def apply(target: Target): Stitch[Seq[CandidateUser]] = { + val nodes = Stitch.join(firstDegreeNodes(target), secondDegreeNodes(target)) + + nodes.flatMap { + case (firstDegreeCandidates, secondDegreeCandidates) => { + salsaExpander(firstDegreeCandidates, secondDegreeCandidates, maxResults(target)) + .map(_.map(_.withCandidateSource(identifier)).sortBy(-_.score.getOrElse(0.0))) + } + } + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims/BUILD b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims/BUILD new file mode 100644 index 0000000000..15c4eb94d3 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims/BUILD @@ -0,0 +1,24 @@ +scala_library( + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/com/google/inject:guice", + "3rdparty/jvm/com/google/inject/extensions:guice-assistedinject", + "3rdparty/jvm/net/codingwell:scala-guice", + "3rdparty/jvm/org/slf4j:slf4j-api", + "finatra/inject/inject-core/src/main/scala", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/base", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/strato", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/constants", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/common", + "src/thrift/com/twitter/hermit/candidate:hermit-candidate-scala", + "strato/config/columns/onboarding/userrecs:userrecs-strato-client", + "strato/config/columns/recommendations/follow2vec:follow2vec-strato-client", + "strato/config/columns/recommendations/similarity:similarity-strato-client", + "strato/src/main/scala/com/twitter/strato/client", + "util/util-slf4j-api/src/main/scala", + ], +) diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims/CacheBasedSimsStore.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims/CacheBasedSimsStore.scala new file mode 100644 index 0000000000..03894b533b --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims/CacheBasedSimsStore.scala @@ -0,0 +1,50 @@ +package com.twitter.follow_recommendations.common.candidate_sources.sims + +import com.twitter.escherbird.util.stitchcache.StitchCache +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.follow_recommendations.common.models.HasSimilarToContext +import com.twitter.hermit.candidate.thriftscala.Candidates +import com.twitter.product_mixer.core.functional_component.candidate_source.CandidateSource +import com.twitter.product_mixer.core.model.common.identifier.CandidateSourceIdentifier +import com.twitter.stitch.Stitch +import com.twitter.strato.client.Fetcher +import com.twitter.timelines.configapi.HasParams +import com.twitter.util.Duration + +import java.lang.{Long => JLong} + +class CacheBasedSimsStore( + id: CandidateSourceIdentifier, + fetcher: Fetcher[Long, Unit, Candidates], + maxCacheSize: Int, + cacheTtl: Duration, + statsReceiver: StatsReceiver) + extends CandidateSource[HasParams with HasSimilarToContext, CandidateUser] { + + override val identifier: CandidateSourceIdentifier = id + private def getUsersFromSimsSource(userId: JLong): Stitch[Option[Candidates]] = { + fetcher + .fetch(userId) + .map(_.v) + } + + private val simsCache = StitchCache[JLong, Option[Candidates]]( + maxCacheSize = maxCacheSize, + ttl = cacheTtl, + statsReceiver = statsReceiver, + underlyingCall = getUsersFromSimsSource + ) + + override def apply(request: HasParams with HasSimilarToContext): Stitch[Seq[CandidateUser]] = { + Stitch + .traverse(request.similarToUserIds) { userId => + simsCache.readThrough(userId).map { candidatesOpt => + candidatesOpt + .map { candidates => + StratoBasedSimsCandidateSource.map(userId, candidates) + }.getOrElse(Nil) + } + }.map(_.flatten.distinct.map(_.withCandidateSource(identifier))) + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims/DBV2SimsRefreshStore.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims/DBV2SimsRefreshStore.scala new file mode 100644 index 0000000000..3d59e6e0d5 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims/DBV2SimsRefreshStore.scala @@ -0,0 +1,35 @@ +package com.twitter.follow_recommendations.common.candidate_sources.sims + +import com.google.inject.Singleton +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.hermit.model.Algorithm +import com.twitter.product_mixer.core.model.common.identifier.CandidateSourceIdentifier +import com.twitter.strato.generated.client.onboarding.userrecs.NewSimsRefreshOnUserClientColumn +import com.twitter.util.Duration + +import javax.inject.Inject + +@Singleton +class DBV2SimsRefreshStore @Inject() ( + newSimsRefreshOnUserClientColumn: NewSimsRefreshOnUserClientColumn) + extends StratoBasedSimsCandidateSourceWithUnitView( + fetcher = newSimsRefreshOnUserClientColumn.fetcher, + identifier = DBV2SimsRefreshStore.Identifier) + +@Singleton +class CachedDBV2SimsRefreshStore @Inject() ( + newSimsRefreshOnUserClientColumn: NewSimsRefreshOnUserClientColumn, + statsReceiver: StatsReceiver) + extends CacheBasedSimsStore( + id = DBV2SimsRefreshStore.Identifier, + fetcher = newSimsRefreshOnUserClientColumn.fetcher, + maxCacheSize = DBV2SimsRefreshStore.MaxCacheSize, + cacheTtl = DBV2SimsRefreshStore.CacheTTL, + statsReceiver = statsReceiver.scope("CachedDBV2SimsRefreshStore", "cache") + ) + +object DBV2SimsRefreshStore { + val Identifier = CandidateSourceIdentifier(Algorithm.Sims.toString) + val MaxCacheSize = 5000 + val CacheTTL: Duration = Duration.fromHours(24) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims/DBV2SimsStore.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims/DBV2SimsStore.scala new file mode 100644 index 0000000000..ae291eddc8 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims/DBV2SimsStore.scala @@ -0,0 +1,38 @@ +package com.twitter.follow_recommendations.common.candidate_sources.sims + +import com.google.inject.Singleton +import com.google.inject.name.Named +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.follow_recommendations.common.constants.GuiceNamedConstants +import com.twitter.hermit.candidate.thriftscala.Candidates +import com.twitter.hermit.model.Algorithm +import com.twitter.product_mixer.core.model.common.identifier.CandidateSourceIdentifier +import com.twitter.strato.client.Fetcher +import com.twitter.util.Duration + +import javax.inject.Inject + +@Singleton +class DBV2SimsStore @Inject() ( + @Named(GuiceNamedConstants.DBV2_SIMS_FETCHER) fetcher: Fetcher[Long, Unit, Candidates]) + extends StratoBasedSimsCandidateSourceWithUnitView( + fetcher, + identifier = DBV2SimsStore.Identifier) + +@Singleton +class CachedDBV2SimsStore @Inject() ( + @Named(GuiceNamedConstants.DBV2_SIMS_FETCHER) fetcher: Fetcher[Long, Unit, Candidates], + statsReceiver: StatsReceiver) + extends CacheBasedSimsStore( + id = DBV2SimsStore.Identifier, + fetcher = fetcher, + maxCacheSize = DBV2SimsStore.MaxCacheSize, + cacheTtl = DBV2SimsStore.CacheTTL, + statsReceiver = statsReceiver.scope("CachedDBV2SimsStore", "cache") + ) + +object DBV2SimsStore { + val Identifier = CandidateSourceIdentifier(Algorithm.Sims.toString) + val MaxCacheSize = 1000 + val CacheTTL: Duration = Duration.fromHours(24) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims/Follow2vecNearestNeighborsStore.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims/Follow2vecNearestNeighborsStore.scala new file mode 100644 index 0000000000..14131ffd37 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims/Follow2vecNearestNeighborsStore.scala @@ -0,0 +1,69 @@ +package com.twitter.follow_recommendations.common.candidate_sources.sims + +import com.google.inject.Singleton +import com.twitter.follow_recommendations.common.candidate_sources.sims.Follow2vecNearestNeighborsStore.NearestNeighborParamsType +import com.twitter.hermit.candidate.thriftscala.Candidate +import com.twitter.hermit.candidate.thriftscala.Candidates +import com.twitter.hermit.model.Algorithm +import com.twitter.product_mixer.core.model.common.identifier.CandidateSourceIdentifier +import com.twitter.stitch.Stitch +import com.twitter.strato.catalog.Fetch +import com.twitter.strato.client.Fetcher +import com.twitter.strato.generated.client.recommendations.follow2vec.LinearRegressionFollow2vecNearestNeighborsClientColumn +import com.twitter.util.Return +import com.twitter.util.Throw +import javax.inject.Inject + +@Singleton +class LinearRegressionFollow2vecNearestNeighborsStore @Inject() ( + linearRegressionFollow2vecNearestNeighborsClientColumn: LinearRegressionFollow2vecNearestNeighborsClientColumn) + extends StratoBasedSimsCandidateSource[NearestNeighborParamsType]( + Follow2vecNearestNeighborsStore.convertFetcher( + linearRegressionFollow2vecNearestNeighborsClientColumn.fetcher), + view = Follow2vecNearestNeighborsStore.defaultSearchParams, + identifier = Follow2vecNearestNeighborsStore.IdentifierF2vLinearRegression + ) + +object Follow2vecNearestNeighborsStore { + // (userid, feature store version for data) + type NearestNeighborKeyType = (Long, Long) + // (neighbors to be returned, ef value: accuracy / latency tradeoff, distance for filtering) + type NearestNeighborParamsType = (Option[Int], Option[Int], Option[Double]) + // (seq(found neighbor id, score), distance for filtering) + type NearestNeighborValueType = (Seq[(Long, Option[Double])], Option[Double]) + + val IdentifierF2vLinearRegression: CandidateSourceIdentifier = CandidateSourceIdentifier( + Algorithm.LinearRegressionFollow2VecNearestNeighbors.toString) + + val defaultFeatureStoreVersion: Long = 20210708 + val defaultSearchParams: NearestNeighborParamsType = (None, None, None) + + def convertFetcher( + fetcher: Fetcher[NearestNeighborKeyType, NearestNeighborParamsType, NearestNeighborValueType] + ): Fetcher[Long, NearestNeighborParamsType, Candidates] = { + (key: Long, view: NearestNeighborParamsType) => + { + def toCandidates( + results: Option[NearestNeighborValueType] + ): Option[Candidates] = { + results.flatMap { r => + Some( + Candidates( + key, + r._1.map { neighbor => + Candidate(neighbor._1, neighbor._2.getOrElse(0)) + } + ) + ) + } + } + + val results: Stitch[Fetch.Result[NearestNeighborValueType]] = + fetcher.fetch(key = (key, defaultFeatureStoreVersion), view = view) + results.transform { + case Return(r) => Stitch.value(Fetch.Result(toCandidates(r.v))) + case Throw(e) => Stitch.exception(e) + } + } + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims/README.md b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims/README.md new file mode 100644 index 0000000000..97bab500a4 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims/README.md @@ -0,0 +1,32 @@ +# Sims Candidate Source +Offers various online sources for finding similar accounts based on a given user, whether it is the target user or an account candidate. + +## Sims +The objective is to identify a list of K users who are similar to a given user. In this scenario, we primarily focus on finding similar users as "producers" rather than "consumers." Sims has two steps: candidate generation and ranking. + +### Sims Candidate Generation + +With over 700 million users to consider, there are multiple ways to define similarities. Currently, we have three candidate sources for Sims: + +**CosineFollow** (based on user-user follow graph): The similarity between two users is defined as the cosine similarity between their followers. Despite sounding simple, computing all-pair similarity on the entire follow graph is computationally challenging. We are currently using the WHIMP algorithm to find the top 1000 similar users for each user ID. This candidate source has the largest coverage, as it can find similar user candidates for more than 700 million users. + +**CosineList** (based on user-list membership graph): The similarity between two users is defined as the cosine similarity between the lists they are included as members (e.g., [here](https://twitter.com/jack/lists/memberships) are the lists that @jack is on). The same algorithm as CosineFollow is used. + +**Follow2Vec** (essentially Word2Vec on user-user follow graph): We first train the Word2Vec model on follow sequence data to obtain users' embeddings and then find the most similar users based on the similarity of the embeddings. However, we need enough data for each user to learn a meaningful embedding for them, so we can only obtain embeddings for the top 10 million users (currently in production, testing 30 million users). Furthermore, Word2Vec model training is limited by memory and computation as it is trained on a single machine. + +##### Cosine Similarity +A crucial component in Sims is calculating cosine similarities between users based on a user-X (X can be a user, list, or other entities) bipartite graph. This problem is technically challenging and took several years of effort to solve. + +The current implementation uses the algorithm proposed in [When hashes met wedges: A distributed algorithm for finding high similarity vectors. WWW 2017](https://arxiv.org/pdf/1703.01054.pdf) + +### Sims Ranking +After the candidate generation step, we can obtain dozens to hundreds of similar user candidates for each user. However, since these candidates come from different algorithms, we need a way to rank them. To do this, we collect user feedback. + +We use the "Profile Sidebar Impressions & Follow" (a module with follow suggestions displayed when a user visits a profile page and scrolls down) to collect training data. To alleviate any system bias, we use 4% of traffic to show randomly shuffled candidates to users and collect positive (followed impression) and negative (impression only) data from this traffic. This data is used as an evaluation set. We use a portion of the remaining 96% of traffic for training data, filtering only for sets of impressions that had at least one follow, ensuring that the user taking action was paying attention to the impressions. + +The examples are in the format of (profile_user, candidate_user, label). We add features for profile_users and candidate_users based on some high-level aggregated statistics in a feature dataset provided by the Customer Journey team, as well as features that represent the similarity between the profile_user and candidate_user. + +We employ a multi-tower MLP model and optimize the logistic loss. The model is refreshed weekly using an ML workflow. + +We recompute the candidates and rank them daily. The ranked results are published to the Manhattan dataset. + diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims/SimsExperimentalStore.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims/SimsExperimentalStore.scala new file mode 100644 index 0000000000..d144640c7a --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims/SimsExperimentalStore.scala @@ -0,0 +1,36 @@ +package com.twitter.follow_recommendations.common.candidate_sources.sims + +import com.google.inject.Singleton +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.hermit.model.Algorithm +import com.twitter.product_mixer.core.model.common.identifier.CandidateSourceIdentifier +import com.twitter.strato.generated.client.recommendations.similarity.SimilarUsersBySimsExperimentalOnUserClientColumn +import com.twitter.util.Duration + +import javax.inject.Inject + +@Singleton +class SimsExperimentalStore @Inject() ( + simsExperimentalOnUserClientColumn: SimilarUsersBySimsExperimentalOnUserClientColumn) + extends StratoBasedSimsCandidateSourceWithUnitView( + fetcher = simsExperimentalOnUserClientColumn.fetcher, + identifier = SimsExperimentalStore.Identifier + ) + +@Singleton +class CachedSimsExperimentalStore @Inject() ( + simsExperimentalOnUserClientColumn: SimilarUsersBySimsExperimentalOnUserClientColumn, + statsReceiver: StatsReceiver) + extends CacheBasedSimsStore( + id = SimsExperimentalStore.Identifier, + fetcher = simsExperimentalOnUserClientColumn.fetcher, + maxCacheSize = SimsExperimentalStore.MaxCacheSize, + cacheTtl = SimsExperimentalStore.CacheTTL, + statsReceiver = statsReceiver.scope("CachedSimsExperimentalStore", "cache") + ) + +object SimsExperimentalStore { + val Identifier = CandidateSourceIdentifier(Algorithm.Sims.toString) + val MaxCacheSize = 1000 + val CacheTTL: Duration = Duration.fromHours(12) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims/SimsSourceFSConfig.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims/SimsSourceFSConfig.scala new file mode 100644 index 0000000000..0bacb339fe --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims/SimsSourceFSConfig.scala @@ -0,0 +1,14 @@ +package com.twitter.follow_recommendations.common.candidate_sources.sims + +import com.twitter.follow_recommendations.configapi.common.FeatureSwitchConfig +import com.twitter.timelines.configapi.FSName +import com.twitter.timelines.configapi.FSParam +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class SimsSourceFSConfig @Inject() () extends FeatureSwitchConfig { + override val booleanFSParams: Seq[FSParam[Boolean] with FSName] = Seq( + SimsSourceParams.DisableHeavyRanker + ) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims/SimsSourceParams.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims/SimsSourceParams.scala new file mode 100644 index 0000000000..e96775a6a9 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims/SimsSourceParams.scala @@ -0,0 +1,16 @@ +package com.twitter.follow_recommendations.common.candidate_sources.sims + +import com.twitter.timelines.configapi.FSParam + +object SimsSourceParams { + case object EnableDBV2SimsStore extends FSParam[Boolean]("sims_source_enable_dbv2_source", false) + + case object EnableDBV2SimsRefreshStore + extends FSParam[Boolean]("sims_source_enable_dbv2_refresh_source", false) + + case object EnableExperimentalSimsStore + extends FSParam[Boolean]("sims_source_enable_experimental_source", false) + + case object DisableHeavyRanker + extends FSParam[Boolean]("sims_source_disable_heavy_ranker", default = false) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims/SimsStore.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims/SimsStore.scala new file mode 100644 index 0000000000..98a00a9a48 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims/SimsStore.scala @@ -0,0 +1,36 @@ +package com.twitter.follow_recommendations.common.candidate_sources.sims + +import com.google.inject.Singleton +import com.google.inject.name.Named +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.follow_recommendations.common.constants.GuiceNamedConstants +import com.twitter.hermit.candidate.thriftscala.Candidates +import com.twitter.hermit.model.Algorithm +import com.twitter.product_mixer.core.model.common.identifier.CandidateSourceIdentifier +import com.twitter.strato.client.Fetcher +import com.twitter.util.Duration + +import javax.inject.Inject + +@Singleton +class SimsStore @Inject() ( + @Named(GuiceNamedConstants.SIMS_FETCHER) fetcher: Fetcher[Long, Unit, Candidates]) + extends StratoBasedSimsCandidateSourceWithUnitView(fetcher, identifier = SimsStore.Identifier) + +@Singleton +class CachedSimsStore @Inject() ( + @Named(GuiceNamedConstants.SIMS_FETCHER) fetcher: Fetcher[Long, Unit, Candidates], + statsReceiver: StatsReceiver) + extends CacheBasedSimsStore( + id = SimsStore.Identifier, + fetcher = fetcher, + maxCacheSize = SimsStore.MaxCacheSize, + cacheTtl = SimsStore.CacheTTL, + statsReceiver = statsReceiver.scope("CachedSimsStore", "cache") + ) + +object SimsStore { + val Identifier = CandidateSourceIdentifier(Algorithm.Sims.toString) + val MaxCacheSize = 50000 + val CacheTTL: Duration = Duration.fromHours(24) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims/StratoBasedSimsCandidateSource.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims/StratoBasedSimsCandidateSource.scala new file mode 100644 index 0000000000..6d862c8491 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims/StratoBasedSimsCandidateSource.scala @@ -0,0 +1,40 @@ +package com.twitter.follow_recommendations.common.candidate_sources.sims + +import com.twitter.follow_recommendations.common.candidate_sources.base.StratoFetcherSource +import com.twitter.follow_recommendations.common.models.AccountProof +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.follow_recommendations.common.models.Reason +import com.twitter.follow_recommendations.common.models.SimilarToProof +import com.twitter.hermit.candidate.thriftscala.Candidates +import com.twitter.product_mixer.core.model.common.identifier.CandidateSourceIdentifier +import com.twitter.strato.client.Fetcher + +abstract class StratoBasedSimsCandidateSource[U]( + fetcher: Fetcher[Long, U, Candidates], + view: U, + override val identifier: CandidateSourceIdentifier) + extends StratoFetcherSource[Long, U, Candidates](fetcher, view, identifier) { + + override def map(target: Long, candidates: Candidates): Seq[CandidateUser] = + StratoBasedSimsCandidateSource.map(target, candidates) +} + +object StratoBasedSimsCandidateSource { + def map(target: Long, candidates: Candidates): Seq[CandidateUser] = { + for { + candidate <- candidates.candidates + } yield CandidateUser( + id = candidate.userId, + score = Some(candidate.score), + reason = Some( + Reason( + Some( + AccountProof( + similarToProof = Some(SimilarToProof(Seq(target))) + ) + ) + ) + ) + ) + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims/StratoBasedSimsCandidateSourceWithUnitView.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims/StratoBasedSimsCandidateSourceWithUnitView.scala new file mode 100644 index 0000000000..af11338930 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims/StratoBasedSimsCandidateSourceWithUnitView.scala @@ -0,0 +1,10 @@ +package com.twitter.follow_recommendations.common.candidate_sources.sims + +import com.twitter.hermit.candidate.thriftscala.Candidates +import com.twitter.product_mixer.core.model.common.identifier.CandidateSourceIdentifier +import com.twitter.strato.client.Fetcher + +abstract class StratoBasedSimsCandidateSourceWithUnitView( + fetcher: Fetcher[Long, Unit, Candidates], + override val identifier: CandidateSourceIdentifier) + extends StratoBasedSimsCandidateSource[Unit](fetcher, Unit, identifier) diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims/SwitchingSimsSource.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims/SwitchingSimsSource.scala new file mode 100644 index 0000000000..0d297a806c --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims/SwitchingSimsSource.scala @@ -0,0 +1,55 @@ +package com.twitter.follow_recommendations.common.candidate_sources.sims + +import com.twitter.finagle.stats.NullStatsReceiver +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.follow_recommendations.common.models.HasSimilarToContext +import com.twitter.hermit.model.Algorithm +import com.twitter.product_mixer.core.functional_component.candidate_source.CandidateSource +import com.twitter.product_mixer.core.model.common.identifier.CandidateSourceIdentifier +import com.twitter.stitch.Stitch +import com.twitter.timelines.configapi.HasParams + +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class SwitchingSimsSource @Inject() ( + cachedDBV2SimsStore: CachedDBV2SimsStore, + cachedDBV2SimsRefreshStore: CachedDBV2SimsRefreshStore, + cachedSimsExperimentalStore: CachedSimsExperimentalStore, + cachedSimsStore: CachedSimsStore, + statsReceiver: StatsReceiver = NullStatsReceiver) + extends CandidateSource[HasParams with HasSimilarToContext, CandidateUser] { + + override val identifier: CandidateSourceIdentifier = SwitchingSimsSource.Identifier + + private val stats = statsReceiver.scope("SwitchingSimsSource") + private val dbV2SimsStoreCounter = stats.counter("DBV2SimsStore") + private val dbV2SimsRefreshStoreCounter = stats.counter("DBV2SimsRefreshStore") + private val simsExperimentalStoreCounter = stats.counter("SimsExperimentalStore") + private val simsStoreCounter = stats.counter("SimsStore") + + override def apply(request: HasParams with HasSimilarToContext): Stitch[Seq[CandidateUser]] = { + val selectedSimsStore = + if (request.params(SimsSourceParams.EnableDBV2SimsStore)) { + dbV2SimsStoreCounter.incr() + cachedDBV2SimsStore + } else if (request.params(SimsSourceParams.EnableDBV2SimsRefreshStore)) { + dbV2SimsRefreshStoreCounter.incr() + cachedDBV2SimsRefreshStore + } else if (request.params(SimsSourceParams.EnableExperimentalSimsStore)) { + simsExperimentalStoreCounter.incr() + cachedSimsExperimentalStore + } else { + simsStoreCounter.incr() + cachedSimsStore + } + stats.counter("total").incr() + selectedSimsStore(request) + } +} + +object SwitchingSimsSource { + val Identifier: CandidateSourceIdentifier = CandidateSourceIdentifier(Algorithm.Sims.toString) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims_expansion/BUILD b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims_expansion/BUILD new file mode 100644 index 0000000000..f5ccb66e7b --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims_expansion/BUILD @@ -0,0 +1,23 @@ +scala_library( + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/com/google/inject:guice", + "3rdparty/jvm/com/google/inject/extensions:guice-assistedinject", + "3rdparty/jvm/net/codingwell:scala-guice", + "3rdparty/jvm/org/slf4j:slf4j-api", + "finatra/inject/inject-core/src/main/scala", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/base", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/real_time_real_graph", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/socialgraph", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/strato", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/constants", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/common", + "strato/src/main/scala/com/twitter/strato/client", + "util/util-slf4j-api/src/main/scala", + ], +) diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims_expansion/DBV2SimsExpansionParams.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims_expansion/DBV2SimsExpansionParams.scala new file mode 100644 index 0000000000..c323ad1f3a --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims_expansion/DBV2SimsExpansionParams.scala @@ -0,0 +1,22 @@ +package com.twitter.follow_recommendations.common.candidate_sources.sims_expansion + +import com.twitter.timelines.configapi.FSBoundedParam +import com.twitter.timelines.configapi.FSParam + +object DBV2SimsExpansionParams { + // Theses divisors are used to calibrate DBv2Sims extension candidates scores + case object RecentFollowingSimilarUsersDBV2CalibrateDivisor + extends FSBoundedParam[Double]( + "sims_expansion_recent_following_similar_users_dbv2_divisor", + default = 1.0d, + min = 0.1d, + max = 100d) + case object RecentEngagementSimilarUsersDBV2CalibrateDivisor + extends FSBoundedParam[Double]( + "sims_expansion_recent_engagement_similar_users_dbv2_divisor", + default = 1.0d, + min = 0.1d, + max = 100d) + case object DisableHeavyRanker + extends FSParam[Boolean]("sims_expansion_disable_heavy_ranker", default = false) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims_expansion/README.md b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims_expansion/README.md new file mode 100644 index 0000000000..6143d58681 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims_expansion/README.md @@ -0,0 +1,6 @@ +# Sims Expansion Candidate Source +provides similar accounts based on the Sims algorithm for a given set of accounts. + +This is a 2nd-hop expansion, meaning that the input accounts could be a user's recently engaged, followed, or algorithm-generated (such as RealGraph) accounts. + +For more information on Sims and how it is utilized in the Follow Recommendations Service, please refer to the `follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims/README.md` file. diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims_expansion/RecentEngagementSimilarUsersFSConfig.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims_expansion/RecentEngagementSimilarUsersFSConfig.scala new file mode 100644 index 0000000000..5642f58527 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims_expansion/RecentEngagementSimilarUsersFSConfig.scala @@ -0,0 +1,14 @@ +package com.twitter.follow_recommendations.common.candidate_sources.sims_expansion + +import com.twitter.follow_recommendations.configapi.common.FeatureSwitchConfig +import com.twitter.timelines.configapi.FSParam + +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class RecentEngagementSimilarUsersFSConfig @Inject() () extends FeatureSwitchConfig { + override val booleanFSParams: Seq[FSParam[Boolean]] = Seq( + RecentEngagementSimilarUsersParams.FirstDegreeSortEnabled + ) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims_expansion/RecentEngagementSimilarUsersParams.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims_expansion/RecentEngagementSimilarUsersParams.scala new file mode 100644 index 0000000000..4b17297021 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims_expansion/RecentEngagementSimilarUsersParams.scala @@ -0,0 +1,17 @@ +package com.twitter.follow_recommendations.common.candidate_sources.sims_expansion + +import com.twitter.timelines.configapi.FSEnumParam +import com.twitter.timelines.configapi.FSParam + +object RecentEngagementSimilarUsersParams { + + case object FirstDegreeSortEnabled + extends FSParam[Boolean]( + name = "sims_expansion_recent_engagement_first_degree_sort", + default = true) + case object Aggregator + extends FSEnumParam[SimsExpansionSourceAggregatorId.type]( + name = "sims_expansion_recent_engagement_aggregator_id", + default = SimsExpansionSourceAggregatorId.Sum, + enum = SimsExpansionSourceAggregatorId) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims_expansion/RecentEngagementSimilarUsersSource.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims_expansion/RecentEngagementSimilarUsersSource.scala new file mode 100644 index 0000000000..0d99c2dc2d --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims_expansion/RecentEngagementSimilarUsersSource.scala @@ -0,0 +1,113 @@ +package com.twitter.follow_recommendations.common.candidate_sources.sims_expansion + +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.follow_recommendations.common.candidate_sources.sims.SwitchingSimsSource +import com.twitter.follow_recommendations.common.clients.real_time_real_graph.RealTimeRealGraphClient +import com.twitter.follow_recommendations.common.models.AccountProof +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.follow_recommendations.common.models.Reason +import com.twitter.follow_recommendations.common.models.SimilarToProof +import com.twitter.hermit.model.Algorithm +import com.twitter.product_mixer.core.model.common.identifier.CandidateSourceIdentifier +import com.twitter.product_mixer.core.model.marshalling.request.HasClientContext +import com.twitter.stitch.Stitch +import com.twitter.timelines.configapi.HasParams + +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class RecentEngagementSimilarUsersSource @Inject() ( + realTimeRealGraphClient: RealTimeRealGraphClient, + switchingSimsSource: SwitchingSimsSource, + statsReceiver: StatsReceiver) + extends SimsExpansionBasedCandidateSource[HasClientContext with HasParams]( + switchingSimsSource) { + override def maxSecondaryDegreeNodes(req: HasClientContext with HasParams): Int = Int.MaxValue + + override def maxResults(req: HasClientContext with HasParams): Int = + RecentEngagementSimilarUsersSource.MaxResults + + override val identifier: CandidateSourceIdentifier = RecentEngagementSimilarUsersSource.Identifier + private val stats = statsReceiver.scope(identifier.name) + private val calibratedScoreCounter = stats.counter("calibrated_scores_counter") + + override def scoreCandidate(sourceScore: Double, similarToScore: Double): Double = { + sourceScore * similarToScore + } + + override def calibrateDivisor(req: HasClientContext with HasParams): Double = { + req.params(DBV2SimsExpansionParams.RecentEngagementSimilarUsersDBV2CalibrateDivisor) + } + + override def calibrateScore( + candidateScore: Double, + req: HasClientContext with HasParams + ): Double = { + calibratedScoreCounter.incr() + candidateScore / calibrateDivisor(req) + } + + /** + * fetch first degree nodes given request + */ + override def firstDegreeNodes( + target: HasClientContext with HasParams + ): Stitch[Seq[CandidateUser]] = { + target.getOptionalUserId + .map { userId => + realTimeRealGraphClient + .getUsersRecentlyEngagedWith( + userId, + RealTimeRealGraphClient.EngagementScoreMap, + includeDirectFollowCandidates = true, + includeNonDirectFollowCandidates = true + ).map(_.sortBy(-_.score.getOrElse(0.0d)) + .take(RecentEngagementSimilarUsersSource.MaxFirstDegreeNodes)) + }.getOrElse(Stitch.Nil) + } + + override def aggregateAndScore( + request: HasClientContext with HasParams, + firstDegreeToSecondDegreeNodesMap: Map[CandidateUser, Seq[SimilarUser]] + ): Stitch[Seq[CandidateUser]] = { + + val inputNodes = firstDegreeToSecondDegreeNodesMap.keys.map(_.id).toSet + val aggregator = request.params(RecentEngagementSimilarUsersParams.Aggregator) match { + case SimsExpansionSourceAggregatorId.Max => + SimsExpansionBasedCandidateSource.ScoreAggregator.Max + case SimsExpansionSourceAggregatorId.Sum => + SimsExpansionBasedCandidateSource.ScoreAggregator.Sum + case SimsExpansionSourceAggregatorId.MultiDecay => + SimsExpansionBasedCandidateSource.ScoreAggregator.MultiDecay + } + + val groupedCandidates = firstDegreeToSecondDegreeNodesMap.values.flatten + .filterNot(c => inputNodes.contains(c.candidateId)) + .groupBy(_.candidateId) + .map { + case (id, candidates) => + // Different aggregators for final score + val finalScore = aggregator(candidates.map(_.score).toSeq) + val proofs = candidates.map(_.similarTo).toSet + + CandidateUser( + id = id, + score = Some(finalScore), + reason = + Some(Reason(Some(AccountProof(similarToProof = Some(SimilarToProof(proofs.toSeq)))))) + ).withCandidateSource(identifier) + } + .toSeq + .sortBy(-_.score.getOrElse(0.0d)) + .take(maxResults(request)) + + Stitch.value(groupedCandidates) + } +} + +object RecentEngagementSimilarUsersSource { + val Identifier = CandidateSourceIdentifier(Algorithm.RecentEngagementSimilarUser.toString) + val MaxFirstDegreeNodes = 10 + val MaxResults = 200 +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims_expansion/RecentFollowingSimilarUsersParams.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims_expansion/RecentFollowingSimilarUsersParams.scala new file mode 100644 index 0000000000..44b1378d45 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims_expansion/RecentFollowingSimilarUsersParams.scala @@ -0,0 +1,29 @@ +package com.twitter.follow_recommendations.common.candidate_sources.sims_expansion + +import com.twitter.timelines.configapi.FSBoundedParam +import com.twitter.timelines.configapi.FSParam + +object RecentFollowingSimilarUsersParams { + case object MaxFirstDegreeNodes + extends FSBoundedParam[Int]( + name = "sims_expansion_recent_following_max_first_degree_nodes", + default = 10, + min = 0, + max = 200) + case object MaxSecondaryDegreeExpansionPerNode + extends FSBoundedParam[Int]( + name = "sims_expansion_recent_following_max_secondary_degree_nodes", + default = 40, + min = 0, + max = 200) + case object MaxResults + extends FSBoundedParam[Int]( + name = "sims_expansion_recent_following_max_results", + default = 200, + min = 0, + max = 200) + case object TimestampIntegrated + extends FSParam[Boolean]( + name = "sims_expansion_recent_following_integ_timestamp", + default = false) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims_expansion/RecentFollowingSimilarUsersSource.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims_expansion/RecentFollowingSimilarUsersSource.scala new file mode 100644 index 0000000000..b5a187fc5e --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims_expansion/RecentFollowingSimilarUsersSource.scala @@ -0,0 +1,99 @@ +package com.twitter.follow_recommendations.common.candidate_sources.sims_expansion + +import com.google.inject.Singleton +import com.twitter.follow_recommendations.common.candidate_sources.sims.SwitchingSimsSource +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.follow_recommendations.common.models.HasRecentFollowedUserIds +import com.twitter.hermit.model.Algorithm +import com.twitter.product_mixer.core.model.common.identifier.CandidateSourceIdentifier +import com.twitter.stitch.Stitch +import com.twitter.timelines.configapi.HasParams +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.follow_recommendations.common.clients.socialgraph.SocialGraphClient +import com.twitter.product_mixer.core.model.marshalling.request.HasClientContext +import javax.inject.Inject + +object RecentFollowingSimilarUsersSource { + + val Identifier = CandidateSourceIdentifier(Algorithm.NewFollowingSimilarUser.toString) +} + +@Singleton +class RecentFollowingSimilarUsersSource @Inject() ( + socialGraph: SocialGraphClient, + switchingSimsSource: SwitchingSimsSource, + statsReceiver: StatsReceiver) + extends SimsExpansionBasedCandidateSource[ + HasParams with HasRecentFollowedUserIds with HasClientContext + ](switchingSimsSource) { + + val identifier = RecentFollowingSimilarUsersSource.Identifier + private val stats = statsReceiver.scope(identifier.name) + private val maxResultsStats = stats.scope("max_results") + private val calibratedScoreCounter = stats.counter("calibrated_scores_counter") + + override def firstDegreeNodes( + request: HasParams with HasRecentFollowedUserIds with HasClientContext + ): Stitch[Seq[CandidateUser]] = { + if (request.params(RecentFollowingSimilarUsersParams.TimestampIntegrated)) { + val recentFollowedUserIdsWithTimeStitch = + socialGraph.getRecentFollowedUserIdsWithTime(request.clientContext.userId.get) + + recentFollowedUserIdsWithTimeStitch.map { results => + val first_degree_nodes = results + .sortBy(-_.timeInMs).take( + request.params(RecentFollowingSimilarUsersParams.MaxFirstDegreeNodes)) + val max_timestamp = first_degree_nodes.head.timeInMs + first_degree_nodes.map { + case userIdWithTime => + CandidateUser( + userIdWithTime.userId, + score = Some(userIdWithTime.timeInMs.toDouble / max_timestamp)) + } + } + } else { + Stitch.value( + request.recentFollowedUserIds + .getOrElse(Nil).take( + request.params(RecentFollowingSimilarUsersParams.MaxFirstDegreeNodes)).map( + CandidateUser(_, score = Some(1.0))) + ) + } + } + + override def maxSecondaryDegreeNodes( + req: HasParams with HasRecentFollowedUserIds with HasClientContext + ): Int = { + req.params(RecentFollowingSimilarUsersParams.MaxSecondaryDegreeExpansionPerNode) + } + + override def maxResults( + req: HasParams with HasRecentFollowedUserIds with HasClientContext + ): Int = { + val firstDegreeNodes = req.params(RecentFollowingSimilarUsersParams.MaxFirstDegreeNodes) + val maxResultsNum = req.params(RecentFollowingSimilarUsersParams.MaxResults) + maxResultsStats + .stat( + s"RecentFollowingSimilarUsersSource_firstDegreeNodes_${firstDegreeNodes}_maxResults_${maxResultsNum}") + .add(1) + maxResultsNum + } + + override def scoreCandidate(sourceScore: Double, similarToScore: Double): Double = { + sourceScore * similarToScore + } + + override def calibrateDivisor( + req: HasParams with HasRecentFollowedUserIds with HasClientContext + ): Double = { + req.params(DBV2SimsExpansionParams.RecentFollowingSimilarUsersDBV2CalibrateDivisor) + } + + override def calibrateScore( + candidateScore: Double, + req: HasParams with HasRecentFollowedUserIds with HasClientContext + ): Double = { + calibratedScoreCounter.incr() + candidateScore / calibrateDivisor(req) + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims_expansion/RecentStrongEngagementDirectFollowSimilarUsersSource.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims_expansion/RecentStrongEngagementDirectFollowSimilarUsersSource.scala new file mode 100644 index 0000000000..0898aabfb3 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims_expansion/RecentStrongEngagementDirectFollowSimilarUsersSource.scala @@ -0,0 +1,53 @@ +package com.twitter.follow_recommendations.common.candidate_sources.sims_expansion + +import com.google.inject.Singleton +import com.twitter.follow_recommendations.common.candidate_sources.sims.SwitchingSimsSource +import com.twitter.follow_recommendations.common.clients.real_time_real_graph.RealTimeRealGraphClient +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.hermit.model.Algorithm +import com.twitter.product_mixer.core.model.common.identifier.CandidateSourceIdentifier +import com.twitter.product_mixer.core.model.marshalling.request.HasClientContext +import com.twitter.stitch.Stitch +import com.twitter.timelines.configapi.HasParams + +import javax.inject.Inject + +@Singleton +class RecentStrongEngagementDirectFollowSimilarUsersSource @Inject() ( + realTimeRealGraphClient: RealTimeRealGraphClient, + switchingSimsSource: SwitchingSimsSource) + extends SimsExpansionBasedCandidateSource[HasClientContext with HasParams]( + switchingSimsSource) { + + val identifier = RecentStrongEngagementDirectFollowSimilarUsersSource.Identifier + + override def firstDegreeNodes( + request: HasClientContext with HasParams + ): Stitch[Seq[CandidateUser]] = request.getOptionalUserId + .map { userId => + realTimeRealGraphClient + .getUsersRecentlyEngagedWith( + userId, + RealTimeRealGraphClient.StrongEngagementScoreMap, + includeDirectFollowCandidates = true, + includeNonDirectFollowCandidates = false + ).map(_.take(RecentStrongEngagementDirectFollowSimilarUsersSource.MaxFirstDegreeNodes)) + }.getOrElse(Stitch.Nil) + + override def maxSecondaryDegreeNodes(request: HasClientContext with HasParams): Int = Int.MaxValue + + override def maxResults(request: HasClientContext with HasParams): Int = + RecentStrongEngagementDirectFollowSimilarUsersSource.MaxResults + + override def scoreCandidate(sourceScore: Double, similarToScore: Double): Double = { + sourceScore * similarToScore + } + + override def calibrateDivisor(req: HasClientContext with HasParams): Double = 1.0d +} + +object RecentStrongEngagementDirectFollowSimilarUsersSource { + val Identifier = CandidateSourceIdentifier(Algorithm.RecentStrongEngagementSimilarUser.toString) + val MaxFirstDegreeNodes = 10 + val MaxResults = 200 +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims_expansion/SimsExpansionBasedCandidateSource.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims_expansion/SimsExpansionBasedCandidateSource.scala new file mode 100644 index 0000000000..018fe413bb --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims_expansion/SimsExpansionBasedCandidateSource.scala @@ -0,0 +1,114 @@ +package com.twitter.follow_recommendations.common.candidate_sources.sims_expansion + +import com.twitter.follow_recommendations.common.candidate_sources.base.TwoHopExpansionCandidateSource +import com.twitter.follow_recommendations.common.candidate_sources.sims.SwitchingSimsSource +import com.twitter.follow_recommendations.common.models.AccountProof +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.follow_recommendations.common.models.HasSimilarToContext +import com.twitter.follow_recommendations.common.models.Reason +import com.twitter.follow_recommendations.common.models.SimilarToProof +import com.twitter.stitch.Stitch +import com.twitter.timelines.configapi.HasParams +import scala.math._ + +case class SimilarUser(candidateId: Long, similarTo: Long, score: Double) + +abstract class SimsExpansionBasedCandidateSource[-Target <: HasParams]( + switchingSimsSource: SwitchingSimsSource) + extends TwoHopExpansionCandidateSource[Target, CandidateUser, SimilarUser, CandidateUser] { + + // max number secondary degree nodes per first degree node + def maxSecondaryDegreeNodes(req: Target): Int + + // max number output results + def maxResults(req: Target): Int + + // scorer to score candidate based on first and second degree node scores + def scoreCandidate(source: Double, similarToScore: Double): Double + + def calibrateDivisor(req: Target): Double + + def calibrateScore(candidateScore: Double, req: Target): Double = { + candidateScore / calibrateDivisor(req) + } + + override def secondaryDegreeNodes(req: Target, node: CandidateUser): Stitch[Seq[SimilarUser]] = { + switchingSimsSource(new HasParams with HasSimilarToContext { + override val similarToUserIds = Seq(node.id) + override val params = (req.params) + }).map(_.take(maxSecondaryDegreeNodes(req)).map { candidate => + SimilarUser( + candidate.id, + node.id, + (node.score, candidate.score) match { + // only calibrated sims expanded candidates scores + case (Some(nodeScore), Some(candidateScore)) => + calibrateScore(scoreCandidate(nodeScore, candidateScore), req) + case (Some(nodeScore), _) => nodeScore + // NewFollowingSimilarUser will enter this case + case _ => calibrateScore(candidate.score.getOrElse(0.0), req) + } + ) + }) + } + + override def aggregateAndScore( + request: Target, + firstDegreeToSecondDegreeNodesMap: Map[CandidateUser, Seq[SimilarUser]] + ): Stitch[Seq[CandidateUser]] = { + + val inputNodes = firstDegreeToSecondDegreeNodesMap.keys.map(_.id).toSet + val aggregator = request.params(SimsExpansionSourceParams.Aggregator) match { + case SimsExpansionSourceAggregatorId.Max => + SimsExpansionBasedCandidateSource.ScoreAggregator.Max + case SimsExpansionSourceAggregatorId.Sum => + SimsExpansionBasedCandidateSource.ScoreAggregator.Sum + case SimsExpansionSourceAggregatorId.MultiDecay => + SimsExpansionBasedCandidateSource.ScoreAggregator.MultiDecay + } + + val groupedCandidates = firstDegreeToSecondDegreeNodesMap.values.flatten + .filterNot(c => inputNodes.contains(c.candidateId)) + .groupBy(_.candidateId) + .map { + case (id, candidates) => + // Different aggregators for final score + val finalScore = aggregator(candidates.map(_.score).toSeq) + val proofs = candidates.map(_.similarTo).toSet + + CandidateUser( + id = id, + score = Some(finalScore), + reason = + Some(Reason(Some(AccountProof(similarToProof = Some(SimilarToProof(proofs.toSeq)))))) + ).withCandidateSource(identifier) + } + .toSeq + .sortBy(-_.score.getOrElse(0.0d)) + .take(maxResults(request)) + + Stitch.value(groupedCandidates) + } +} + +object SimsExpansionBasedCandidateSource { + object ScoreAggregator { + val Max: Seq[Double] => Double = (candidateScores: Seq[Double]) => { + if (candidateScores.size > 0) candidateScores.max else 0.0 + } + val Sum: Seq[Double] => Double = (candidateScores: Seq[Double]) => { + candidateScores.sum + } + val MultiDecay: Seq[Double] => Double = (candidateScores: Seq[Double]) => { + val alpha = 0.1 + val beta = 0.1 + val gamma = 0.8 + val decay_scores: Seq[Double] = + candidateScores + .sorted(Ordering[Double].reverse) + .zipWithIndex + .map(x => x._1 * pow(gamma, x._2)) + alpha * candidateScores.max + decay_scores.sum + beta * candidateScores.size + } + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims_expansion/SimsExpansionFSConfig.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims_expansion/SimsExpansionFSConfig.scala new file mode 100644 index 0000000000..b145ee607d --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims_expansion/SimsExpansionFSConfig.scala @@ -0,0 +1,26 @@ +package com.twitter.follow_recommendations.common.candidate_sources.sims_expansion + +import com.twitter.follow_recommendations.configapi.common.FeatureSwitchConfig +import com.twitter.timelines.configapi.FSBoundedParam +import com.twitter.timelines.configapi.FSParam +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class SimsExpansionFSConfig @Inject() () extends FeatureSwitchConfig { + override val intFSParams: Seq[FSBoundedParam[Int]] = Seq( + RecentFollowingSimilarUsersParams.MaxFirstDegreeNodes, + RecentFollowingSimilarUsersParams.MaxSecondaryDegreeExpansionPerNode, + RecentFollowingSimilarUsersParams.MaxResults + ) + + override val doubleFSParams: Seq[FSBoundedParam[Double]] = Seq( + DBV2SimsExpansionParams.RecentFollowingSimilarUsersDBV2CalibrateDivisor, + DBV2SimsExpansionParams.RecentEngagementSimilarUsersDBV2CalibrateDivisor + ) + + override val booleanFSParams: Seq[FSParam[Boolean]] = Seq( + DBV2SimsExpansionParams.DisableHeavyRanker, + RecentFollowingSimilarUsersParams.TimestampIntegrated + ) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims_expansion/SimsExpansionSourceParams.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims_expansion/SimsExpansionSourceParams.scala new file mode 100644 index 0000000000..f03ccceea2 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims_expansion/SimsExpansionSourceParams.scala @@ -0,0 +1,17 @@ +package com.twitter.follow_recommendations.common.candidate_sources.sims_expansion +import com.twitter.timelines.configapi.FSEnumParam + +object SimsExpansionSourceParams { + case object Aggregator + extends FSEnumParam[SimsExpansionSourceAggregatorId.type]( + name = "sims_expansion_aggregator_id", + default = SimsExpansionSourceAggregatorId.Sum, + enum = SimsExpansionSourceAggregatorId) +} + +object SimsExpansionSourceAggregatorId extends Enumeration { + type AggregatorId = Value + val Sum: AggregatorId = Value("sum") + val Max: AggregatorId = Value("max") + val MultiDecay: AggregatorId = Value("multi_decay") +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/socialgraph/BUILD b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/socialgraph/BUILD new file mode 100644 index 0000000000..e0392df453 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/socialgraph/BUILD @@ -0,0 +1,18 @@ +scala_library( + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/com/google/inject:guice", + "3rdparty/jvm/com/google/inject/extensions:guice-assistedinject", + "3rdparty/jvm/net/codingwell:scala-guice", + "3rdparty/jvm/org/slf4j:slf4j-api", + "finatra/inject/inject-core/src/main/scala", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/base", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/socialgraph", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/constants", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models", + "util/util-slf4j-api/src/main/scala", + ], +) diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/socialgraph/README.md b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/socialgraph/README.md new file mode 100644 index 0000000000..bfdbacd259 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/socialgraph/README.md @@ -0,0 +1,6 @@ +# Social Graph Candidate Source +Provides candidate expansion based on the Twitter social graph. + +Currently, the expansion is mainly based on the "follow" social graph edge, which allows the service to identify recent following accounts for a given set of accounts. The input accounts could be a user's recent following, engaged, or other related accounts. + +In summary, the Social Graph Candidate Source utilizes the Twitter social graph to provide candidate expansion, primarily focusing on recent following accounts. diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/socialgraph/RecentFollowingRecentFollowingExpansionSource.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/socialgraph/RecentFollowingRecentFollowingExpansionSource.scala new file mode 100644 index 0000000000..577213cc18 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/socialgraph/RecentFollowingRecentFollowingExpansionSource.scala @@ -0,0 +1,102 @@ +package com.twitter.follow_recommendations.common.candidate_sources.socialgraph + +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.follow_recommendations.common.candidate_sources.base.TwoHopExpansionCandidateSource +import com.twitter.follow_recommendations.common.clients.socialgraph.RecentEdgesQuery +import com.twitter.follow_recommendations.common.clients.socialgraph.SocialGraphClient +import com.twitter.follow_recommendations.common.models.AccountProof +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.follow_recommendations.common.models.FollowProof +import com.twitter.follow_recommendations.common.models.HasRecentFollowedUserIds +import com.twitter.follow_recommendations.common.models.Reason +import com.twitter.hermit.model.Algorithm +import com.twitter.inject.Logging +import com.twitter.product_mixer.core.model.common.identifier.CandidateSourceIdentifier +import com.twitter.socialgraph.thriftscala.RelationshipType +import com.twitter.stitch.Stitch +import com.twitter.timelines.configapi.HasParams +import javax.inject.Inject +import javax.inject.Singleton + +/** + * This candidate source is a two hop expansion over the follow graph. The candidates returned from this source is the users that get followed by the target user's recent followings. It will call SocialGraph `n` + 1 times where `n` is the number of recent followings of the target user to be considered. + */ +@Singleton +class RecentFollowingRecentFollowingExpansionSource @Inject() ( + socialGraphClient: SocialGraphClient, + statsReceiver: StatsReceiver) + extends TwoHopExpansionCandidateSource[ + HasParams with HasRecentFollowedUserIds, + Long, + Long, + CandidateUser + ] + with Logging { + + override val identifier: CandidateSourceIdentifier = + RecentFollowingRecentFollowingExpansionSource.Identifier + + val stats = statsReceiver.scope(identifier.name) + + override def firstDegreeNodes( + target: HasParams with HasRecentFollowedUserIds + ): Stitch[Seq[Long]] = Stitch.value( + target.recentFollowedUserIds + .getOrElse(Nil).take( + RecentFollowingRecentFollowingExpansionSource.NumFirstDegreeNodesToRetrieve) + ) + + override def secondaryDegreeNodes( + target: HasParams with HasRecentFollowedUserIds, + node: Long + ): Stitch[Seq[Long]] = socialGraphClient + .getRecentEdgesCached( + RecentEdgesQuery( + node, + Seq(RelationshipType.Following), + Some(RecentFollowingRecentFollowingExpansionSource.NumSecondDegreeNodesToRetrieve)), + useCachedStratoColumn = + target.params(RecentFollowingRecentFollowingExpansionSourceParams.CallSgsCachedColumn) + ).map( + _.take(RecentFollowingRecentFollowingExpansionSource.NumSecondDegreeNodesToRetrieve)).rescue { + case exception: Exception => + logger.warn( + s"${this.getClass} fails to retrieve second degree nodes for first degree node $node", + exception) + stats.counter("second_degree_expansion_error").incr() + Stitch.Nil + } + + override def aggregateAndScore( + target: HasParams with HasRecentFollowedUserIds, + firstDegreeToSecondDegreeNodesMap: Map[Long, Seq[Long]] + ): Stitch[Seq[CandidateUser]] = { + val zipped = firstDegreeToSecondDegreeNodesMap.toSeq.flatMap { + case (firstDegreeId, secondDegreeIds) => + secondDegreeIds.map(secondDegreeId => firstDegreeId -> secondDegreeId) + } + val candidateAndConnections = zipped + .groupBy { case (_, secondDegreeId) => secondDegreeId } + .mapValues { v => v.map { case (firstDegreeId, _) => firstDegreeId } } + .toSeq + .sortBy { case (_, connections) => -connections.size } + .map { + case (candidateId, connections) => + CandidateUser( + id = candidateId, + score = Some(CandidateUser.DefaultCandidateScore), + reason = Some( + Reason( + Some(AccountProof(followProof = Some(FollowProof(connections, connections.size)))))) + ).withCandidateSource(identifier) + } + Stitch.value(candidateAndConnections) + } +} + +object RecentFollowingRecentFollowingExpansionSource { + val Identifier = CandidateSourceIdentifier(Algorithm.NewFollowingNewFollowingExpansion.toString) + + val NumFirstDegreeNodesToRetrieve = 5 + val NumSecondDegreeNodesToRetrieve = 20 +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/socialgraph/RecentFollowingRecentFollowingExpansionSourceFSConfig.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/socialgraph/RecentFollowingRecentFollowingExpansionSourceFSConfig.scala new file mode 100644 index 0000000000..ac6403ebef --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/socialgraph/RecentFollowingRecentFollowingExpansionSourceFSConfig.scala @@ -0,0 +1,16 @@ +package com.twitter.follow_recommendations.common.candidate_sources.socialgraph + +import com.twitter.follow_recommendations.configapi.common.FeatureSwitchConfig +import com.twitter.timelines.configapi.FSName +import com.twitter.timelines.configapi.FSParam +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class RecentFollowingRecentFollowingExpansionSourceFSConfig @Inject() () + extends FeatureSwitchConfig { + + override val booleanFSParams: Seq[FSParam[Boolean] with FSName] = Seq( + RecentFollowingRecentFollowingExpansionSourceParams.CallSgsCachedColumn, + ) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/socialgraph/RecentFollowingRecentFollowingExpansionSourceParams.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/socialgraph/RecentFollowingRecentFollowingExpansionSourceParams.scala new file mode 100644 index 0000000000..4c51233ad4 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/socialgraph/RecentFollowingRecentFollowingExpansionSourceParams.scala @@ -0,0 +1,10 @@ +package com.twitter.follow_recommendations.common.candidate_sources.socialgraph + +import com.twitter.timelines.configapi.FSParam + +object RecentFollowingRecentFollowingExpansionSourceParams { + object CallSgsCachedColumn + extends FSParam[Boolean]( + "recent_following_recent_following_source_call_sgs_cached_column", + true) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/BUILD b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/BUILD new file mode 100644 index 0000000000..e89bade430 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/BUILD @@ -0,0 +1,28 @@ +scala_library( + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/com/google/inject:guice", + "3rdparty/jvm/com/google/inject/extensions:guice-assistedinject", + "3rdparty/jvm/net/codingwell:scala-guice", + "3rdparty/jvm/org/slf4j:slf4j-api", + "finatra/inject/inject-core/src/main/scala", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/addressbook", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/base", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/real_time_real_graph", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/strato", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/constants", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/stores", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/common", + "src/scala/com/twitter/onboarding/relevance/features/strongtie", + "src/thrift/com/twitter/search/account_search/extended_network:extended_network_users-scala", + "strato/config/columns/hub:hub-strato-client", + "strato/config/columns/onboarding/userrecs:userrecs-strato-client", + "strato/config/columns/search/account_search:account_search-strato-client", + "strato/src/main/scala/com/twitter/strato/client", + "util/util-slf4j-api/src/main/scala", + ], +) diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/BaseOnlineSTPSource.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/BaseOnlineSTPSource.scala new file mode 100644 index 0000000000..9f43656426 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/BaseOnlineSTPSource.scala @@ -0,0 +1,55 @@ +package com.twitter.follow_recommendations.common.candidate_sources.stp + +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.follow_recommendations.common.models.HasRecentFollowedUserIds +import com.twitter.follow_recommendations.common.models.STPGraph +import com.twitter.hermit.model.Algorithm +import com.twitter.product_mixer.core.functional_component.candidate_source.CandidateSource +import com.twitter.product_mixer.core.model.common.identifier.CandidateSourceIdentifier +import com.twitter.product_mixer.core.model.marshalling.request.HasClientContext +import com.twitter.stitch.Stitch +import com.twitter.timelines.configapi.HasParams +import com.twitter.util.logging.Logging +import com.twitter.wtf.scalding.jobs.strong_tie_prediction.STPFeatureGenerator +import com.twitter.wtf.scalding.jobs.strong_tie_prediction.STPRecord + +abstract class BaseOnlineSTPSource( + stpGraphBuilder: STPGraphBuilder, + baseStatsReceiver: StatsReceiver) + extends CandidateSource[ + HasClientContext with HasParams with HasRecentFollowedUserIds, + CandidateUser + ] + with Logging { + + protected val statsReceiver: StatsReceiver = baseStatsReceiver.scope("online_stp") + + override val identifier: CandidateSourceIdentifier = BaseOnlineSTPSource.Identifier + + def getCandidates( + records: Seq[STPRecord], + request: HasClientContext with HasParams with HasRecentFollowedUserIds + ): Stitch[Seq[CandidateUser]] + + override def apply( + request: HasClientContext with HasParams with HasRecentFollowedUserIds + ): Stitch[Seq[CandidateUser]] = + request.getOptionalUserId + .map { userId => + stpGraphBuilder(request) + .flatMap { graph: STPGraph => + logger.debug(graph) + val records = STPFeatureGenerator.constructFeatures( + userId, + graph.firstDegreeEdgeInfoList, + graph.secondDegreeEdgeInfoList) + getCandidates(records, request) + } + }.getOrElse(Stitch.Nil) +} + +object BaseOnlineSTPSource { + val Identifier: CandidateSourceIdentifier = CandidateSourceIdentifier( + Algorithm.OnlineStrongTiePredictionRecNoCaching.toString) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/Dbv2StpScorer.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/Dbv2StpScorer.scala new file mode 100644 index 0000000000..82308282a9 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/Dbv2StpScorer.scala @@ -0,0 +1,30 @@ +package com.twitter.follow_recommendations.common.candidate_sources.stp + +import com.twitter.cortex.deepbird.runtime.prediction_engine.TensorflowPredictionEngine +import com.twitter.follow_recommendations.common.constants.GuiceNamedConstants +import com.twitter.ml.api.Feature.Continuous +import com.twitter.ml.api.util.SRichDataRecord +import com.twitter.ml.prediction_service.PredictionRequest +import com.twitter.stitch.Stitch +import com.twitter.wtf.scalding.jobs.strong_tie_prediction.STPRecord +import com.twitter.wtf.scalding.jobs.strong_tie_prediction.STPRecordAdapter +import javax.inject.Inject +import javax.inject.Named +import javax.inject.Singleton + +/** + * STP ML ranker trained using DeepBirdV2 + */ +@Singleton +class Dbv2StpScorer @Inject() ( + @Named(GuiceNamedConstants.STP_DBV2_SCORER) tfPredictionEngine: TensorflowPredictionEngine) { + def getScoredResponse(record: STPRecord): Stitch[Option[Double]] = { + val request: PredictionRequest = new PredictionRequest( + STPRecordAdapter.adaptToDataRecord(record)) + val responseStitch = Stitch.callFuture(tfPredictionEngine.getPrediction(request)) + responseStitch.map { response => + val richDr = SRichDataRecord(response.getPrediction) + richDr.getFeatureValueOpt(new Continuous("output")) + } + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/EpStpScorer.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/EpStpScorer.scala new file mode 100644 index 0000000000..d022595756 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/EpStpScorer.scala @@ -0,0 +1,65 @@ +package com.twitter.follow_recommendations.common.candidate_sources.stp + +import com.twitter.bijection.scrooge.BinaryScalaCodec +import com.twitter.bijection.thrift.BinaryThriftCodec +import com.twitter.relevance.ep_model.scorer.EPScorer +import com.twitter.relevance.ep_model.scorer.ScorerUtil +import com.twitter.relevance.ep_model.thrift +import com.twitter.relevance.ep_model.thriftscala.EPScoringOptions +import com.twitter.relevance.ep_model.thriftscala.EPScoringRequest +import com.twitter.relevance.ep_model.thriftscala.EPScoringResponse +import com.twitter.relevance.ep_model.thriftscala.Record +import com.twitter.stitch.Stitch +import com.twitter.util.Future +import javax.inject.Inject +import javax.inject.Singleton +import scala.collection.JavaConverters._ +import scala.util.Success + +case class ScoredResponse(score: Double, featuresBreakdown: Option[String] = None) + +/** + * STP ML ranker trained using prehistoric ML framework + */ +@Singleton +class EpStpScorer @Inject() (epScorer: EPScorer) { + private def getScore(responses: List[EPScoringResponse]): Option[ScoredResponse] = + responses.headOption + .flatMap { response => + response.scores.flatMap { + _.headOption.map(score => ScoredResponse(ScorerUtil.normalize(score))) + } + } + + def getScoredResponse( + record: Record, + details: Boolean = false + ): Stitch[Option[ScoredResponse]] = { + val scoringOptions = EPScoringOptions( + addFeaturesBreakDown = details, + addTransformerIntermediateRecords = details + ) + val request = EPScoringRequest(auxFeatures = Some(Seq(record)), options = Some(scoringOptions)) + + Stitch.callFuture( + BinaryThriftCodec[thrift.EPScoringRequest] + .invert(BinaryScalaCodec(EPScoringRequest).apply(request)) + .map { thriftRequest: thrift.EPScoringRequest => + val responsesF = epScorer + .score(List(thriftRequest).asJava) + .map( + _.asScala.toList + .map(response => + BinaryScalaCodec(EPScoringResponse) + .invert(BinaryThriftCodec[thrift.EPScoringResponse].apply(response))) + .collect { case Success(response) => response } + ) + responsesF.map(getScore) + } + .getOrElse(Future(None))) + } +} + +object EpStpScorer { + val WithFeaturesBreakDown = false +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/MutualFollowStrongTiePredictionSource.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/MutualFollowStrongTiePredictionSource.scala new file mode 100644 index 0000000000..10a269931f --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/MutualFollowStrongTiePredictionSource.scala @@ -0,0 +1,61 @@ +package com.twitter.follow_recommendations.common.candidate_sources.stp + +import com.twitter.follow_recommendations.common.clients.socialgraph.RecentEdgesQuery +import com.twitter.follow_recommendations.common.clients.socialgraph.SocialGraphClient +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.follow_recommendations.common.models.HasRecentFollowedUserIds +import com.twitter.hermit.model.Algorithm +import com.twitter.product_mixer.core.functional_component.candidate_source.CandidateSource +import com.twitter.product_mixer.core.model.common.identifier.CandidateSourceIdentifier +import com.twitter.product_mixer.core.model.marshalling.request.HasClientContext +import com.twitter.socialgraph.thriftscala.RelationshipType +import com.twitter.stitch.Stitch +import com.twitter.strato.generated.client.onboarding.userrecs.StrongTiePredictionFeaturesOnUserClientColumn +import javax.inject.Singleton +import javax.inject.Inject + +/** + * Returns mutual follows. It first gets mutual follows from recent 100 follows and followers, and then unions this + * with mutual follows from STP features dataset. + */ +@Singleton +class MutualFollowStrongTiePredictionSource @Inject() ( + sgsClient: SocialGraphClient, + strongTiePredictionFeaturesOnUserClientColumn: StrongTiePredictionFeaturesOnUserClientColumn) + extends CandidateSource[HasClientContext with HasRecentFollowedUserIds, CandidateUser] { + val identifier: CandidateSourceIdentifier = + MutualFollowStrongTiePredictionSource.Identifier + + override def apply( + target: HasClientContext with HasRecentFollowedUserIds + ): Stitch[Seq[CandidateUser]] = { + target.getOptionalUserId match { + case Some(userId) => + val newFollowings = target.recentFollowedUserIds + .getOrElse(Nil) + .take(MutualFollowStrongTiePredictionSource.NumOfRecentFollowings) + val newFollowersStitch = + sgsClient + .getRecentEdges(RecentEdgesQuery(userId, Seq(RelationshipType.FollowedBy))).map( + _.take(MutualFollowStrongTiePredictionSource.NumOfRecentFollowers)) + val mutualFollowsStitch = + strongTiePredictionFeaturesOnUserClientColumn.fetcher + .fetch(userId).map(_.v.flatMap(_.topMutualFollows.map(_.map(_.userId))).getOrElse(Nil)) + + Stitch.join(newFollowersStitch, mutualFollowsStitch).map { + case (newFollowers, mutualFollows) => { + (newFollowings.intersect(newFollowers) ++ mutualFollows).distinct + .map(id => CandidateUser(id, Some(CandidateUser.DefaultCandidateScore))) + } + } + case _ => Stitch.Nil + } + } +} + +object MutualFollowStrongTiePredictionSource { + val Identifier: CandidateSourceIdentifier = CandidateSourceIdentifier( + Algorithm.MutualFollowSTP.toString) + val NumOfRecentFollowings = 100 + val NumOfRecentFollowers = 100 +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/OfflineMutualFollowExpansionSource.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/OfflineMutualFollowExpansionSource.scala new file mode 100644 index 0000000000..37981f8437 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/OfflineMutualFollowExpansionSource.scala @@ -0,0 +1,23 @@ +package com.twitter.follow_recommendations.common.candidate_sources.stp + +import com.google.inject.Singleton +import com.twitter.hermit.model.Algorithm +import com.twitter.product_mixer.core.model.common.identifier.CandidateSourceIdentifier +import com.twitter.strato.generated.client.onboarding.userrecs.MutualFollowExpansionClientColumn +import javax.inject.Inject + +/** + * A source that finds the mutual follows of one's mutual follows that one isn't following already. + */ +@Singleton +class OfflineMutualFollowExpansionSource @Inject() ( + column: MutualFollowExpansionClientColumn) + extends OfflineStrongTiePredictionBaseSource(column.fetcher) { + override val identifier: CandidateSourceIdentifier = + OfflineMutualFollowExpansionSource.Identifier +} + +object OfflineMutualFollowExpansionSource { + val Identifier: CandidateSourceIdentifier = + CandidateSourceIdentifier(Algorithm.MutualFollowExpansion.toString) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/OfflineStpSourceFsConfig.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/OfflineStpSourceFsConfig.scala new file mode 100644 index 0000000000..152fc97a62 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/OfflineStpSourceFsConfig.scala @@ -0,0 +1,14 @@ +package com.twitter.follow_recommendations.common.candidate_sources.stp + +import com.twitter.follow_recommendations.configapi.common.FeatureSwitchConfig +import com.twitter.timelines.configapi.FSName +import com.twitter.timelines.configapi.FSParam +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class OfflineStpSourceFsConfig @Inject() () extends FeatureSwitchConfig { + override val booleanFSParams: Seq[FSParam[Boolean] with FSName] = Seq( + OfflineStpSourceParams.UseDenserPmiMatrix + ) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/OfflineStpSourceParams.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/OfflineStpSourceParams.scala new file mode 100644 index 0000000000..fb1672cdbc --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/OfflineStpSourceParams.scala @@ -0,0 +1,9 @@ +package com.twitter.follow_recommendations.common.candidate_sources.stp + +import com.twitter.timelines.configapi.FSParam + +object OfflineStpSourceParams { + // If enabled, we use the new, denser version of PMI matrix to generate OfflineSTP candidates. + case object UseDenserPmiMatrix + extends FSParam[Boolean]("offline_stp_source_use_denser_pmi_matrix", default = false) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/OfflineStpSourceWithDensePmiMatrix.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/OfflineStpSourceWithDensePmiMatrix.scala new file mode 100644 index 0000000000..6a37ff222f --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/OfflineStpSourceWithDensePmiMatrix.scala @@ -0,0 +1,22 @@ +package com.twitter.follow_recommendations.common.candidate_sources.stp + +import com.google.inject.Singleton +import com.twitter.hermit.model.Algorithm +import com.twitter.product_mixer.core.model.common.identifier.CandidateSourceIdentifier +import com.twitter.strato.generated.client.hub.PpmiDenseMatrixCandidatesClientColumn +import javax.inject.Inject + +/** + * Main source for strong-tie-prediction candidates generated offline. + */ +@Singleton +class OfflineStpSourceWithDensePmiMatrix @Inject() ( + stpColumn: PpmiDenseMatrixCandidatesClientColumn) + extends OfflineStrongTiePredictionBaseSource(stpColumn.fetcher) { + override val identifier: CandidateSourceIdentifier = OfflineStpSourceWithDensePmiMatrix.Identifier +} + +object OfflineStpSourceWithDensePmiMatrix { + val Identifier: CandidateSourceIdentifier = + CandidateSourceIdentifier(Algorithm.StrongTiePredictionRec.toString) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/OfflineStpSourceWithLegacyPmiMatrix.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/OfflineStpSourceWithLegacyPmiMatrix.scala new file mode 100644 index 0000000000..7e17b2e8a8 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/OfflineStpSourceWithLegacyPmiMatrix.scala @@ -0,0 +1,23 @@ +package com.twitter.follow_recommendations.common.candidate_sources.stp + +import com.google.inject.Singleton +import com.twitter.hermit.model.Algorithm +import com.twitter.product_mixer.core.model.common.identifier.CandidateSourceIdentifier +import com.twitter.strato.generated.client.onboarding.userrecs.StrongTiePredictionClientColumn +import javax.inject.Inject + +/** + * Main source for strong-tie-prediction candidates generated offline. + */ +@Singleton +class OfflineStpSourceWithLegacyPmiMatrix @Inject() ( + stpColumn: StrongTiePredictionClientColumn) + extends OfflineStrongTiePredictionBaseSource(stpColumn.fetcher) { + override val identifier: CandidateSourceIdentifier = + OfflineStpSourceWithLegacyPmiMatrix.Identifier +} + +object OfflineStpSourceWithLegacyPmiMatrix { + val Identifier: CandidateSourceIdentifier = + CandidateSourceIdentifier(Algorithm.StrongTiePredictionRec.toString) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/OfflineStrongTiePredictionBaseSource.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/OfflineStrongTiePredictionBaseSource.scala new file mode 100644 index 0000000000..a46d49662c --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/OfflineStrongTiePredictionBaseSource.scala @@ -0,0 +1,57 @@ +package com.twitter.follow_recommendations.common.candidate_sources.stp + +import com.twitter.follow_recommendations.common.models.AccountProof +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.follow_recommendations.common.models.FollowProof +import com.twitter.follow_recommendations.common.models.Reason +import com.twitter.hermit.stp.thriftscala.STPResult +import com.twitter.product_mixer.core.functional_component.candidate_source.CandidateSource +import com.twitter.product_mixer.core.model.marshalling.request.HasClientContext +import com.twitter.stitch.Stitch +import com.twitter.strato.client.Fetcher +import com.twitter.timelines.configapi.HasParams + +/** Base class that all variants of our offline stp dataset can extend. Assumes the same STPResult + * value in the key and converts the result into the necessary internal model. + */ +abstract class OfflineStrongTiePredictionBaseSource( + fetcher: Fetcher[Long, Unit, STPResult]) + extends CandidateSource[HasParams with HasClientContext, CandidateUser] { + + def fetch( + target: Long, + ): Stitch[Seq[CandidateUser]] = { + fetcher + .fetch(target) + .map { result => + result.v + .map { candidates => OfflineStrongTiePredictionBaseSource.map(target, candidates) } + .getOrElse(Nil) + .map(_.withCandidateSource(identifier)) + } + } + + override def apply(request: HasParams with HasClientContext): Stitch[Seq[CandidateUser]] = { + request.getOptionalUserId.map(fetch).getOrElse(Stitch.Nil) + } +} + +object OfflineStrongTiePredictionBaseSource { + def map(target: Long, candidates: STPResult): Seq[CandidateUser] = { + for { + candidate <- candidates.strongTieUsers.sortBy(-_.score) + } yield CandidateUser( + id = candidate.userId, + score = Some(candidate.score), + reason = Some( + Reason( + Some( + AccountProof( + followProof = candidate.socialProof.map(proof => FollowProof(proof, proof.size)) + ) + ) + ) + ) + ) + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/OfflineStrongTiePredictionSource.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/OfflineStrongTiePredictionSource.scala new file mode 100644 index 0000000000..6a1cb3983d --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/OfflineStrongTiePredictionSource.scala @@ -0,0 +1,44 @@ +package com.twitter.follow_recommendations.common.candidate_sources.stp + +import com.google.inject.Singleton +import com.twitter.follow_recommendations.common.candidate_sources.stp.OfflineStpSourceParams.UseDenserPmiMatrix +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.hermit.model.Algorithm +import com.twitter.product_mixer.component_library.model.candidate.UserCandidate +import com.twitter.product_mixer.core.feature.Feature +import com.twitter.util.logging.Logging +import com.twitter.product_mixer.core.functional_component.candidate_source.CandidateSource +import com.twitter.product_mixer.core.model.common.identifier.CandidateSourceIdentifier +import com.twitter.product_mixer.core.model.marshalling.request.HasClientContext +import com.twitter.stitch.Stitch +import com.twitter.timelines.configapi.HasParams +import javax.inject.Inject + +object OfflineStpScore extends Feature[UserCandidate, Option[Double]] + +/** + * Main source for strong-tie-prediction candidates generated offline. + */ +@Singleton +class OfflineStrongTiePredictionSource @Inject() ( + offlineStpSourceWithLegacyPmiMatrix: OfflineStpSourceWithLegacyPmiMatrix, + offlineStpSourceWithDensePmiMatrix: OfflineStpSourceWithDensePmiMatrix) + extends CandidateSource[HasParams with HasClientContext, CandidateUser] + with Logging { + override val identifier: CandidateSourceIdentifier = OfflineStrongTiePredictionSource.Identifier + + override def apply(request: HasParams with HasClientContext): Stitch[Seq[CandidateUser]] = { + if (request.params(UseDenserPmiMatrix)) { + logger.info("Using dense PMI matrix.") + offlineStpSourceWithDensePmiMatrix(request) + } else { + logger.info("Using legacy PMI matrix.") + offlineStpSourceWithLegacyPmiMatrix(request) + } + } +} + +object OfflineStrongTiePredictionSource { + val Identifier: CandidateSourceIdentifier = + CandidateSourceIdentifier(Algorithm.StrongTiePredictionRec.toString) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/OnlineSTPSourceFSConfig.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/OnlineSTPSourceFSConfig.scala new file mode 100644 index 0000000000..ff61cc29cc --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/OnlineSTPSourceFSConfig.scala @@ -0,0 +1,15 @@ +package com.twitter.follow_recommendations.common.candidate_sources.stp + +import com.twitter.follow_recommendations.configapi.common.FeatureSwitchConfig +import com.twitter.timelines.configapi.FSName +import com.twitter.timelines.configapi.FSParam +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class OnlineSTPSourceFSConfig @Inject() () extends FeatureSwitchConfig { + override val booleanFSParams: Seq[FSParam[Boolean] with FSName] = Seq( + OnlineSTPSourceParams.DisableHeavyRanker, + OnlineSTPSourceParams.UseDBv2Scorer, + ) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/OnlineSTPSourceParams.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/OnlineSTPSourceParams.scala new file mode 100644 index 0000000000..e6224d359e --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/OnlineSTPSourceParams.scala @@ -0,0 +1,19 @@ +package com.twitter.follow_recommendations.common.candidate_sources.stp + +import com.twitter.timelines.configapi.FSParam +import com.twitter.timelines.configapi.Param + +object OnlineSTPSourceParams { + // This replaces the old scorer module, located at EpStpScorer.scala, with the new scorer, located + // at Dbv2StpScorer.scala. + case object UseDBv2Scorer + extends FSParam[Boolean]("online_stp_source_dbv2_scorer_enabled", default = false) + + // For experiments that test the impact of an improved OnlineSTP source, this controls the usage + // of the PostNux heavy-ranker. Note that this FS should *NOT* trigger bucket impressions. + case object DisableHeavyRanker + extends FSParam[Boolean]("online_stp_source_disable_heavy_ranker", default = false) + + case object SetPredictionDetails extends Param[Boolean](default = false) + +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/OnlineSTPSourceScorer.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/OnlineSTPSourceScorer.scala new file mode 100644 index 0000000000..16bc60776e --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/OnlineSTPSourceScorer.scala @@ -0,0 +1,29 @@ +package com.twitter.follow_recommendations.common.candidate_sources.stp + +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.follow_recommendations.common.models.HasRecentFollowedUserIds +import com.twitter.product_mixer.core.functional_component.candidate_source.CandidateSource +import com.twitter.product_mixer.core.model.common.identifier.CandidateSourceIdentifier +import com.twitter.product_mixer.core.model.marshalling.request.HasClientContext +import com.twitter.stitch.Stitch +import com.twitter.timelines.configapi.HasParams + +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class OnlineSTPSourceScorer @Inject() ( + onlineSTPSourceWithEPScorer: OnlineSTPSourceWithEPScorer) + extends CandidateSource[ + HasClientContext with HasParams with HasRecentFollowedUserIds, + CandidateUser + ] { + + override def apply( + request: HasClientContext with HasParams with HasRecentFollowedUserIds + ): Stitch[Seq[CandidateUser]] = { + onlineSTPSourceWithEPScorer(request) + } + + override val identifier: CandidateSourceIdentifier = BaseOnlineSTPSource.Identifier +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/OnlineSTPSourceWithDeepbirdV2Scorer.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/OnlineSTPSourceWithDeepbirdV2Scorer.scala new file mode 100644 index 0000000000..b8f348feab --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/OnlineSTPSourceWithDeepbirdV2Scorer.scala @@ -0,0 +1,76 @@ +package com.twitter.follow_recommendations.common.candidate_sources.stp +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.follow_recommendations.common.models.AccountProof +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.follow_recommendations.common.models.FollowProof +import com.twitter.follow_recommendations.common.models.HasRecentFollowedUserIds +import com.twitter.follow_recommendations.common.models.Reason +import com.twitter.onboarding.relevance.features.strongtie.{ + StrongTieFeatures => StrongTieFeaturesWrapper +} +import com.twitter.product_mixer.core.model.marshalling.request.HasClientContext +import com.twitter.stitch.Stitch +import com.twitter.timelines.configapi.HasParams +import com.twitter.wtf.scalding.jobs.strong_tie_prediction.STPRecord +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class OnlineSTPSourceWithDeepbirdV2Scorer @Inject() ( + dbv2StpScorer: Dbv2StpScorer, + stpGraphBuilder: STPGraphBuilder, + baseStatReceiver: StatsReceiver) + extends BaseOnlineSTPSource(stpGraphBuilder, baseStatReceiver) { + + private val dbv2ScorerUsedCounter = statsReceiver.counter("dbv2_scorer_used") + private val dbv2ScorerFailureCounter = statsReceiver.counter("dbv2_scorer_failure") + private val dbv2ScorerSuccessCounter = statsReceiver.counter("dbv2_scorer_success") + + override def getCandidates( + records: Seq[STPRecord], + request: HasClientContext with HasParams with HasRecentFollowedUserIds, + ): Stitch[Seq[CandidateUser]] = { + val possibleCandidates: Seq[Stitch[Option[CandidateUser]]] = records.map { trainingRecord => + dbv2ScorerUsedCounter.incr() + val score = dbv2StpScorer.getScoredResponse(trainingRecord) + score.map { + case None => + dbv2ScorerFailureCounter.incr() + None + case Some(scoreVal) => + dbv2ScorerSuccessCounter.incr() + Some( + CandidateUser( + id = trainingRecord.destinationId, + score = Some(OnlineSTPSourceWithDeepbirdV2Scorer.logitSubtraction(scoreVal)), + reason = Some( + Reason(Some( + AccountProof(followProof = + Some(FollowProof(trainingRecord.socialProof, trainingRecord.socialProof.size))) + ))) + ).withCandidateSourceAndFeatures( + identifier, + Seq(StrongTieFeaturesWrapper(trainingRecord.features))) + ) + } + } + Stitch.collect(possibleCandidates).map { _.flatten.sortBy(-_.score.getOrElse(0.0)) } + } +} + +object OnlineSTPSourceWithDeepbirdV2Scorer { + // The following two variables are the means for the distribution of scores coming from the legacy + // and DBv2 OnlineSTP models. We need this to calibrate the DBv2 scores and align the two means. + // BQ Link: https://console.cloud.google.com/bigquery?sq=213005704923:e06ac27e4db74385a77a4b538c531f82 + private val legacyMeanScore = 0.0478208871192468 + private val dbv2MeanScore = 0.238666097210261 + + // In below are the necessary functions to calibrate the scores such that the means are aligned. + private val EPS: Double = 1e-8 + private val e: Double = math.exp(1) + private def sigmoid(x: Double): Double = math.pow(e, x) / (math.pow(e, x) + 1) + // We add an EPS to the denominator to avoid division by 0. + private def logit(x: Double): Double = math.log(x / (1 - x + EPS)) + def logitSubtraction(x: Double): Double = sigmoid( + logit(x) - (logit(dbv2MeanScore) - logit(legacyMeanScore))) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/OnlineSTPSourceWithEPScorer.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/OnlineSTPSourceWithEPScorer.scala new file mode 100644 index 0000000000..1c6163852d --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/OnlineSTPSourceWithEPScorer.scala @@ -0,0 +1,58 @@ +package com.twitter.follow_recommendations.common.candidate_sources.stp + +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.follow_recommendations.common.candidate_sources.stp.OnlineSTPSourceParams.SetPredictionDetails +import com.twitter.follow_recommendations.common.models.AccountProof +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.follow_recommendations.common.models.FollowProof +import com.twitter.follow_recommendations.common.models.HasRecentFollowedUserIds +import com.twitter.follow_recommendations.common.models.Reason +import com.twitter.onboarding.relevance.features.strongtie.{ + StrongTieFeatures => StrongTieFeaturesWrapper +} +import com.twitter.product_mixer.core.model.marshalling.request.HasClientContext +import com.twitter.stitch.Stitch +import com.twitter.timelines.configapi.HasParams +import com.twitter.util.logging.Logging +import com.twitter.wtf.scalding.jobs.strong_tie_prediction.STPRecord +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class OnlineSTPSourceWithEPScorer @Inject() ( + epStpScorer: EpStpScorer, + stpGraphBuilder: STPGraphBuilder, + baseStatReceiver: StatsReceiver) + extends BaseOnlineSTPSource(stpGraphBuilder, baseStatReceiver) + with Logging { + private val epScorerUsedCounter = statsReceiver.counter("ep_scorer_used") + + override def getCandidates( + records: Seq[STPRecord], + request: HasClientContext with HasParams with HasRecentFollowedUserIds, + ): Stitch[Seq[CandidateUser]] = { + epScorerUsedCounter.incr() + + val possibleCandidates: Seq[Stitch[Option[CandidateUser]]] = records.map { trainingRecord => + val scoredResponse = + epStpScorer.getScoredResponse(trainingRecord.record, request.params(SetPredictionDetails)) + scoredResponse.map(_.map { response: ScoredResponse => + logger.debug(response) + CandidateUser( + id = trainingRecord.destinationId, + score = Some(response.score), + reason = Some( + Reason( + Some( + AccountProof(followProof = + Some(FollowProof(trainingRecord.socialProof, trainingRecord.socialProof.size))) + ))) + ).withCandidateSourceAndFeatures( + identifier, + Seq(StrongTieFeaturesWrapper(trainingRecord.features))) + }) + } + + Stitch.collect(possibleCandidates).map { _.flatten.sortBy(-_.score.getOrElse(0.0)) } + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/README.md b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/README.md new file mode 100644 index 0000000000..f3d415d3a4 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/README.md @@ -0,0 +1,47 @@ +# Strong Tie Prediction (STP) Candidate Source +Provides accounts with a high probability of potential mutual follows between the target user and the candidates. + +## STP: Strong Tie Prediction +STP refers to the prediction of p(MutualFollow) for a given pair of users, which powers the concept of People You May Know (PYMK). + +For training, positives are existing mutual follows and negatives are mutual follows of your follows. Features help distinguish between friends and friends-of-friends. + +For inference, candidates are the topK mutuals of your follows. These are rescored, and we only send the topN to the product or next re-ranker. + + +### Online STP +Online STP generates a pool of candidates who are then ranked via a lightweight ranker. +It does this through a two-hop expansion of the mutual follow graph of users, where the first-degree neighbor is another user who has a link with the target user from following types: +* Mutual Follow +* Outbound phone contacts +* Outbound email contacts +* Inbound phone contacts +* Inbound email contacts + +The second-degree neighbor can only be a mutual follow link. + +Currently, online STP can only perform the two-hop expansions on new users (<= 30 days since signup) due to compute resource constraints. + +Features used for the lightweight ranker: +* realGraphWeight: real graph weight between user and first degree nodes +* isForwardEmail: whether the candidate is in the user's email book +* isReverseEmail: whether the user is in the candidate's email book +* isForwardPhonebook: whether the candidate is in the user's phone book +* isReversePhonebook: whether the user is in the candidate's phone book +* numMutualFollowPath: number of mutual follow path between the user and the candidate +* numLowTweepcredFollowPath: number of mutual low TweepCred path between the user and the candidate + * Tweepcred is a social network analysis tool that calculates the influence of Twitter users based on their interactions with other users. The tool uses the PageRank algorithm to rank users based on their influence. +* hasForwardEmailPath: is there a third user x in the user's email book that connect user -> x -> candidate? +* hasReverseEmailPath: is there a third user x in the user's reverse email book that connect user -> x -> candidate? +* hasForwardPhonebookPath: is there a third user x in the user's phonebook that connect user -> x -> candidate? +* hasReversePhonebookPath: is there a third user x in the user's reverse phonebook that connect user -> x -> candidate? + +### Offline STP +Offline STP is powered by Pointwise Mutual Information, which measures the association between two users based on Twitter's mutual follow graph. +An offline job generates candidates based on the overlap between their Mutual and Addressbook Follows and that of the target user. Candidates are then made available online. +Candidates in OfflineSTP are "accounts that have a high overlap of mutually-followed accounts with an account in your follow graph." +This can potentially mean that OfflineSTP has a bigger reach than OnlineSTP. +For example, in the provided diagram, B and C have a high overlap of mutual follows, so it would be considered a candidate for A that is three hops away. +![img.png](img.png) + +Overall, STP is a useful candidate source for generating potential follow recommendations based on strong ties between users, but it should be used in conjunction with other candidate sources and re-rankers to provide a well-rounded set of recommendations for the target user. diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/STPFirstDegreeFetcher.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/STPFirstDegreeFetcher.scala new file mode 100644 index 0000000000..477e19eb57 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/STPFirstDegreeFetcher.scala @@ -0,0 +1,155 @@ +package com.twitter.follow_recommendations.common.candidate_sources.stp + +import com.twitter.conversions.DurationOps._ +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.follow_recommendations.common.candidate_sources.addressbook.ForwardEmailBookSource +import com.twitter.follow_recommendations.common.candidate_sources.addressbook.ForwardPhoneBookSource +import com.twitter.follow_recommendations.common.candidate_sources.addressbook.ReverseEmailBookSource +import com.twitter.follow_recommendations.common.candidate_sources.addressbook.ReversePhoneBookSource +import com.twitter.follow_recommendations.common.clients.real_time_real_graph.RealTimeRealGraphClient +import com.twitter.follow_recommendations.common.models.HasRecentFollowedUserIds +import com.twitter.follow_recommendations.common.models.PotentialFirstDegreeEdge +import com.twitter.follow_recommendations.common.stores.LowTweepCredFollowStore +import com.twitter.hermit.model.Algorithm +import com.twitter.hermit.model.Algorithm.Algorithm +import com.twitter.inject.Logging +import com.twitter.product_mixer.core.model.marshalling.request.HasClientContext +import com.twitter.stitch.Stitch +import com.twitter.timelines.configapi.HasParams +import com.twitter.util.Duration +import com.twitter.util.Timer +import com.twitter.wtf.scalding.jobs.strong_tie_prediction.FirstDegreeEdge +import com.twitter.wtf.scalding.jobs.strong_tie_prediction.FirstDegreeEdgeInfo +import com.twitter.wtf.scalding.jobs.strong_tie_prediction.FirstDegreeEdgeInfoMonoid +import javax.inject.Inject +import javax.inject.Singleton + +// Grabs FirstDegreeNodes from Candidate Sources +@Singleton +class STPFirstDegreeFetcher @Inject() ( + realTimeGraphClient: RealTimeRealGraphClient, + reversePhoneBookSource: ReversePhoneBookSource, + reverseEmailBookSource: ReverseEmailBookSource, + forwardEmailBookSource: ForwardEmailBookSource, + forwardPhoneBookSource: ForwardPhoneBookSource, + mutualFollowStrongTiePredictionSource: MutualFollowStrongTiePredictionSource, + lowTweepCredFollowStore: LowTweepCredFollowStore, + timer: Timer, + statsReceiver: StatsReceiver) + extends Logging { + + private val stats = statsReceiver.scope("STPFirstDegreeFetcher") + private val stitchRequests = stats.scope("stitchRequests") + private val allStitchRequests = stitchRequests.counter("all") + private val timeoutStitchRequests = stitchRequests.counter("timeout") + private val successStitchRequests = stitchRequests.counter("success") + + private implicit val firstDegreeEdgeInfoMonoid: FirstDegreeEdgeInfoMonoid = + new FirstDegreeEdgeInfoMonoid + + /** + * Used to map from algorithm to the correct fetcher and firstDegreeEdgeInfo. + * Afterward, uses fetcher to get candidates and construct the correct FirstDegreeEdgeInfo. + * */ + private def getPotentialFirstEdgesFromFetcher( + userId: Long, + target: HasClientContext with HasParams with HasRecentFollowedUserIds, + algorithm: Algorithm, + weight: Double + ): Stitch[Seq[PotentialFirstDegreeEdge]] = { + val (candidates, edgeInfo) = algorithm match { + case Algorithm.MutualFollowSTP => + ( + mutualFollowStrongTiePredictionSource(target), + Some(FirstDegreeEdgeInfo(mutualFollow = true))) + case Algorithm.ReverseEmailBookIbis => + (reverseEmailBookSource(target), Some(FirstDegreeEdgeInfo(reverseEmail = true))) + case Algorithm.ReversePhoneBook => + (reversePhoneBookSource(target), Some(FirstDegreeEdgeInfo(reversePhone = true))) + case Algorithm.ForwardEmailBook => + (forwardEmailBookSource(target), Some(FirstDegreeEdgeInfo(forwardEmail = true))) + case Algorithm.ForwardPhoneBook => + (forwardPhoneBookSource(target), Some(FirstDegreeEdgeInfo(forwardPhone = true))) + case Algorithm.LowTweepcredFollow => + ( + lowTweepCredFollowStore.getLowTweepCredUsers(target), + Some(FirstDegreeEdgeInfo(lowTweepcredFollow = true))) + case _ => + (Stitch.Nil, None) + } + candidates.map(_.flatMap { candidate => + edgeInfo.map(PotentialFirstDegreeEdge(userId, candidate.id, algorithm, weight, _)) + }) + } + + /** + * Using the DefaultMap (AlgorithmToScore) we iterate through algorithm/weights to get + * candidates with a set weight. Then, given repeating candidates (by candidate id). + * Given those candidates we group by the candidateId and sum all below weights and combine + * the edgeInfos of into one. Then we choose the candidates with most weight. Finally, + * we attach the realGraphWeight score to those candidates. + * */ + def getFirstDegreeEdges( + target: HasClientContext with HasParams with HasRecentFollowedUserIds + ): Stitch[Seq[FirstDegreeEdge]] = { + target.getOptionalUserId + .map { userId => + allStitchRequests.incr() + val firstEdgesQueryStitch = Stitch + .collect(STPFirstDegreeFetcher.DefaultGraphBuilderAlgorithmToScore.map { + case (algorithm, candidateWeight) => + getPotentialFirstEdgesFromFetcher(userId, target, algorithm, candidateWeight) + }.toSeq) + .map(_.flatten) + + val destinationIdsToEdges = firstEdgesQueryStitch + .map(_.groupBy(_.connectingId).map { + case (destinationId: Long, edges: Seq[PotentialFirstDegreeEdge]) => + val combinedDestScore = edges.map(_.score).sum + val combinedEdgeInfo: FirstDegreeEdgeInfo = + edges.map(_.edgeInfo).fold(firstDegreeEdgeInfoMonoid.zero) { + (aggregatedInfo, currentInfo) => + firstDegreeEdgeInfoMonoid.plus(aggregatedInfo, currentInfo) + } + (destinationId, combinedEdgeInfo, combinedDestScore) + }).map(_.toSeq) + + val topDestinationEdges = destinationIdsToEdges.map(_.sortBy { + case (_, _, combinedDestScore) => -combinedDestScore + }.take(STPFirstDegreeFetcher.MaxNumFirstDegreeEdges)) + + Stitch + .join(realTimeGraphClient.getRealGraphWeights(userId), topDestinationEdges).map { + case (realGraphWeights, topDestinationEdges) => + successStitchRequests.incr() + topDestinationEdges.map { + case (destinationId, combinedEdgeInfo, _) => + val updatedEdgeInfo = combinedEdgeInfo.copy( + realGraphWeight = realGraphWeights.getOrElse(destinationId, 0.0), + lowTweepcredFollow = + !combinedEdgeInfo.mutualFollow && combinedEdgeInfo.lowTweepcredFollow + ) + FirstDegreeEdge(userId, destinationId, updatedEdgeInfo) + } + }.within(STPFirstDegreeFetcher.LongTimeoutFetcher)(timer).rescue { + case ex => + timeoutStitchRequests.incr() + logger.error("Exception while loading direct edges in OnlineSTP: ", ex) + Stitch.Nil + } + }.getOrElse(Stitch.Nil) + } +} + +object STPFirstDegreeFetcher { + val MaxNumFirstDegreeEdges = 200 + val DefaultGraphBuilderAlgorithmToScore = Map( + Algorithm.MutualFollowSTP -> 10.0, + Algorithm.LowTweepcredFollow -> 6.0, + Algorithm.ForwardEmailBook -> 7.0, + Algorithm.ForwardPhoneBook -> 9.0, + Algorithm.ReverseEmailBookIbis -> 5.0, + Algorithm.ReversePhoneBook -> 8.0 + ) + val LongTimeoutFetcher: Duration = 300.millis +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/STPGraphBuilder.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/STPGraphBuilder.scala new file mode 100644 index 0000000000..0d2fe7ffcb --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/STPGraphBuilder.scala @@ -0,0 +1,32 @@ +package com.twitter.follow_recommendations.common.candidate_sources.stp + +import com.twitter.finagle.stats.Stat +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.follow_recommendations.common.models.HasRecentFollowedUserIds +import com.twitter.follow_recommendations.common.models.STPGraph +import com.twitter.product_mixer.core.model.marshalling.request.HasClientContext +import com.twitter.stitch.Stitch +import com.twitter.timelines.configapi.HasParams +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class STPGraphBuilder @Inject() ( + stpFirstDegreeFetcher: STPFirstDegreeFetcher, + stpSecondDegreeFetcher: STPSecondDegreeFetcher, + statsReceiver: StatsReceiver) { + private val stats: StatsReceiver = statsReceiver.scope(this.getClass.getSimpleName) + private val firstDegreeStat: Stat = stats.stat("first_degree_edges") + private val secondDegreeStat: Stat = stats.stat("second_degree_edges") + def apply( + target: HasClientContext with HasParams with HasRecentFollowedUserIds + ): Stitch[STPGraph] = stpFirstDegreeFetcher + .getFirstDegreeEdges(target).flatMap { firstDegreeEdges => + firstDegreeStat.add(firstDegreeEdges.size) + stpSecondDegreeFetcher + .getSecondDegreeEdges(target, firstDegreeEdges).map { secondDegreeEdges => + secondDegreeStat.add(firstDegreeEdges.size) + STPGraph(firstDegreeEdges.toList, secondDegreeEdges.toList) + } + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/STPSecondDegreeFetcher.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/STPSecondDegreeFetcher.scala new file mode 100644 index 0000000000..b7e996ab3e --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/STPSecondDegreeFetcher.scala @@ -0,0 +1,94 @@ +package com.twitter.follow_recommendations.common.candidate_sources.stp + +import com.twitter.follow_recommendations.common.models.IntermediateSecondDegreeEdge +import com.twitter.product_mixer.core.model.marshalling.request.HasClientContext +import com.twitter.stitch.Stitch +import com.twitter.strato.generated.client.onboarding.userrecs.StrongTiePredictionFeaturesOnUserClientColumn +import com.twitter.timelines.configapi.HasParams +import com.twitter.wtf.scalding.jobs.strong_tie_prediction.FirstDegreeEdge +import com.twitter.wtf.scalding.jobs.strong_tie_prediction.SecondDegreeEdge +import com.twitter.wtf.scalding.jobs.strong_tie_prediction.SecondDegreeEdgeInfo +import javax.inject.Inject +import javax.inject.Singleton + +// Link to code functionality we're migrating +@Singleton +class STPSecondDegreeFetcher @Inject() ( + strongTiePredictionFeaturesOnUserClientColumn: StrongTiePredictionFeaturesOnUserClientColumn) { + + private def scoreSecondDegreeEdge(edge: SecondDegreeEdge): (Int, Int, Int) = { + def bool2int(b: Boolean): Int = if (b) 1 else 0 + ( + -edge.edgeInfo.numMutualFollowPath, + -edge.edgeInfo.numLowTweepcredFollowPath, + -(bool2int(edge.edgeInfo.forwardEmailPath) + bool2int(edge.edgeInfo.reverseEmailPath) + + bool2int(edge.edgeInfo.forwardPhonePath) + bool2int(edge.edgeInfo.reversePhonePath)) + ) + } + + // Use each first-degree edge(w/ candidateId) to expand and find mutual follows. + // Then, with the mutual follows, group-by candidateId and join edge information + // to create secondDegree edges. + def getSecondDegreeEdges( + target: HasClientContext with HasParams, + firstDegreeEdges: Seq[FirstDegreeEdge] + ): Stitch[Seq[SecondDegreeEdge]] = { + target.getOptionalUserId + .map { userId => + val firstDegreeConnectingIds = firstDegreeEdges.map(_.dstId) + val firstDegreeEdgeInfoMap = firstDegreeEdges.map(e => (e.dstId, e.edgeInfo)).toMap + + val intermediateSecondDegreeEdgesStitch = Stitch + .traverse(firstDegreeConnectingIds) { connectingId => + val stpFeaturesOptStitch = strongTiePredictionFeaturesOnUserClientColumn.fetcher + .fetch(connectingId) + .map(_.v) + stpFeaturesOptStitch.map { stpFeatureOpt => + val intermediateSecondDegreeEdges = for { + edgeInfo <- firstDegreeEdgeInfoMap.get(connectingId) + stpFeatures <- stpFeatureOpt + topSecondDegreeUserIds = + stpFeatures.topMutualFollows + .getOrElse(Nil) + .map(_.userId) + .take(STPSecondDegreeFetcher.MaxNumOfMutualFollows) + } yield topSecondDegreeUserIds.map( + IntermediateSecondDegreeEdge(connectingId, _, edgeInfo)) + intermediateSecondDegreeEdges.getOrElse(Nil) + } + }.map(_.flatten) + + intermediateSecondDegreeEdgesStitch.map { intermediateSecondDegreeEdges => + val secondaryDegreeEdges = intermediateSecondDegreeEdges.groupBy(_.candidateId).map { + case (candidateId, intermediateEdges) => + SecondDegreeEdge( + srcId = userId, + dstId = candidateId, + edgeInfo = SecondDegreeEdgeInfo( + numMutualFollowPath = intermediateEdges.count(_.edgeInfo.mutualFollow), + numLowTweepcredFollowPath = + intermediateEdges.count(_.edgeInfo.lowTweepcredFollow), + forwardEmailPath = intermediateEdges.exists(_.edgeInfo.forwardEmail), + reverseEmailPath = intermediateEdges.exists(_.edgeInfo.reverseEmail), + forwardPhonePath = intermediateEdges.exists(_.edgeInfo.forwardPhone), + reversePhonePath = intermediateEdges.exists(_.edgeInfo.reversePhone), + socialProof = intermediateEdges + .filter { e => e.edgeInfo.mutualFollow || e.edgeInfo.lowTweepcredFollow } + .sortBy(-_.edgeInfo.realGraphWeight) + .take(3) + .map { c => (c.connectingId, c.edgeInfo.realGraphWeight) } + ) + ) + } + secondaryDegreeEdges.toSeq + .sortBy(scoreSecondDegreeEdge) + .take(STPSecondDegreeFetcher.MaxNumSecondDegreeEdges) + } + }.getOrElse(Stitch.Nil) + } +} + +object STPSecondDegreeFetcher { + val MaxNumSecondDegreeEdges = 200 + val MaxNumOfMutualFollows = 50 +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/SocialProofEnforcedOfflineStrongTiePredictionSource.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/SocialProofEnforcedOfflineStrongTiePredictionSource.scala new file mode 100644 index 0000000000..140b8b1561 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/SocialProofEnforcedOfflineStrongTiePredictionSource.scala @@ -0,0 +1,28 @@ +package com.twitter.follow_recommendations.common.candidate_sources.stp + +import com.google.inject.Singleton +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.follow_recommendations.common.candidate_sources.base.SocialProofEnforcedCandidateSource +import com.twitter.follow_recommendations.common.transforms.modify_social_proof.ModifySocialProof +import com.twitter.hermit.model.Algorithm +import com.twitter.product_mixer.core.model.common.identifier.CandidateSourceIdentifier +import javax.inject.Inject + +@Singleton +class SocialProofEnforcedOfflineStrongTiePredictionSource @Inject() ( + offlineStrongTiePredictionSource: OfflineStrongTiePredictionSource, + modifySocialProof: ModifySocialProof, + statsReceiver: StatsReceiver) + extends SocialProofEnforcedCandidateSource( + offlineStrongTiePredictionSource, + modifySocialProof, + SocialProofEnforcedOfflineStrongTiePredictionSource.MinNumSocialProofsRequired, + SocialProofEnforcedOfflineStrongTiePredictionSource.Identifier, + statsReceiver) + +object SocialProofEnforcedOfflineStrongTiePredictionSource { + val Identifier = CandidateSourceIdentifier( + Algorithm.StrongTiePredictionRecWithSocialProof.toString) + + val MinNumSocialProofsRequired = 1 +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/img.png b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp/img.png new file mode 100644 index 0000000000000000000000000000000000000000..1a27bb70e78e49e5e9c32420a50c6fce560662c9 GIT binary patch literal 54417 zcmXt9V{~3!w~d_!jjcwFZQEv(G`4M{v2ELE%!ZBa#$tuR(A_#~dh@^<1ikt4K4x}5J>Ehcv z9Ga^urCOtaIs6~|6(K4mJUDBAgFox*;q4f%uF^!vc=5}h94E`}LR zWDh)(FA(v|z)za%gy%);!Fvn@n7;rKG7JRU(8p~z@cX|%H3)_J{`bKt2(kYE{>8;F zmO(IJ;AF9eii0DS?#FKdIXSsT!=W!Lwfe%9P`Zw{hl*UT=Xj8ikkex+^hf=nNY5>( zz$*puuI|sY+gvXQ+&w&IMiMB@H@mzW-ET|^+&RnZjK?sz+%Ac-_`K6!UtbS_k8j`K zJm0iA(Sm|tQ6U7P&o?@y4<>U=uURY?XpYxfnN@3av$`I);iVGD5x>c%Co-9gO0+s0 z6m~q_OdSMY&tiTHzob-YB-;BPo8oXVBAV7!BfKG%zxl?mR3z8Wqf$D}7(3DBy9wo@&|jk*3vdF03;gnt8rE z8Ts=kSV%}H^^g+>*w!{{HlMdt3Y|_qtL1_MV2v`hDy_ut>U89atvCDm*XW~)>x34{ zH*B({jx};DPNyL0ES5B~dJMnzX)?s57sq!PefMOZwV&2^mv_ttf_mqw{5SXLUF*)$imC_=GmMfI>`pMRz%YgM6Jzte2(7KO!n10uc74wY7uIpyH=+i1^6 z$#{l@VM$ajU!bQ#`=<7E3_p9W)I7Ca61hvdfkyb(~*r~MH}a7o>He>IsE7b-Q&#G-Ime)fQ@ba``|%AaOc zqoIsUo z$XQ*vMoD6qosyVBHhkYaCcUJ61~-2&NH+t`HYX}elv8g*cpv~tL8t^oat5Pt#hs7m z@?_G`m=by6=PR{kHECQ`hi4F0F6sl<{%TN)vtQzYq^;KbBFd&RhHHIybUCX3<~|{I zix{62h09)#F|G@5Mywy;1`0!i^7OnD!~eK1NOpV@t~ae0=rL1|cysjlC$Ix~?GfaacY zQHT4D5)htd59jNZ+FxNH0tUbijWB7oM4V3+_P|Oyo*b1+_*bXnn{EyV_Jns`SI5l= zn)dv7j92rVSq%D!%2hhSzR722cD-Ip%PHq8^duvrlG|zUc-)4~dQRyWPh_UeMr-t* za@5DsL1(eG)t&No2+>UuFRrr~pSzW-RWKDz#%4<;(~O|=Q?@s&|JDX9VigRyw(@GT zb*mJk=qTD;y|=Pft1)A%2LW?B^BFHsbyiW<5ZiRFqIa19t#Kd7|Y z4LMnDHezQ>|AeC#04F$ew%T$>__^b4+Bs3?e7)M>L(YHlE3RI+?x8`0d znVB(U^Sp8nc1FiT5OJ?>hSNW7jOFLoi_m1Z)t5-^Onv{b;$p@d8X0Ai>c-R|Odo#} z9GEFKoufJ!rI^STh900aA#0|A5X$FNB$(N9Vpz9W?Ge7!hC8L<>F9rLU7R+Z-06dp}h z>y<>v+=aF}NHG7N6~Ffim$Q{jpUM|++Itm_el~&Ss{y)kx zM92f|0y}UN9wHIxEpKUGTF<4BBTSrsUU^L%a71OYBq~=*H1%x}RwdKG0wuQHaIRGG zcG)yX@tCE|Y>U4kt@6+|EQo0i9b<)9KR8&SrKYAvWB@3p9iZu!zHfJnx0>~nhYvRR zczrH?)H}`G&=6DmbkfZA_(<7Y*x#;9-m|Iyq1_F?J5Vak$=cRHUWzu&?FE0*UG(T7~!jmXRpBa{1QkhNR z0VjZPJzvYaKU-CPMmQ>4^y5G3$k}1vZXQ~4_2M=j&kn9P8J8-N%bLj%0}662tO8jO zn**ogUZL%GZ&~dxmCWfis4spB-4D6viNuB;7OTy)kPr|9T(nfcf}I7-zVy0xc|H9` zwfp#mA0HBFaEptPoNTrByKb4-<`c*9 z%>y?Fldb9`ZK*6)N;z`hNcb5X2Q&A4X8~mCIREiW7*lAcyPflsektAT(<$C0%u|4j*vnc4(FaWzX;1t^hWnDrSCuNyP2s0jP@`ObBOGrc`?4;Pqlj4Eu zqz3M#Jeo))tZJc_pC!(Yty9aCPmmHQ*N`RN=5%ECc)9gF`*}_ne;Q0Dmn=Y&;dhE- z7}67rM0lrm+a7nKCU?O3UOETpX-l+f0wOVda8c%0E>J5Z`p0$Gb z^rfEF)r@pD>rvx&c=gHVGWt{irqY24z_GM3dXA`QwKzg%DUjpkOi7Eq)IV_8KOw7@ zX~rqXd)}W;=a_uP0?brC49*77mDm>%mN(wr#PJRSC10yfpZJdO9ka=7GttN59-LKa zklxz27=YMQK+?dsh>O{IYui>2;p>@(y$ue6tpr03LH#DB5E$aT{8!F5Ps{Q54?Y>a zM9Mz$z-h!=;js^QK5E4_4sA0UbsHBLn!A_A#ou^`Qh{O*1-D1D4NIR9U+M4r@s{HA zT!vCK78|{Kb$CFc9g{#HZuS@?s8Tn|p2+TNBbCF*`*na!GT~Ap`IK2Ig^zY`Bqf7d zwL;*lX|x)~zi39+2b@QWBB`u+03rW5V5J6Xga+Qvce^igNz~}hCpXY7E>n7i3wX$l z!n2p%A8+)7J0I^CABW6)#S_{@NQn5WoBAu%`mEtI2fI}9jz)~+IN*Sx!f5~9W-}T> zjizx!XWenyuwX7*5Lwn6%#xI=wZjZ=G%C!+Pu9Q?k z8<89YPhBC?rR%v|stRhF zoHx6L@_cUt0)uvIv-`3{^3Y}bYpw>|=(K97)M}Lyy1Kg0v$TN2*oPz}eD!`lDp4u0 zQO%fKKJIy$(B(?1=;N2YrkWff^uF~S#nX*%`V*lxnHS}$c9q)gK&~xhG?M&>!BrS} zv(cE8YiG}@(N_D5Q!tfUJ>@4b4G0LhE#U$KdPJINpf)PKJ>QkBE$Ypz;EzSoDOe)$ z!oGO+lCj~j5KwUX`i|~kanJ1xs@v{Uvvw7%53%8%trGPAIb$f1f=8zo!DY9Z%lXWd zVa#HYSiQ@B{-hqbF3MdOUmCNMvjgj@xxZ?;rKF~UBLai=a6E(kd6o`vBolCSuc^Sj;k0qO z1+)^LXeFTUM{uOa=zo#7Vqc*v_xEmmZc4j8mF%M$ap`W!ckH5wY;KtMQ$>~d_P1I8 z0|oojUBZWA2pf`w24%3#Zg=&?FpFtJTFFiim%UZG^>uv40T3mVa1cof;8n5JWkW08 z+UV>3$$l;F#=uXGM!I))Hi7T?khEOk3Adni2*dEMxk#nWN;2CsE*g)^hA3k7ZdH^O z!0OXmvz~`Jw9=87-1{7bB#%|0=^2~RP7 z5Db55sjwUXhxg`kkK>uyqOg2jRHoHr;j}l3{$>s!MX(0@!ztqD!=0E?`y0ZjeFE)B z$%Y|^<1dc36F;T`##vX$`J!{uS$82jf%-U(jEDDEXdnPC&A(R1S8>JQP!h|=PCdp@ z41CK{g@8a&O};ex^*Dox<~WGX;NLKxQ}0LNbk`Xxa(a!%ljQHN1Y9CSU`x?d-I=<-UEUmh?xX-V(yE|63Sggk9UZAg= zwGfl#QANXb(myg5cX$HN;m%1AJ0)uETU>F>^Lj%cb$Hn7Ew=h+Xxi3{UMimK z7qHE8i7&g*WuYcA?6s<1l^YWy3rY^-}QcV z0Od67+%oNQW&g@rn<3OkhEG*XT&>eCmc`|q-zN$`6M&`iP0R35QbVSklpeaxK~!$< zcvCUHQ1a2)cr>NhI&@+A&xs6wZd|EK71%=zkLP{md*dfZv#inW!wY}odp0xpX2 zpPXLa-;{7SHceVd=8JW*pXR*^6`~XUN?uF#Pk+AC({)sxh`zn!gkDM;0Z30!GH|B! zSz?gzz7sXV;i%qYc!Gq@L+;-nwkx%os9W3H|9^7;EN6nX+<2w{;`12j_@XlS0<>R$ zZUpmA9iNRwwnX#__)0c1IG`n*E#PYbX$@c(On&=*@FC)`gf*#sHHrMsbAX(LHAGKP z=)u3h)`6h~xukj6(r~xr;|7jStrUAaS0*edDER-GY1&yObg@!<9N60*0gQZw=>VRC zV#$<{PXx_ZXWdg3+uAv5aCd+arEwU;CP8V&v@#J=@E+rww)S6IsL|^-i8BQNoyM+t zjEe3L39OlTGm?rfjQwCDljk!l-i5Otf=4qY1wf}i1t7ZH5}nT`WDwZvQhvTd+~gUP z>4ocZuZv)+V_foUm9{6>*^yDBjtaOU1I;Und%%^|lkM15N7?3t8u&q-MwtOBkc}`1 z`Mh#fE7a_}E9G=nucGq0IjVTKh5CMOzFwawBt@%d4U%KkPB~Q{-h7WcpGwgo z`c+#{?}-bD7rE{KG9g;K>qW6pFr4DfKH|~9UcVS-Nt5^wiKYB`M>jA9J#M2@km%^6 zmWI9Gcy8$-)I)Jp%H%lgHktpcg#O89Q^wZ3zc)S$q2HIvzv@Nq8=|864Ga8rUDDlI zW&WHvjz;AwR8MzA{QZN7L^&BLkOFm}c?5Ls9m}Kls|3RJV>T0hoZ}mIvy%1ef zCwnc=RN+1^TixO+=mn*$3rid;YKOfM$%^Jd8&;Dj|9dIgRrV4vNI1JhM#b%ZXpfgy z?9SgVNd%rhip5hVW(*&sZ+RH}Hh4Tl2=h~SAp%lC7N=HC<=s0xufVp5-^gKccs*`& zcSjPY-MG!&1q(fpj^&Q}Jtg|O>zs%54iD3KcQOCqMfR0VD~DcrMJ?v#!nyE?CoUef zH0`DGc+w>F0bDig=h@Yk9WK?8e&Wjt#5=%vq1i!qhl^>+wi;#*G(UJVzB_WgPNz$y z(2kr%T;haLO?M5?&%Wos22!{xQSB}LLrGqG7V~lYh|Yd;pN*nDq5L_4cAdqP-a&yt zT4OIf?D@)Yt@wVyZ^;B%qf?FC;0UMaS-lzDGo6ck>{_FHWk=TGeVzJhDz7+b)Yj#b zBy>)A%q}^#^xK|vl}cc#Iw#PI^oOaLX&4TajRd3 zmn#;C-SVKsMLvuQa~Z9Qk!zm^dQ9zY_=<3?K|7JwFoGYg@Jk#R!O{<~k?xvVJ@pQe-ATwL*a8HP*ke7pu@RB7uXB~Eh8gOZ$3U_j`T$V!XM@4^Wgr|O!0oNens+Tg zZnpLcoXY$8mc{9Cic(z#7H;&QqsygLsLMl)NXut-Q1M^AyOZ@i6P#98?5l|kSq!nS z=o0zQ^Tt|eF9S~XD~Otl%>vwv#hIin??-MApLxCRUpjG1l1tx>r^Zeqs2W4Jlj?r=_s~5#9Q18x2MLayu*U_9}n#QYe|Akc5+Db{389Lh%zTdjPv|~Xv6gdUTghq4F6)m(|6PBj*G7rn`nhx z9^WIx(Kr!s*&^TH9`xSIqCZD|6(G85;7|zvF7IpY1Fh2I2M`fRg1cfEv6mKho;jnv zmitL5>bEM8dugE;0RNHYLXL5A5-T$EM5_r_%KfvxzkRw?*FmDLlyCX&lmUofTU6EKpIRo zp(}_`6`KQ}7g~RD*;m=U-0Ht8Vej3Z?`+>)XpzQB!&!^ag%YnSQ}B0E|F^vyGDjwX zt=}N}GIV*jv7wNGr}(u3CY?>vk#N z8y@vZt+777yeQ8W%Tj4iST!oNGTdQSM3zF!7)%e<1AMpB{$K33BiULvf`9TU{e&T5 zR-M9VG$FrREi;7HJg!e_s(y&r67bJhUx9v0*&>Y}PRCdx$5{1odq~y|*~m40_qmmN zdOF{W2&;)YdhcuTytxGBy}2nm@K@)&vsmczyR;TmvA0?}7}I9|wb321e4-lUJGcm^ z=u;d^d3rEEuJC+$Z$0;-xbfmXr5YR*g-D^FkaV$3@w{`e^PZbIbZ{%=&bx8(=gxBQ zfR!9(6H|gO2Yl}_Zbt2alq{0Nx!LQ<;rX;-^0#&vX###I9-O4rMjPxm>Fh`hdfmUG z7Sni?uU$3D+v{*#eSD#w+X$Min5MQ;>Baz(ym( zdJ>?OZjveGGF@-{GQ9oQ2+v=;d2)*-U;R7o;n07V*91mgNemJ>7dKzqm5sV>Um(Vg88P6hp?52hXYFBpVO+_s>j{kAxG#cxR;e0;C(&)+_F9|kXK z=u*qPhT2L~uiG9zx{5;&Xp7Qh$6=ihcFEIh)B@fIvj1nB4M4 zKpN=@Zb+5r@V&7;Eg2R8;-K@CzSoq)sLuM5c1mV9$j|T-Sfest{dhuO`vJnDw zciP)d-a&a7DlFz=j?u|*%;Q?o#~|Y$>(~>s=-M*0hx-8rVT$%hIkcFaO>Uj`fMyh#2$Qo8QGF6DJ|AkN9f2F*UFxa<9LB3|%O0 zJ=65a--Hk*I{KyrB#$3V)?`DAaP)7~lx#gbeneU09-TE%e01u<$M$3p!zpGJb&yAb zY;$jUn@>=6b-%#5W1?adAK){aAXqADH`z*Uu+7~J9)Rj-y=<;10O+^1XX|Qb=vmnk z^|NL{dYDAvb0>RB_95RD4k;EysEZBwqgrDPlfd22%JWZWuuoUsekHmjVn;^V1HIrj zCtK%riPhcsJwYWV5|J7tMsvPQf2-B{*Fs#DNY zRvL-^;}3PrzYNF-8k|40>tfHdIHZs!S{-)BTt~6UzsN}SopH;=F~s%{*CNd;OvNL@ji8lJICXd z0Cg;8%2R^AW(XK^?==BlD753JsL+~QJsEobCcUAu|@Ifc`NW^B2Xy2I`VML zh3r8mdYW{PS^p(>5B~9$fl75)Wq1Mc2V({F1S5$~h&NSDYjv#p=}E6<<}B?fCk}zw zUpw94AbY6(v9XBn_0v~C;7XpoK}ZP zbY*2eHU%BwcB~7~x|)5wzdg^UQGy8aN}!UkyHS>dBdz(l&pvLwJ*H_xBmQ^0yUe!fXA&A5~fr}wm1E!0Y!2@ zl!15lPF0#T6H}QC!K0=E&K|5x0D{tNna9KZB^kxs`{fiH(pC#sEH}~m%~VcB7uVvE z8wPurzFhX|;a`+xD%sK1``!28H5yWp)xLY?U&^Uk10tz0FgOyG$4CdfK9nsbBxX-6 z0*$v@t+RN&{Vqgp5Aygi0z~$57kF#Z>LJ0Rs%b{l2e+Lziv5uI@TQ`VKB&yF;K+Zl z`F(gk2LW1*mO?4jD=og&1xRR$O-D=dGsV)<1>!Nqp-8e4h$Q}>#m?Vs3BcM^fU-j6 zVuBwiyD%VHIwLQyWI$x!oeeJll!cgHa| zwl&mLUbHP|(bYELA!sYJS!DOLijE2WD=O4g=VW1={zVBgzj7QW`h{M1u_r}VA4%Y7 zlb!Wb01O1m0gK(1Oh0GmiC54_K9!%HIjh^RtRUiWM}Rh>RWYiN!`ac##E8x#I27odBvYn^WRJ>-WfwV8A9kcNh}hj@(Gyf zj8Z=p!^?f=5X1Qd1`&H6@jE0^5IAh%AZjKet=q#_Y&ly4PhMEhn_Wgo+8n}r&KIXG z?-O{mMv&!xQ{hoGBdBL8WM+^zd|?+WHnSn2W?KVLPa_DxYq>jNz9{is2$No;eMNP7 z>7ff;T%8YY$Zx`OcV-SMa9!Fky_WDe^_jkG3B{bS^GKbXo~!GyHtON*biDQ3dXYRO zL-3pBrvAokJRzcL;U)f}gSc12tr_(4rw%x$nIItPLX9U=|Ii_y2|r zTwl*dW4j-h6wF~rFqV5-|G2}iZgMo+{-iuW3UF@11tswT5UxRfM5(GqugGqkIBhWizRYQeik)6V=lx0B&I+y(*38@KoutozW#1|8UmhrxOBQz;e&zJmzd zMMMuCVtsvqBBP;b{MHW}fMr`^Q02G%wh!8{WHTy6;h6k)E>9R54k6P0TMOKOBiGK6 zpIqt0!cgTp-B17nM>QY61wwY{;N%jZaD5Dm!x?Se|f>)LrI=+I|0q zfg;F}?PT26v;?s}f`nvdW+n%1TK7-BbRN%2;ad}@ApaW%=M4yIjo%<8#4&iB2FYF` z0Fo>2k;*e3Oa7FpSbTVX=Q`Q>3pAeaJdB8wxCiE!hd!Ah14b0)!%LuXSXhM{y!AP- zzi`F{3d5BZ(#-&mScBd+7^CYkM}sDbmpO-=Tz6ar<4xorXWc#9VIOJ>MKTvmRBJ7gU^RfPI%@ z+*&^G{2B)#7Kg(%S6Ol^?S|GAXJMcwimFH-frEWPvYQr9r`7B7QaL|iYIK;*7pXz& zKhSG(rt;R=`aYV$PCuT*&p()q=JA;r@Ji%!ilvf(`Pl7|*;HxQZg!Zu+V3<|+H9d^ zN0WmLcR(TcJeySN2Te)+a@n&V+6b$oZY1Dw+?%Ehg^!$1B#~hb8~;aQ$r&FuFjSKz zEHQ{`C~6zbSuafJPjFCOmzC2}9kFO=Skgx(6(d2duV1$YM2D|@jm_q+69Y9FN*$Br za9_+pSHxHD6H63Y*mm!+Sh2k~8T|g$qZs;fe`2U;!Eh z>TkAZtG!0ZL9ld>$4@=2(fXzMSxe)hyLNts!_dLHuK_fx?QBrJ(YO#WH7A#s zmwzh@M+d;iDcJ3tRaDH5N!SXd!=qJj4KY5~`w>SXRcHnbjRYAzc3b6Fbas%SCB{Axp;+cWNDTDc7ol_p zrUtdcl)3I+R1#`ybdap%v?zGEkp~}UG|1FYm zp+eQ;VW32|ve79~!go&TtRnbWaFk4=cr<2TBk=nG+w|i@&_uwJDCl*KXqG+tHu0IFg=srhuxVjx~y|Qx5rV}!p=p<|=$Co0C zjK9#xU^H55x{a6z^!r1mv$$Oi_GlA+=Eb5HPF@m;7{LTMD?jHCU)~=npHrEIQD-@s zik z55nmGsOwz1m{6NS%#QHofulD{q;5&Vd+kow76Hg@e$0;>xinY}3h?SQ@Q5WSpYJ`B%E?-0Kntc`8*a+p-cS#Q0y z$lX}PkBncr%a*khCr1*C51#M)@b99bPHEGlsTt9^RzaO-T}^@e#|B<}E=ZE(TFj|O zuTc4mVqmtz1X`tDLja5r9Q$v63jz7B`nUHBTg3FV?RV32t2s{vR|j}5Yp1VdiJuD8 zbS76G@t&zAJV-cJi@97tlQa#?(aixq&*}bDbKxhMnv%=G5;6EMgH`_mXalhUWNR+R z_>RHgqHW|PcNDCiZgqb)VQLg13*>1eWf1e$1Fxl@<3GcLcwM@!%$A}SxN6-QAUTE# zrW>{D%)*26)bsgaa;{tB7TSGBIhAyjkC{lz z;%KYv2lQ&YQ?6bYjo?$gXLgTio&RiRV__YzPf zlNJw*ng_W!V`nVp%M}yIWlI6+B8%Z5!b+VXv`(ATuk+2AC0hn!p^9-(D#0L_ICA9D zVG^#ewcL{2W-iowSdWu1*U22m7y)v%K#%&A#O%|5yaH^$mFP1O(n<8gW^zdsqOyNv zac9LS!t>n5mBXio7OyEb(kg|K%(#~;QibgXGSi{N$P$xEya}EZHsAFEx?dUGNKPcY-ek)uY|&q8v7~c6`KC0=|i8O+u z5omb`*vz$QCxg5nWWQyg(j9C!x!W@0tOguV7hL>nYUrEHrUMED z0hOnr9?h*Jjcx}=kz7JJ4ojspAkt!OquM?ZplqMbIr~re&hP8ZYnn}&QckV)H#Q`oE_@&@T~;4Mb;VsSzr!S6O9=)En(Exrp(2-{nb1$bW7jSJCBXF!xo zgO0=?(pybg2=T%q+UIw$2!?3jX$4KIcDa(|ba3bsta4_W0g)p~%JC-zCgoTgXi<7H zhGLel^rC(eCxuWhoCSIwo!1GUsv)_$A%Ej5w(CrwLWPp2@uCyfF48xH>I%c(yXqn; zIduF*Njg)%W>_ze8Lc1)UjlYO|JU7z@&Z}}Qs`A2ZbUeOGM42O>EP^nR1}Jyv1U53 zHC+iFnJjQnj*~D+3kTg0(CGYypE~4Zz9~zlo1V*q{ku~C>|0O>F&D^x<7pKI@ z+pO6gT3q1{Z!7k1hpybOBJ5y4*eR`gx?;Nq?h4fs+9>>-$VgZO&uyHDqg@D}?T>0U z&1J_1Rl4m2rEWBps_pauv3Qg9YH}x`bRwM?AXbrJm_HxZdhN7BO?s|}_>?LGQ+81s z8=G75c(d}>&vL-|Lr13!C=ZD|?hfR*fI57?T;RtVpli%qX|hyS+9My5EEbj}F@A!KaD7H{Bt3}^Fz1TTAlOJxqD8;OM_SG^dRb*aNz?(XZqQ67}nRUE?B8pJH z(^Bo*;yFIz0XN-_)ubFtyCN@j1r14pHx8Tne3f>}L=OKX?GZ3e8M5}QHyIN;^RX4J z&KBQkk~@uTVSZa_Mf_A;!jdzgK4vbI>V8ULsHWOV0a5u{uGTm7Kqy3}!yE;3RAl6y zY!)KQ-(}j(wxh28!C741MMK;{e3W#SJ!EX_z^e|uk~nhZgndM~!Bidq5{9w4{Dy|ITY zspRQ5YJ`LY;D`?w#p4lP6vLJRXDK-13sFVhe&R0$qdh^F;Z|01FNZssiWGLJ-V3Jp}nW;XB_AzQoc{`iGV~uN#M#>UhICXHLiwmas2zE zOrKr1Vcue%8tcl^zG=ZE?&{XHDKw5^1=AiN5ayfys~m;HYG7#ocO8j=`|=$((Ev6@O+$6|H}D!8L2|8YJ1X@HmVxv#%2A&5U7c#Q*q4MX#F)> z9gKtE@jiSCd2$6*!ZGOw73)4_3bE5~hn17|1sQ|WA%NsC9gt=xeKg}0Nu*`6+J2>w zGZN-cLN=Ggf=NtH)Uxn94GDQKiThR41D>WVz8IJ~?1#0!R3H8Qii6etl!*84MxPf2 z*J1Ey{qBM|oS-{_Q_?vH{=N#Klk$U3Ov4zTwum1x)r*KK;l`>;;U5s=)rT5AdsWh5 z+-C6tOBKuVp<4k?m!#V|;{e#GWVl!?WI|-7Q+RXMOz>DlJnnR(isD8hk|lDc%dW&p zxO^9JPIY2#wqWm#EbS#>1cKZ!fHQ&G=TezGAL`fS(ji_fzR|!O}C*s~hLU9|dw5EB7 z2gbNt(3pM(Af&nZiu|fZ`0_Oqnl8HlxvU-K-3E?B{Mb}Abow9)bneZsFtwBLu6>}# z6?5LQza`3T&9=Zt)@bT%%#2^rYx4@$7VX;CU{7gfx-|=>E_~#7VI(iUw!&EeB;_xg z>B?w8{*B98Ifw`CeF5Awqf)Q_LDL`s$VERjpIO%&CM>30eRg+t`PpynO2vP%AKe^x zhXp@xgiY#K9j0LXN3xecJ>Q>=kxBV}HT;(!Q@)MF4hr)ZxS_I?0l7+grdDnRo6cm4 zr*rAb*j8rs-;JDOYw0qNEb2rSbJXbDBud4A{+qb<;RSMiYP@#Nzb;)HS%kQfnk1$7#ju@)6VErU4E2%z@?~)*!3Yr}gHSsBvy^RR z6DYog_}*rpw|LQ5EJ{k9&RL*%E6okuZIofRc~u|CgKShDHo3jn%qe))Kr0@LP3Cej zmlmHmWlR@7V~oyTm+p49l*eW+Qr|y~dE|JxJ?c;AvNY=^2?Sh46=d5U(2*Ta<$-(O zFPn<7*K)xDQcd|JRc#r+GdA0ezs+BkC~Qu_Aa9fxH@fR$B8Ki9cO7r)Rh?m$S}v-_ zezZjtZ;*| z4HWdf@xoH^*(eT>whEvidl3|X!v*<#yt&em_kpn*mnY8sEAg<>KwKpNbk4h`dp1-{ z{2S00EoOIk&szsjKZ0x2a$QfC;=H`PZbNOp0+fpeo&;&W1=I>i=esmNa-Cos6I;(c z5w^r#9tm_1jpPiO%6>)soUaD`=k1`Z8{UcHn(W?#ej5koFl=3LaG^r%yqmT|-}o>E zw;xP$#0Tdfdltl_Y`a>4&p`1r6pYFSSLWd;H)UPoOvf}yPX%3dC0Y{9CqBq?#JRfZ zNr;S3S#eX7stImG(1=u1E{+@#WDPgH)HvlW+KL59^Aq)4s9jaSY`>I3!IdMkhQlIj z!ok694`WfcDMkkUA))b9aBDKVv*Pl(BGPF)9*rA|3eCpO#ebldaM~t4UP-02JtD4tt|w>{Vq~i+a9S zmVZG>crodI4buJ$8?rc;;Z1 z zlF2K45{gvJw&^1p6@mGyUg~;VuY0z4Q(HMJAP0AxR51$ByZl+nK^_V{#AY$P)ca~PhZV%-kY3G*Z8WLLX9Ez_km_BE(G_++F}lAB(lPpDUu#a;>z*DAx92=(70PkDs>4wo~t$Y2DljP7;;)1 z#q|qIg(xo7-yK}q(Um`l11WKBbg_#2wg^nhcs=iT?}h5J{2FW7VEcP4kjz1l=ToaS zAnS&d_2|jvi40G**=nH^$ZWt_3Nf{HAICEX%==5JN&MAa&ueqAW}=9!&jaVYolW_f z?|-Lr#M@YDPM$vnK)@i^T!#NtvFR(lKkm~6*m9{ICsz)J4-#R(+!I~=3$W-y9(WTU6TPBbPmU*mFI{znw=M#jmaFl{ zKbU$9g5aP5n$$K*mC^sof;W3Hp;i;=bkfV!szwvQdBzs1b;iDC7DdnFkRhOO$?!eH zsid}TGyiUHmkytig1I5MDdrYOQo7u4vdXbHyy0r|{)yQ>x8sk4+W~`E^g@xfGz#_t zw*kQ&O}nOha95QP#}^5Cg-Ka0@Qh7m!bW%>ifnT2bL1v5J_Om~_Z?MnJR=z!H5MzKbE? zO;>h$&R{L~M`pafDlUhnMeKmKaj>utOqs znfrTC7#qN;>)l9Jf+NfD=uOeVz`OjNUTv{8wUvAw_#lSv-2?{*cYk?=0z#?*G#-qZ zyOL`TL!=Iyh!m4+6P zRlV5?J&;qr3}Ia}7qsp#B&6K;^M>{K$>-Aj_l9m5GjmO!vb=^58XNq{Vxbb7p5bD> zUEIee<>>bJmQ%Ahh1O;p_>v(?1KV# z?s|Y74`bg983GqiBekNl#iI`}Hz}1)p?hV2bA@mS8C8KX%1fvXq5~Etf_S^P8@^nH z`x`Q`8Ag~pmqLVowj#nr@o=K*c4>#{?d2ABv*MEQ_30o-^QNq`1SJcYs+-RiF6714yH0$McZhGL&Up*9?DU|#QJV4Ru_Q~3W zSF~|D5fDL$`+n;a^nS7v^EacW3H+i3kXWskd`$)a?=Yj#)r2ok!vCW1mxIz*Jf`-y zKx|9()gG&6l{O8OtCT7I@TtQ5C!85Bi5CPP471{`PpBbS5bprol{ zE{%i|N+T(tAl=<8-5pXQ0@B?e2+|- zo)A81rQ3c($%Pd9+il)=5z89hUfp#OKk^EK?NEmb%_Hr=27n3z-;$O z=(ApKsn>Y(PE{d;@7>OLzAXNVfe8(;w;T}hrLkOzAxX~LENv+`L7ud7IRXpgyIeFe z-+=@uxwKFvQ1SxAC+=~)rW8e+M=tHy2apv2WKdb*jrMr1&(*TKtrU{u-_EDUvei&{ zd{bV0?kwKXFY`t$7w0QMOhdjj)N>t$^LId%hDawVG3m`7|Ek3@Grt#se>5WMx;K6> zU)y}E_g4pa-{cKo9<-&EvM4wjXFzt66>ZB98qtgrf&$_0XFfKmk}f3Y`Rxclpi^HyQ1A>FsA65?}DeS=5<+yC{+ z|M6#%k41nVN+q?36Qj-Gd71r=(ub*%ZW?nSz&_Xa#X`ZL7;L+|+$B81@^Zd%V}2`{ zq>!lSA?xP4B+~N90J{%@cwOl52e1S&XAAJ7L*WZ+CfLNv;a24gfe)-bpz(!zzwzay z|6LNW(r9m-yet9lRsV>w%BefM^VeSsxqowSuXHt~3;SOlh>iaF`zK|!C~5PMtm)Uu zs9cvC_`xBu>4`k)6*6~$=O{`9%!>oeuT_7fmuM7JFq!@9iIT0pvH(^0T!CeKC4Fy? zt9PHVO+ZQkVRoIziP_sXEHkfbV2*15PxE|uEX(e2^@X$w@rHmWQDeR)^#KF+mOGEW z?-jdcr~;3Qf@mecKFen3-`WnhCe%dmq{XTd1oX`U{s6-A5qE~Z5{!_YOK<@ZAvb(y z5Pl5h)w-~PnUCj6=jMG%t0L<)Wccrbav&Aa17QA0iqKvR<}%2ZBmPuIA6rKKqC3`2 z<`%P~H>5-Po=lVNX8Oc^ z9UzYirFYqayHISZsANywVVyC%7{QDYnu-l`;P*koG&B4Q=S-TB%dPHKX2e-{gm<#N z2FrLPMmu;Ihh*ivb^68}*!eFA|BKfo;BzCWXDJF&5*E87Y_6XX2`R8jEM!eruRD#M zYsC1s*f}FMA&U~hZ82SFUJm*Mhqz$Zjlu$yu-%O~3Z2Qz`iI47FQynHW601r@A_Wf zoi0;Z#gcV47nBjDt;IX*zBq3C&#s-X8e!$L?{}KxHK;C6Mb9NVNyz?e5vx%USfGb0 zv6e_LX!uK+Iz~o*cl`Z>_okD)scf8rbWt>w58r zpW_oKz4&Js^zO)LxE2#q1-3k~#Q*&wM~IzfvKE7D@k@x3EMQq9+QW`Zh&DhGvIYi& zt8*C%2;e>A+B@O2erw7sh>yIE^lIQnn+x%vVYmGmwWm`uFd-{_{K6Z)N1SSjj;wT$ z9C?2zKPvOPb`!E*Tnhq?=ZGuRplfVsDM0RuL6GXB8Ra|)ExSq%OBw@sPtn!5l{~>` z%|H=cXQO{EIZ6ri3Y%ESS7QLz*G#ShG+a2qXaP9&1X+MqjIc;J5Oq)%(Jpa=Wzxy` zcq!{`d+3WfV>m(J4Q4<4|DOeHDJiK2C3Sm~qAKW1Atrj%y!Vl6>Ed)qC>S9`Ouxsy zty#pZ8>fibnDvS%hjVm!HGcM|TquNLqNU(#tKW8-=FS z?9FikZVzob6=$uAxI8C8!R=4c7%tn#lFVxFQK2HKY&>KMozp>3!tQ|3E?fDC>`S zU|23`?1t*1{mWdC<&ZXBSqImJn9kR{(@4aYp6c2Rf1x4qMG2=%Sbz(@i7(^zO_%nW zwOoC?ax5p$I>SaNWL>|}FD3eGSmm6%*+(^|Dq47UF6y;9kiKs|HzkueX(D_;!$d{4i09_;IN!vY;6Y!KPOln|WV=C9w=Mju6m*9hB_KCtUdt`Pqazzvd_*cp zI{p{a_P?wpkeIe-cW)%>4Y{^@MDyMRL01tvbws92uLaP|b*C_eAcsCyH;6*HOO$3-jOSQvtu{niK;d|_f>c6&GG&=_ zTl5Vf063)Ag-rRZ1<<6DF+++PHxdO2@6n)o-$rrSI_9jK=#ddn4({_L(c52ip=5-w z=(M_ua0=c4P;t^opecXUp5BfzRQ~z4DI?T6aDb%u1YnvJEQVhJyeaR_3WWkMZ2nZSzq+x(&| zY0$h=%T!6M;(G$5RoC4pK^1Y>Yk@(^ya>O9V*V>%lnBD(6PZr=^S5n(H+oZ|0V*{= zy%8BAq#5tvWiAlj)(Ym&_eMD%j?8eD#M19fmV+=I+A#zw++V{BJOvwgg<7KJ=hAu2 z>36uKlaHSO+oe5YGeIO|WCp~&21k6lSjODHC%+oPOyZ&QUUn|sMZJNe3fm!;Zt0DD z6HEoxo~k4T(v|;Cie9m#-z_$OIC4S}_z8R*UR z`NI6uxQ%l!_)Ch_hX1q&?SMyg#!o{PdiR}ZLp^Y`5wq!mL_F>k{}{tr?+e-aDnK_q zL1qBemZP-4J)g-~H^A)iyKiJiR0)X?sRuV1+tcX~H@*dpF2c0hVppDvzunsrKj3M0du108+92Ondq6wghv#5RV1(n*bOCuG=h?ZD6T)mFrL^K zrAZeI8%>u5K(KGSr5>+?bBiDz4q$6ctF7*qFwn_h-LK?r=2ooU9yD5 zGlo(I;4EwI#B!cbjfJmde0@KKJPRX{M#H1AFy@j&Tx?=>B8vb%3>1X*O>)lb#OU9c zkB~oFQ0NZ=rVq~fFAXp4F};F6o?pkPoc}*}HAD6H_n*Q2D#W#50Y_%656Vm z@)+9z#~2co7o}aBxG^NRYCzI0o$7aYDC9QqgI%Jc%dyY_8Oc{SV`uq_4pX5H{BR84 z<9ry$7p8$##Hp1Yk0kYP3i(o@4(BKfHQ42`~H-WWz;@uyeCf%AWT;Lragr%yn^sj`YSo?SqlF|(@i$MFszO~8J@ThPm7 zTod8xR4D=_1O&Qh>6`Umja45@cg4SZ$X;cn|(Z zW1SShP$cdSPoWRM5*%aaHY6N<3#XxVNgwo~QYVklKdY_oub%j#Knd@cy!W-h)r zx-NmfGJ@AcI`0GVR0I;8OW*~lr{Ka&OQO9T)w(thO?FSq_f`350)dF%eUrSaL3ele z5@Z)JXWO2N?MNiR3j-rV#sQzaO)&myJ~S54;7tyVQ>wPv9nN?6c*fJH`nHZO1_z|rCmXRx2y?+sXliOD#E zy@oLlfWRLFq$0#9OM4%1G^&6=-}j=46G4-HtPxSF{eoT~B5Y+mUKh)uM{$}6{nM>) zFbt?9;uug<8O=Z~TyQINEgj!rbMa(=EhrBVu`MU6K5JDY?Fad`*@aD-hk7haRz)F^ zj^#gNxT`K!A)fJ$qf~w+zMN4qPcUxSvpP}(Jzl;er z{CDVMRNON6M9xDio5Q@%KTf!+dSzp>y5F`yjg$mXd6&P5#{tpD?XZ{Yj_a$?28NOl zq>5Q9#3^+$MM#nG0D{u;!;qb~D%U2`$ePw{HKgQ!Msp~oodT{>`syA4ECtG`o`LL* z;#w2*5?x{vr2&ktu|1lDlf=x9o&U8)Fi){N1jifWn+e%0642VFo0&z%V-vqV2FX>5 zw|_>}l)c!W{Q_74I=3=CZ99E$8UMu8kAzV8Qt+CgC_2?U zlS2k1=tst7dhoC(u~d*}pOd_V`7hj|Z+I*X4Y2pt5Ard?UHJ+Y_wLN4FV?z|A1sk5 z0n!>wxv8nCO7~5S5{#4af$k#~G#cRh$w6@n+o)*XHh^egfE5V;UP79X{o71Y*V1Xt z;17Ki{Crh|<0M!nYq}2DRc-%t?Do&9Xmio7$>lqiZe5uG-43hAGN#@0^WoL3N8j`D zJq>P9f__nj#t8a1;<{H}%Q}E(VG{*G$f`b;N_s&7EyHbR>EEc|T@TTBXsGNBpi#8a zPxV1QRA~bA3qk1s-T~8uV}ZSqzBMz`cr+lo(CLwmF2)l|R*|)jU@Q$t9nRP8-QC7thSc%>4rpKy56vu=Pj@*gR7|d=&JBtP zktXYO+EOmVm(mJgdwyh`g6R2;FV9OL-@pw39rS&GfgSzpf&4_Yq^zz>xB%k;)l@h} zxy+APYMWngI97%LCsi%c1q2yCC#z?p_KS8*xh3d>tQXPfhS#=lQN~&$$`^k*Tu!6D zjJbdYDf?WBnCfs+ht&&|WiiL-+?%`cGH-XzNm15fxG3K38zTDvpNtxb?t**(a8&Kc ztFL2*u1`|JWTQ|;&`9_j-g2z9S(ounboWfQ2uU9RR+=UR>3J~NbW0{fIk$}U9jNTh zb=_CyMA#3&n#}CEjseoDl%{ye_|yl3<2>mzV;cqH*8%ZnfM4k-Rp9^?q;wQk6;>bc z>C-2}+G@~bNQtf7-Z7vTX7xB!_QH30)2UbFadrAxX2Dj)&(S___X6Mur~3q2TUV-K z%YFBIxX?+}%%cphztAap8}>O)+doVmS?+{Kx&8tPeCGK*3~kP-`?q|-KVe*IF^L;219+3u&_M%Ak7N48 z03xi+eg^8VFdz zp;NKs9lrUHj0K%s?;e>30g%yRX%Zu+XjtPhjHUcw^)hgvk5 zGHB794kW>%S@i)=Ke{i9wDw(C*W3>b9Y}5g_2?ELgWZmNgBUi#%voM$uUThb-u89L;#*)xVeL2)JcGsh z?;XDA&YDRa3t4x0r%UTZnaA+fu~<@IMHQO+n&DCIRvo z7B&Zym?>SFM(#(AVI~1DV5$-&n-_xb3&6N%Ni%3?JJ)$8J0viT9|n zV4&DRK0Xd|uGNl?lMHqGE{dxjh~87^C*=X-G-{bTD%00}-Q}!AnE_%8cA)vDYl~CF zmS7j?Po{6XOS>Q}7|$z5Fi|I-|9k73x*c^9i_zXO*;10x(S+gMa@4k|WFQ4hZq-&})A(>f8*3pgr=&-yu=B6d7C z*+l_8!Fy{1ScW{9VQN`Oez1(yfz$_O?d zo2nB))U6ppVL?z>jpcp@s$~jh?V^XZs4a95zN|M%yd3AcG(TpCbHn6LI?}lfz)ahi zie8yP!Y8qQm$%xD%r8ll?{j@DsXb8&XltsjB~AWVv~0ViuZY+9c;_~OdLduM;<-I^ zrN=8+j1iOHuU&@IdEFR9m0W4WJ>s+cZ7Ln9C^6q0e0`ckau%XGo!+4XBDnJ1OTBD`wTXhT$@xN=ko;KCFJqR5((5pa&p<=efPn3bfk;TNXlj1 z&crqaJjOU_2{wR-TPD=iKmzYDX+=6D(l>tJ>Rf{S^>C@#*%`G$qTahn+mSkNmLXP4 zLXv|Iwg^X=&ivMDh9#E`#1HEo!iZr=T~UK=5+eezp<;3}*%I_Rx0cyt$4yN0|G9wM1*43lzx5BXChxri zoW%|E%$}_aA*L5e$9AXN#m=YtlVln`GJc9+h(Wo~AsyK$(e(imhN1(9_8?YRX%Xqt z_iZeF<{~EP$pZ|ZZNS4OdvN9huk_LcGnN}!?4Rb)5_%H+y=wZ122=#tY_qOmXo~9xc8C`lzf{AQn=dd+5C;4m znJvi_8ZyPXXe6sHmN*f#`EEU&mR@|o;b43MHfU0YuJyW0q_4V52SY=l{VyS8!9Cvu zmBw62N{_y#>w)6$O8%D>ot>Euqkm|DC-#AzTno?A?E`>xpv6B8zaVO}LzNKs4_guT zIKdvpADsq}7wgM|`SRvpHYf0L4YI%)NnJ8g84HYl5T4{~q7QMTvz)eU4miR+u4QO{ zTy?R&&T^f9i07VLtW-6jkm(^{tMUr=peTFk>csY@lt_`2=K3Dc7mkov)ZH|2XLCCr}CdmW;>5xe;?e*&mey2k&Qz(sHD*l)+G(DxRA%V z+%VuaZIj5x)*#@$xa44}T0!=Er`%p%Q(Q%4?~C+_1q|Y8r2dfjDmhF5 zPvH8ZtpgB;0Pd%9wBB!glV_8qZ=QX3M+ZNTCZj%C&)WdJI03z3IJ!Kl2efQ~R*}vu zfC6K8x{Q=ty>9pQMUyFgi|srgbAsh3_fXa4xw^0=v`YkVAz9XiYzUThTceMtrIfiV zj0>;aUXJwg?jG1mz={m(>VQjr1&g`8&sHXTG` zyB8B#%5xnO!6bi=YztG*|Lm8Bhz4BV)jBi2gA;VmtA4f21o<0 zJPQ~gN9FK3k<@$xO}1aDi?BmO4fAyBgYvv=_jSNqb`_NyyKme57EJhF3ob7(K+6Am zck*+TgI#=2s5;fiG(c?pX)&G9VGi|2pS`<0w9(BrgC(N)uC=)CQ>_415FrQ3R3A>o ztLyc?@EaJftdcH0a?d1`q-+KnitPRAL6IOcMJz3;u9_V4zyI@lCe_do$ z+#$>q#5xN<2h>W89l$U}j8Gi~2g1A#-Y5Ag5TE`jo*3~fR&oCFtA1oKSgiOxZ0pYJ z>h?|B`Mr#n-N{Cuc${@%7CU|b9wmL~F-`Bqo^`@YS5r)*%mxP(XT}2MDmMfk*ICmi zz(dtc*pI^j9(ibF%8vMV`|w`}EAJRgQ4+UcCqEyGp^yrU{1S+aMxi92$%5b)mT}cn z4cPfNFF{zj;G0|s%n#s1;i@o8Gse4R37JFc%e0afQl6<5?rRrLb5c%3R2=I4+FJVZ zY=r!Iro@kp@3P_zaAJW0R`k?Vr7{Z&6k<8z6i-P>N!-s@zmP^xfucVy-cliw+)F@5 zXV!yPXhSlP=Jnzo=}O?Z2__ZgzZIIqMz{%;3dOM_+!&3*V3)eBZXZL~XIK61Vuq%E zW38Rz^*eevzNg(BUk^|hUHE?Y(D^hOwI9flif7myl6`vx`{J zR8jUX2`$ae1h|(ROUU{fV6OI0xLk;Rhl7zP2VutJPa=C(kOT!Ua>7wj+-@SH~9y>qs6QI@J0dl-%@?~sUyJME zLQKT{qv9{Wa+8_6q4RD&hr>3hX{SwBtoCt1@@ zT_ow)-l>Lv*1~5*m5$ebFPE&KPDJ!No>) zHn#riZF|Rw_*wJz;KH-V+aE&P>E@-<>Zn=|vc5Ts_d#DbHSLRCCGIkZJ8zJOta0y@d7E>=O&7= zAI;^RzNG6YxsLK`IxU)cxbTAbxy_8HX_xR=Fx)192SF1S5(;j5k&`q1{xdC~g;LjF zU7RJfWh3jja_4Noza7O}`-alGi_G!+n*BEFCoVDmm=3|qUcQa{V{k$*7TR5{tk=-R zyC{B@9|Lq@AAhayVArWzv}#wr{nmiv_Fa=tnV%76tN5Clb;FmhZ1^K%mdt#DNW|zW zUj;?O zbK_v+cz}HUV5j+=SA~CrepsNa}<6_!A?`xMvu%!xvTlQY6jouhV{Fe2*|84rExeB4X@jFLrtBqXRwNzv9!)KiUqkbE zr2|F!t5>)cLYeuH{EcUOB?37a9>~QZL6Q9|UdPbYE?H_MbF}qG7~J^ox+G zT#l}_Nj#kYO6V*z=??nPYzh_UVdMyr_M&b@+7BG$iU$vN=PmOfq{t@s#s9`WJ&Xt) zwKky9!{~+Vmei@^a;h9STMgX0+aJrmQEuKJKN3uQP02oZOjv61nRw)@3|^JB#76o# zz&4mL+8O1Yaz`|Y{57n}lMvB3u^_$jbA0|MDW&iM#emO=k&isK1h^FB0(aub!QdvB zbg}JjlLlwwiAH`~Rk+N0kNf9_#_-?jx(g1L6k?l5CTgXvgZilbnTHx9!kuBGW~%7N zP3G|p&tI&$pL(<9%ZE$(J-+Bzt~KI{AurB}>X-a*DrKZ|_s7Y1Ugqx8HBmMx#9!-e zecVm8AFUe$;z~lzPq*YGMMlM9YyjVfR?uNa-Sw4!N27E`1wb^>?{IrG+QHoI!a?UU z0M=>HXs5ydCcY-^t=9|OgH>*Sk%1oC&inH3Rh6pLEG(k+PP>-TjHVy625<6x2@Unn z`R8~hIA5UEN>HM{F!Q4uy|eZuP)2w#!wUM{NAwET?nSI42YC2y*~z-ArGLxol-3r< ztSZdMM$qNwrEM5G#g=;q+CUzPxa$)kxq)=1beNM8Cs#VFH~aflx`V$_axWg znpSF2my(Bf5IS_4QGjC>WoWGz&FUFSz$y?!#w!!^3QOWKUDdxmvt?@WMH)fnCFiEx zte{hazfg07b?&{>x@wn2rx}tyZS<>*6AB~CXw)=RVJ>g-~x`;qDHYUn_ z>E7Vf9M?Pw843>?7hmbNZ=wH;;d!~Hx-nG^cKOQHi{hgzZj^~OTnD5FV%=}lM{)Pq zbhbwlX&GV%jlmn;#Iz1$W@yrvk+JGlNT@#*kf%ER{i!v!RDZx2IVPjYf|BRdv^|;g zP``5_x9jnwZC{UAcmZ=}isX84W5l0Whn^ENP91o7{CJhg&vx3FNEd1ic^-jl1q2*3 zc+r9FQh!ATz93{BSv(KbXC|G_Rz<-Rpi4wBc7|anwQwqJbLI5Iujlb>`B;^?7qnlWqWSddl zJ1a+R=Yp)9lgk+7;qjtRFWJEsPr>bC!lH1|oR|#9f$$It6c%g@EIRebo~mDyvQ-9_ zZ!K84x*)v4PSkd1+Ak7S2Y~GBkQ+K|D;ETvRrPO`9ycwTQeEPStt0{Q6&YTvZ8PDg4JX)t+sRfvps_HVGr z;cTUGvI0@a9xyK<4dB&<9~c_Ce9}haAM>io={|BN8T}Viy5fztQ*CM5x)-)A;)Qv{ zCOuVgLm1UhX8oU@b&4QnzeQPX?H^`s4W6Hz{`$J}k);5cJf|;;`OM$&Z9`TnG<34!YFfW*(_|YedQ1#JGyZc$Ldd(87CV~F?hS~n0eQ1F3W!e6V3>6#e~IsqNcRTl#1-V~`VUN_)dxIkek(tGgZ z{?+Oj7BSMZr%QG)GX0OP8|8lBG!kdL(e37B&r?_IG#;-PXk z!)|qBg)X=?OeySd5&=(J$d+O6PM5uvHvR9gY%$nw>bshrnQsQN`ji*I>=${jK|Q0t z)<`%(q(CMB8ds?L<;o>%q6PgB^cwj}`Z8RDQ#%;aV~0RUN$YIbulir%oh-2HY`;va zMfNUn4QVesv6vM+RTLk30hoUd=*eb&^9Fw;`jOM^Cv|+c=p&O0XntAwT+d+4E zKj4?~6KHVMVUrKb5z%8z0L)XLP1lbY$E*HNz9LSWh=*4)xcp|i+o_5Conq-@U;l+N zu;!V3BvHf^MqX52-!iOCy13WVg!lQm4^GHQ=hJ!Nz0*m#PpwUskEDM?Hsy>Cy=a3A z-CH@i@Kh%^2m2e_Ka5PaC#YBi8HoUoQx3nl)7GWjJM9{e=CSjdGYV01u!2Z^xY-kqw$n1C3EgH-&LG#3)Crz&}XkPRL#| zn=dM#$!r=|G+!-mqDTEl_Ahs?DbIef45U$Ne@&assq8#0NnqC}S8x@vSBLRhhap$gXzmed!O=N;({H(6(J%d4anHFnIM-(ilcNz(UTj1sL;GVYwZSm16V(=nB2qDxH}HcqDsZJb=zr8rv;M z-0&M3(G#WPhmv8%=St!{TXo`}bwl)~d?F0e#)dv)F!0+|7&)eSEH^RxuQVUlHpg&i zC$D9z)N=euJ{i|TMcO&CXyVtS6<^5~6UX55*a=Jz1HXuh<0k6-WTzLV&tT?0lcd+e zJ|Ox#X5*}51%=$tQV)>%I6Ux)IZJ(C_cquyEBN~n@M zn(-q+MooNT-dXIh;Ol=@%rA6Z{JZu6G|b8dY_F&Hy9)3aG|*}48KswkYfkSqiO#6p?l zHHDJB1LQt3EE;tMEXy_Q0r`JJ5Ov@1l+PR8pPaoq)#}w~?(pa^q!Lqod9_cU_9Z$& z>Y+}F1+o*q19hd@m)8!Io_8U?SK)P2D z#mIQ}0y(Z-?!^p0V7v;z3|iBfH0T|T;GqF)@P2|-do3@gq$4CB83;+($r#6djw|H(+j&s~7Twqd-vO2hgV?67$xr^d zETPBShfb&cL8j(rubbs)olm1q1hec_jD-iiY z^c2FHBqRD$U6fF>rGtC|#SvF$T=!(oJWig5utLv2B@(CYzr_LpEd{VKmrqR_zdO$3 z+kGFL-endW@#O;cDogMi<|jIhMteUxg zR=HwmxmFJrDkR?~vo8~Gp^45l!6Fq=?5HUbpe=aFF$QtB0X$fN4d)c;XH^on11 zfZMq6n$77uGz)4zg)2Ivx=}(j4n744O_DkXJwZ}_&~?Je@AhopVc|zWPw-3gYiC?| zXE!hWlU0J&k-=Fxo&=B|%B73BSEQ%Wpeh; zny4gefsu1Fk57P4`@7(!o4w*&4^*37UM{#eYrpB+JyjGKEE-)IL6+^rLLFv7OQ-KP zLYl?OCo!iG?})`;)6JXDsM3(!{dd$Tl7yVWTL%El>U$&XNpfckhZwMSRaI}ODDYVq zGp#bM-!Gj)7+^m-ST!iGNq|d*0gQIdqY;_G`5Byp`xAIIe?Dp=2U`aYHMu_UAXyHb zzH@TCcVK#J(L+}J(6>kIwEd$#p2>(e@D7W`*Qy50mZt)3|14$_XrYDDU9+gFyNT*c z{JW^_v)CE`I4!4PlR6XBjqe4V@8@W%O13pkCa<|d@(=W$y?}e5PjTQ+kP2e}#$fKU z_Gf3598sAEul@V7Ah%dx%3|Dxqg+3Bt2#1r=58Bx7j?X|%r5$DR76F4_jl84<2Fc6K5hQDu!d8pwTM)|2QB(Mpa+RrTf%^@a_H5_ z(XfcVA34EL1HS3n+InBM*tQc!s{+l=irp6>bO+&zm`=YZqsRM;)L=QI*D4WXR%!`i z#w&H5nvMr^)dQz^SZtCS>s&q!-}k1RX;2jegK#ZE_DQ;{&b8vX`Bg`IN9ak>OLvbc zdClYsF5U$q#o~F1Sm0Y=9$H|_2mqe(OolV{ebw|?fNHXzmARVgxpE>v8xfezC!BMU zz5iz3$6H+eovLs=gw7OW|O#o`WK>7#FztM}qL_GRg03A}`BPvrk#1rz6Z(2!+!1m^eD#k76={8z$D?6q~(RC_=?t^ zTgzDgJ0+mbq?=9SoKnr8=ExkTt>bt(U^2dLIk+$p98Icc4KZBANXzjtWr7QB0V>f0 zd%{yVVgC>4Pq{E`@&e*x0~Bm0pr*o_UL|irk~NGibZwW`eSe8~X=Mea04gSUckoAm zP`}V;Dbc%Q_dR_Yb}s#Azty+(^>5b}9W>I(cdq=oI}1Og+q{0Ju$ga_so4NKSXuIt zgE?>7(Jhi?=tCH|HHd(`xCkJGB}WoJYq9V|KX*;0rXyx9bDkyr2(Tb{-84%Dl&j53 z!NF^pd_~VJN3+TM!Vs2t$}Nz5EZH6;q64BA^9Di;zh%iTDa)7NKu*5@mGY@nD0SQD zBg-c=(jS`dc8{1HseiVWc4oCXCU!(_HE9Y^Im!n1Bl!pljDuSP$Q6cb$k~Danr>cg?Rn?@DlmmMBcqEB7gz(7 zBiUmY-qu7uFHLW@mH!S(bG_fVx>>82^h9T@Hr4!iu$JreZcq5eY?|+8_jT{`g!BnY z3s1D%x8c!XJjoI~Kdr>J{gGaYA5`)#s-rnRC{I#XMb6sMnp=QWEyhyuNVKi-!RPkCK=Yrx3P~P-pP6~*8tB$|+w{NM7MK*WU~CCB13peyfZaCmw;!ggmZQac z!|-VBNCtV3mSd!-#2Bq0m+?bd{i29jO`zx~xxajH=e-SR#yUU(MkG>r`T`CX^yRdd z(W~7M2woZAyFt0?FEALfg*~8nET6Yl{xB*ReDf_H{-{Fe2%H z-K2U@)G7t9^?qvffAehyFC1?mr*i4jgF^UkJVm^{VTBVDAr3q@2(kedT}X6|ViS0k zsfGv2+B*(KvI>HegxmE4)(ZoG*yMG82t2bn0=Q;zGQHOZBqI)4hyA>U?OyekdpQ_n zg496qRH5$)%C(^qI(OdLIbeF*qGs%UC?N3hT@utz#zn#EO@*O|W_+Bg3DC$ib*JjV z2i_rQ)B2po#;S!xc8%2-7qHQSsYKF{!HC96U#``|RR*9anxdQg;_aSn-|%tL|BPkM zbquig)D&0ygO;P)lrEjvY&yZvdO!HAQqIwVg;mIMC{=9*n02RA`OFGmGGgL29bT%8py*I5s8+s`W!8KZ=B7qTjc@ zYl2k0clQdzjX8=U)0M`?VoW$PwEtF2cGLZGVgaXGSkQJ(oz@T%i1*`hspBT}-ofG1 zf}CNdux145C4OqvV{Jy#Mheau#A8S6v_aIqd1U}4H27l|!0UPkVAGoQAEsH~s-k32 zk;w1~$Ad_^=LDsIc6j{1lKJOeJeb~@WY2Q&UvDGTez(qbG~nHEy2g{YEg{mJ?M543 zpgDmw;DAx68vq}kLzsm{`6W0xvLYwGvRt6AiN6tX{zE41m;Nv8f`fK}8BFi>SG>r0 z?PEzigF#`zR2~fxjJgnP8B{y|CK1z1D3EO#B+K ziK$sEkdHdx#v+f;p?Gc?oybk%YjuhWUh7QoO&7AywRBkGL{88!FWgj>n-k|9L}*&E z2L1lk>}&Tn8ldn?B<*kbLY^u~Xk*X_%NZ{qDW(GT~=07g~2BKrfom2$*Bz&$;ssuRNUy&s@ z_W^9Og*=iQ#*|E#^W1DWrd*>i3=o|5Kc416en$>0j7P6!x3e2Gl-to0YECl$>bv7J ze_mWM1#HCL!Hc(~s#P4X%;<~7SG_Oxi0849GH7{B0<`6^dys)M0-AF*f6*PBMVYYt z4(Qvu{>hX!**LqY-5Q3Yo$Q?lV(rq$HriJ(m(G%mru@Y_)G`G|)jW_Lsw(<}y0+Y- zL7>;FFXW_O^Mz6e%d6_DlS}+4Tl&rNBd<`ytKEi#eSCD=$YTv>3-KiaA(Xt=54g?kuT#?xP zRXJtCBznPm$iX0vPq0nYb}K<;gW8tMr_s}!rHs>m8Teghu*btSz&}R9-*M7?{cXPj z9*9%}a2<`-{`twhpDE7(yoZ&6X~X&C3@E?iW8e^S6a+B})<{-67T#n4a)*3}`1=UZ zNpOj%x0&=HE4QOz@EZ%fQ1;9c~=E`#=8I1ArzUfkD(~zv)s9?FH>$aqogKcg`I)s8%aRQ^tO=kFD5dP z14|;oA?l*)l`5ha19QO3r}8sxxjLt>6*yvi06~R#+k40vPY+D-C7Df2!B0?|B>7i* z<<5(m4J>sohC+nk&`mmh8jL~y&lIH|CHgazsOAWlucAp(LEB6yF&qRTqUs|~Z1tx; z?vE>mlvJ-W31AH~5|Cy&Ei1v*e`~+TZYjxbt>dwjhf$Ze{>ekddLd3iSYz-H1Gjgx zPZATVu=$!0xY&*4ww^3QhW#fbg*-<{0^0CGx_f4 zv1jY==sS{E93)yt@AkzUqTko$GZGtI;usN=E89aUv69o-Zu(G_$6Q3snAVp(x*RR1 zQn)!7g>hZNpL^sU6rWvH#XiG9inq^9Hxp}k1nKkCgkIVtdPmz#-*|We{cO+AO_$~3c5!Zag*Q(kLvF$^)_iP zfYvnE`$%MjQC&1D!eu^~+BZ+wqjnC!9CTLv?mkY}%PV&5%zTe{0NO{AonHoE8a>W5 zKd=My>Qst3L>EzOBmrxesbM_1QG>4b+!GZ*Nm})!#R&V5jQn#!*qHX_bSnu6%{G62 zE}{&RTR?wC1Ea4spJ!e=kKBY)O@1Bx)04$0D7hx;AgB<_2}{bRJTO-kjK zYI}jZ1dei)lM5k80me2rCsSiYYlq`fmy$CL#&$4bjjq-Hw58)~?4az=(c~`l$yTw} zHG*?N3k&NPT3pL?5GcRDmJf?Dii=X>4A%c`JBuAc&?#H_0Ar_l86ok6eC7bbF{+@D z2^N)l$u3%}>mT5(nmXebqd;m%neA=k1=^Q&u~>%H0l(o>S1|UM_xq5|KL93>vq}=J zj%E}g^+|$3XL~eROJZ^8#BF4APEoA@eSgBux6%E}!z)=N01o{nKl`{A_i8R8n7i8- z{I%yfCV?rp3cdva8FlZzZaMP$+YN8H;ednQ>Mw#cS)yugIAUagV!MW?=Pcb8hX8pX zW!iP%3Z7*Vu_*EuYRLa}w4c-ozRj`nW(5|80k)v36dNp^(2F~!oOdzt#$P26?ay&m z&@nd)JWBkDd1)h1v34!X+HA-A`A#&+W=)KY;1A3Aym@`?*|^CS3CH1+G?>IhajD}r zol-_-fOH*rT?!g}*Q8f5toAC4&Yq^`DJCTTO5( zJnKAb9}tzXMaHM1WtOP3o!1oeD17$qmvzwS%rXv&sL?1f3gXn)tn1<>Wa2Sr38w;o zu=86CcfBUeq)S zGrv@2n-^DbuEt+a|B(z-IhuQ$S7$=l%>9+T`Lm=MFSjEVeS(fGI#gtYBq7~kU~`uL`oW@yAhF8x{*$4kZus^PALKD?k=Uf;Y)XScXx-> zJv{II{l{9cSZ8L=oY{MSikk>^y-?;rX$Onw?Dy`Zr0kVBqbhZHG2RHikE z4Z{4=i6+t`#x&UO?+>N^7Fz_H&4S_ZSL|@cp${@F#jm2dI2E-dk_Hq!L_a3}ysl63 zf;R-I?qfzRS%^HjJZA)Wp$0US%rHB@!^F8Le1GV}&5VFB_gHHYvF_Q^@j>tFIE75s z`gN~pnOw6jSKb+dB!){P1Ty{GX}wl+{%#u>WY*c9PIfzs7l3rIr`0clf%w4K00;kV z*u1wNcbLeNQn(|imc(SXzU)k!XStfS!{52~VAo$bIHP_Ecr)j~Zv*qj6bRKV8NW24 zKsf+AgKTG>uZ}%4eX`rLN7>)9?M4~usMV1ACP!rkZ{XP=A59|ADLV9@L&sn|HG{Ac zgmZ8g-f3YE5*zN7jqBFcJQBiSlJdulNX5{if5gic^dDjheH+32$~)V;D=m?dVhVJT z#kzHFkKbCIr%Tj>KipbM>jaj@4Uux_i6_9d3F$A5j260%CSEY^uy>Oi)iRsuLknSfY%x@)>+weubQ>e*P2{4c>9+|;mx;@ahL z&d{H3+u&%Wp0JP3aGK5Mgk8B1CvRy{O=Ipvvos+t7TXPjP!04Z{jFM$dCpfppfEg9>i1~o(PlX&}=w>pe`y(L;piZ zctZwK9ne@Q67<%kx_%-=q)J>cEWDvw` z>^0r3dPRGP=-@8o`MJT!P!S+GwVl;k(d)l8Hc%*$v)G;>tplfNfWUk`nA0?>+VpY>t;haElhsq0qTqMlFc2zmhLRg7lloEW1>-QfMrpywSiZ2l*&+{l8l12>Cm8M+6ei``qS$j!S4itX&5c!;Gjn<1H@jOMEs_T(Fm$94}`m<*`Ex1 z(Ig_t{Rd8**h#HSx0xIr;CR0E+H#GQhRB^3tzYaGAUP6gOaIhWU<>#&@8xCD_F2j} zidnIoh~UG8@w3}bhME6opBevo2dhc+Byi()t=h?>LI*rXv@eGwXn@eOVfHX>}-U z!-tkE=wP_bAQm}yR^DBUUfgI2lV9eT?lH!?O_^J`k)Z{yg4I5y6y%G7~bQsf>l?zu;bdX6W9`Wk0%_gbs? z#g^;(D0b)*@1j>4@d*vrq^O1sS-r#op-Z{7V2yaivAwTcUi4?QtXH! zS(E9~aM#8K8~ox3oN-TG5jWXLCmTh_ktXF17WS|gvIJWv6sDqIn||>cVnfqAX%AEZ z%f-)aNMV3vnr&Z3q5CDQS&%GGV12a7+4z8e3AywP< z8`zG(;F=S^XAZKG*A~4nxe2W5_Tz+LR|Lzz_HZ%iK?I#!!TJ0u|HODGseUoV1_o7< zD^FYkbjCt|L>x)bzS?Jo1%-Tt6%K!wT4E1^ehQ9)2Vi}X zQ0A3-N#1ZwI_Qi+Va_Ck*|+dX(@MyzueGIE@l(`tR!$8``ZnYdT7^XKT$X@;EtRCFL+S{v$A6#5@H>oIR zmKPmLvi@j2)LmvH+bi8O``X9f!9Uk(lW*Pq)M}9?rSiREUY+3}Qai!R1Ev@Ee~V>X z|9%N!@Y`Neg$dqk^a$!I4W!-{7NepRqVaFCj`!l^TvZVx#xik`j0x`Ek2zCQcn3X#|75MZdn4WXA zSkU5Xk9g%9_R0UV9-YJM{BLU_ts1`pZp0xUOgY7HU*ZDhS6Hl9of8)C)_=J-drUkB zs@{Ze*2$=OS*r09F&d(A>9+08l%{Jh?1s?d2z93Hq<3Q*j?ud5EeY)fvn+teUlr{P zCj1)LP4;_H+Rq#t1h27qdy8}ng4NP6<}XJ0ODRMAQ$^o~5#&k)R@ep5{bK=4P8*@^ zvVM4{b0hDKSA8P#;V2lTJY1@MuSSwSOTmq>R$?ZH15A)dPEHvY*T3-d}ERjDo2MiRq&C%|zYM{I{=;`*xK>G3)W^{-pY0m>SCGS0Y`d zN&qrJ-anmgbG;wvM4Sd6Gh#;OJMS_p6N*^ggkJ^Q=^sNuBSJ~En?k5_2ZYlDka2j- zmW$LY&9zdoUB#sVAV=x2KiNQGSUc>svjPI;C7K1T%ftqR{`#(Ulvg{@yrZW(`(bDN zgv)N2flMW;`)tsOe&;@I$jMz65QiVm9g1R$0zU^va!Q!p);~J0yq?jQUDI@-2?((L z`=V*2fEl`dx$bthLptyEtbwbJUhu24b?t0TgBfpT%RsWlxYBn?)D=i;B`16htP(h*RNdLhTlfV#>*ys>d#kUsWa32**~}yglX{b7@}N6 zsjq{fsDGi+GFUKZj~^V;6HA^)7!LpQYm#qBP%2W;E|@H$wQ5!sTQ)n76P?m1?fsxU zAobUlQ9LZaqR$s~>5==1IFs%q0Id>u_R)oflq~U)WyG|H8q#Pm!#weV!WyL)W`Bgc zR@snbjCdU@G_rtV`0|8-J&rN%Te_BtO>@$pryu>|ZLcMN8!FH$mgDI{8!VZ`)Db(M zLb*Hy_2fDSkt>wjayF0bw_GGm+I1FMDa|}T#};5+@(tW6p=cQv-G|()J(_-Ud49Z! z{o+?KmQT|Li!OshB?6B_AlZ;lrZFqNXm`RJ#o{2@r$d{r^hZgBPPvq{dZj3Dq@{M* zL(9BLMdbP0-S6C7flL!hha{zcP7$A$eQ$i=Bc=|gi<9Hnkjd;&c?$cG6#r38AO>Yc z`h_CQ@cSLCiIK#m)Z2;I$4mAODC@+Shh6PnxqjXZ{a2mincbjJ-z{|>wc$zkB!u1< zb7%9&eay~chGvTK)|65LKf;Cr(*%w9^aDbru@l+wR#pwR4<$-_EJ!hPDOHEYsV@J<1v}6J2aTM0hVjYyoA8y(=UM}IOoxy=WkIROZeF+G@m8wi0gFDH{B%n zAv&tR!U#yE*azczX=+59M@p>IiALRyI{9zZHa#3&N(o!azNuJhQBn}z>@}hYonErm zJDfq>NNC0cM`LKrv=KT(1Nhs{1Oi^->Xw`<{sCqvgxi{MHXAC30|= z4{AHK{T>^2kn5CLa>N-=a%j=ZDE8%&k8`#>e|P-d0`6h6#IQ`k@P}F$d;JsndjY;#|VVV<+tIvd8#>!-Zbsp4ymSzC1GTx4q>bvdwAa+WB7o7Dfv3l!%}^6 z&y~lpCO`STtX1?GtEgn?OrZo_k8ETBQThnXh&0=L+(sj}p(8H0D)*W#3O1$WX9PUi z5SP7pg6ZDG<IqS-cM?zr+HkffAqp{mBqWJNDU8z*;W~IS*=F?+WvJq-3R&#nji6XtNsNk%lYcZk+uy5& zAe=Vg;r);xC)c?%riY`2(yjT2M2xd=(x;31iLD!wrNc6t2i`ag$n$CQ!@oJKdq_8Y zg=KB2PBedLqON@Vd|NgZ`9T+K0Nfp7ZhAD?#=n6vse8?17%LdW!8qveIX`ApqCUu> zJQ5#Xz7G=Y9-Z+U$g)B>NE``L2@b_WbT3)t{FGf`b3&&Ta?~@LT&_PLGD7)@+#?gVItrL!9>WIBx3 zJ8}1=-ux9OqVDqn28d=b?^~wc`L_DcNZ`JAvRABXSi!np-kH^-oY1&yNFC5(;M0L? zOr#>%mN;_#OaV#_%^@0ip*Gl=Z8jk|SH2#U5Kg-r?jgNqUDi#O3lv7uZ-H)2Q3_+TZ|#{1hKyW1aM#6`&C^`3hm;$$X)o zEL2w7&Sx4mExgeRYZz3}w>IQ>@(hb~0YP866NRk-4jq;D%ah~lZie{dc)t(|A-gLu z%(l(*V#oRV0Qec3$w$d2aj?%&oBJaRkbbKb4~iE)j%r@0>Hz!>q(gf>9&sm?uZvWv z)#Fz>P7)&$%~*7fY_jBj^L>(~?X2N!sht;WjzRCNr$U)2)Rbq|P^Z4PKT{2|Tv=NM zSH^vn1n?4S@%%JLp-^#n!Q#0sJ_S1$9twdj9?$sdkt;uDQcW1gC+zw;%9tG9kSa#N z=$GYJdKz&Qo?TL%-HOo>B96+At=Tt1?iChEO}CQjOPY~H9Yr_;7`+LCr|lmoLjp!X zp_>CT@uFCaB&T$gOU^3jm+D$~-rx9jJ}9#)_a;$mnJ*DBvokid)tD~O-g96wIYycC zp0WhbFm#=bc5km86ccP2#-nP%Ct$t6E!->a#Tw<@Iv(%&2{~!gM-mzG8){<>*r1jF z>wvS1-K$+vA2E@A2&bFcQ28I^8!;5~B-fn9b^5Y7zrAGEeA~ zEBQucpg8R2{!SRU+G64@*5m7h@3)<3av@PFsHNQcgCE-)+Z1NO;aT36Csr$ptA%^x zFKzaDQ}7JyM*%3aOztWGLpA7m9-f7ax0c^6+ZbjKl*4!8a(mPn$B-F_qqtrl{&n4! zpYQa;NU;w@;T_ElC*|FKS6#>gsG9|`*2g)U z?OyjswG>x6YI;pi{|+?tgK!9Yj;>YNtD%{*v{+Tix8CgpT*oFC#t%%%@kg>gHgub2 z;E(M=b1vmqrjRhpmoVy3)|P12cohhu`5xjl6ugLbh#eM9?%whgXDtiwIMr(oc;mxI z?-ZwyF{&1m_9;@UK7X(1NLldKshH0c|IJk*F0gkXcZVGu z{Qxq5Fqj-Eotrvqmy&JQiwAbL*8R{KzR)%3YDRb%%2S^0&nn2?9OZN#%xAfrZv<^< z6nugwvQF3g^&5)0#pzaX%DMSB{TvfJZN=bNSEgNKj`IC>)Mq#BAhIBj$Ut$f=L)a8mMo$NX9G8V>^3c>E%9D+Jv$%Wt;CMA?~t2kvmBMUzy0 z$A~b}bMi$Qf+;kX043RXY(g)-O=eSJWFI(UGQGnmYxF$&b`;;Rur%n#-Y3=tErt}` z5Jgo~sS&}{E2lH-T%JD58^Fw+FZT)lTrzoBu?`K7vDL{6-Oz|e=R^SpW>nfZqNtGv z>7L~LgnXosWU_KH1Z)g0C*}qBhqE3^x8b7kmivH;gIntO?`*TLkLDBpcI3<=iYSE# zE&HPzwkbUM#Jte!2EyZ(hAU(cH9bw3y5|UbCX+HXj{5f_MVx5bPX&nTjp2;k`olF9 zO)q}O)6zw6hak+$1lvW(#7wk5rozO@nqVe)LyRHQJ}R;H3YA$R znk`g&6?ayT)D)uY!@FIY@}8=nhX$2DMMYPz*kYXN4Wy#A-+lE&uii zIma^n5$kPfXvMTsS+zVL4=Wz9RuN+a*Vi;9B@$WuT^D2$*%?;5ZV*C=df562kj)1u z!uiNuF1DtIM+C5l4EUKrFv!%Fq31cmmE+sb`iF!{>|9soALNV&vQ%gRgC?}xZE+<- zFy#1}T;#WS1Jd5+(J!n1Y2WWcVL#QqrTFpqFv!d%a;J7+jMv+uWKdKo6a3EP4b!H- zSvBW2#1xdOjF;17x_WSl6kl1t@GY!SB$54ybi&@(Q6cm;=mFtUqW(zqI>QPv4Cqpv!KKyKS2AT9Gxkm46Xr6T-{+mfd)*-61noQ1=;P6IB9|^p z&^_DRRl|pOi;6(5XKzB*-m_WK{$a~-tU+$m9hdhfJW*eTS7_7RssoiiU_%v{Z)8Ec z7AJpK?M>uM&AvI86!>CqO15&eWYix?hU7OAcuJ-_5>K~1yC#cT=$|I=Z7Rn>m`WeG zgiH^GLE7}$yq?@07`*5w3|Op{NVIMbS9m9V+a{4Rx_|j@=Kyljn@*vTTe44$_PgvB zo->3znrsXzqzL2`Diw!_Q6QRk59qTrA)ZT~e~@0kI-}O3M}i+i&&2j9Ib9kT4~!=@ zcq=J%;P^3K1GK?WUw|nQ`O$}WVQ?550VTu9WHgN)n?gAl1TmCe>{NXbZyezO5#AgBQGW95{5YJyEOdqzpX4APcI8i}i{igN@<6LS96gZ-bv9W>h1(&DPIL zym>8!9LdboOpdyHOdLk-m^_)}0wXj#lEgkd%Qu5>KV^geNh$IhXWW7+)bQ*FDeEcU zx=6bku1b*_8IWYe7tN(B?JQf53Up#0{=_Lw|ByQnFR#yKy((}NM=pG207Mp_Y+~^9 z=MiFP)(=~W>4w?iKg!#7Sp613&XzW2_+?%h%nfz7SY;K<)|n$7DFkR=6f5o*p^s&E zZ(0Y{B6c}&lmRjggwE;&NhX(jLNy*)(S2J(;VtE1vT9h>6iqdi72yk|E zPdSYDK&uj+pYEwq9qX;Y##V<|D}iff?1%?XWOkK{NShFP%!8M+l?WHlZN>0oDNpHa z;Oe`%;MrZu-}AXa32lVsdq6oYGhdL42cPg7FC>1v%f$v^yB_k|nXM$mTHQ)^uPOTX zYB@iT_dy=)(M6bqahJlhL>f=MQi1l@AicUiCjGafL^jgL05Nm{Spgm`quJ0M`N+@) z7WmY|#8{Km^=Tm#IF3>aCB^o_J*;&sr7D)tG>_361!~bxm3$uMKJ4M zDXmV77kh~vb?W(bX7V;V{9$utvg+1xIyRk~AhmB*f)zDlE^<2gTg)({Q3q=TiHr{T z0~=IsYKZ88&|O~M=Sn9hn+UP}<%|0(9fiI|4dUj%*`KO4&+x9f z;)k2(lR|?(((XZ2YV+bo)q5UeCD^R}zfp#q!)Mc>t%IOxEzs+4+D6HI>Amha*b1%H znMh?R>B@VuMAIRq6CG(ky#;+py{5q$G2HOsXxmCin&ACLA^JyYIxqnJWhr!zw1;h9 zY3+e-TADXVly4*uHNXFU zk11uwP(u?sbYwjAV#|NLtD3?J&qAbxQjzv zm0T7S{OqjuXf8GfiYb{E9dwyt8pe;Ixj5OJ%`Q1RicFxNJ3eBF#$y|Nk&I_F8*j3C zvm|uXP%~Y{h?+^e1Y|truRf9m<1{MwFI;~g=v>mxB;j)W2|5K0I7!9d=hVHjS%Atby`gMrr*|qEv>1~^bN|~*m8^E5s$!)gJ@s;bT?EX*TxI@#3VyYz?k88vH|#f$P_0uFRBtiL9B^XR3udG++-~oSIy*y}?|c>X`IK>;QV;l~MF9!8Sbq#77KY-E z2YOCNcBVjNKzoe~Aa)f``*;~rKe7nl@132GG8M*Hr7<{t2@8ZW z`MCt!LcB)yakRRVS4u}fL;Y!<=L1w2rdx>O*P|$~nczi_uZz+}YYPi37btbnc zlLEQ9E={wExSbKpzq2w0q9n6hsgOsOwoVwclZ@OtKL|)9^A%45;%$j$y*__c(Cguc zmpKffn;xlHotWo`9qp+s zS666*RX%-6y+ciHuSW+eJtXt$bOG=u?Vs&(bjMyR)LH8-Rpa5#wsnL@ea?jvT}`l6 zeb67ers9H3Pb~E|jQ|wSp3ksaBX%lb>S>NzA9b+lo0COm9PgmO)nq*snq|{^#a3?t zKFz&Jav*oN^t~Ff%O(0>rN2=lq2b6+0k{&l*)+4M`QnM1VcIOiw+IL+mpWdBapoSl zmIvQ)%zRsWKtiuQl3K-Pcz;RYM~0b*pjqDv)nwU^R1#5?Totg~L~=jz>Va|Dl(ba1 zx7MaTZg{AqDU(Ion4ojRnAZM#UwQZr0Np!mC>-aq*silqpeIIfc)Fw;4xQyJKDR#R z1+Rn$M*5M>ZjhJW3==TWjHI_6oA*Q_S8TQ7hk||!)*iuR^bgKwpg2qLZIdyotp>B9og&B>uwHD4#G1wZupU|wlPU7TIfOQa>j#6XK03-MEqdNcMKyKfkV?^>4B;Jw zz6G(P859=$D(69h*BqHOPb3w zBxif(4HYDliKA&8?67BIRg@vjX!au1Qfbm*nr{j1rVZk6B#s(TY@JuUCgp-!-&r4p z<$5k1bzy@kW|VpL2dOAZ)V}9PvIAu`x$ISv2zItGmzm z%7F5R-bfvh{GY~K-CZ1hcQ5EcVcq&Nb`r$84<@2xeRq;8HCS19^ML!SV{N9sB@>jaYvuG() zESi_@;SSB{suM%wgP^0|B#EY_GiLY}vg z0b9#n#0v099_sx;R!lyyBTVy4GTdd}{=l>CdyRjW-+A<7!fN#=$LH!T@Fp17t+$_Ex`jd~Fm^#JZT&b?&U6*=2N`azJkAD%CPoRntsiM`g zR`>?hqk%j8Bdv#kaoz3Stb2Wba09$G#XC9;fh)Re2trH(i&aYA@b{?#BYQpANL)_a zoPUIZzubAz=4IPyBL_8k)VGUM)-Wou<;G+SWSi$Dx00}?NMqe1AlaF^ojGKlUb6IR zpQW@W#%!o+4#QK0z}CzClDOj_oBLSljy_BoVgY$fsUv8lV7u4qsJ-b_aXDz_I^T?z z>mN*HJqqB~&Ny37by^;aR7R7&H@PEEkx;Yjfi;@Fg<|O8t+}-yO8JS;GA!`V>+xP^ zK2U$v?Tb|V3*|Odk9}i8!qclxqfG=vHoxtk?V}5|=ebqqc!2??&>|uwpFC$f-&NJ( z&|Ze{Tnw_>J^@Ak8XLSKaAT5`($*ZvFC}rTw`{_5#1M5EOAVO@!-!nBXW5s=-j(sZ z*067sZ(iqMAVWQ1C4$vNb`G#!ALV-x)wD<{y{cs$<{aFk2mZHWfWtLDxG}+Ojiy0! zO98cCE(jjF?jT~6mvBC=--$^mnQxfg`tN=e?OymBK3YC&-v=SH;ae2Uw&LCrk3}(k zaYZ9&usfs~)M=9?l*46s*pb4Do?3|Sy(ej+K;@58dj1wVQ}9oOWL5SQjqd9h1&&A(r}a>?noRzbVkH)~n?7E@76b4R zvdfQJ?oSkn>hfs_a63c6`{T=NlbP2%=h^;%-a|;a{TO%B3ysQ`f#Z3>b}|6f6XrfV z{n4U#J!+g?tfvvywVjWP9I_3HrY;#l+N(P>iW(+*UAaUf8jQr?fLaCtV5Kj>PB z#KjODy;8|F78&~Wc%bu~e?vA$dw<}LH$()1!|MXFeoXpdu^4`f*|Ns0hqND?KIwr* zVEy_}{fUyAxOy@2Hhh~$U}w!&KWp~sdr~}RiniBvleqC@FLS4z+p=1dfP397ol&|f zcwpQ9fHuH39FHT(pO52V+tbBw{LgwRl%%mqHH}f3M}hv~Yy7llBlE`ZZjz=#A|X?9 z=TQPb_Te<%4?yV|?|yf2+x42PoH9fpzyWX%cUSmhHpw7z)4R)%>>PLmdITsU zeYN*stfCDYcDQkl*YaDQLT`YbF+vj{mVkRvbw)-;DkVIDtB|N+rDAu7q$AQB$t2da zswT!+Vq6h?f=+?pwikQuc~U?|J8}6~7r>`kzk3G%t={45CZzV{EHjGW3Hgkhq4oL( z-p_R9(K28&5S@Z&d;bSoQ!CQD6OM=4>y*=G#TFXz*}sJW$^_jj1_T%XzJ1ox-WaI+ z9`c};Wn4Y3KKG&ttY2F)eg-S}6Uo#qh4o%r9;6Y(>Po<3`PJL)6qG7wZ(nKgb*Q5M zrYv$TR#oz;vRvZ2*JD3r0TqsSR{1ejc!WFMe)#{scxZHRJ+XH_qYfnPNW`mKvA6Go^ zD5#ND&cAdB3;m$OT^TRa6YfjoZj??qf|y8vH3mxdRg4t=@dWuI*B!^$q#^+cM@t#PUasFzEznXUgraaY`|br;DWR$f-uO4;-l`i{bn`Z*5bY+S zG;%%D&&4;otoHl+Wbre%bloF_YozpZIUqpx#ffhLwf-BB@%u~`e^Wi*9-)}P4sim{ z*Pj(ZDw-iQPGf_>zB#pFd!b|@!3X8+ZEU~9TD!jykf*&|3{wL!@J8Z!uvXZcJfMgl zIaTI}F0Fh%Ja;GNYw?7zXxB}>y=h4H zfXOW#23HOIn%mV(K|_-c>6_H9h%x{Z&7n4W$G^Ou#oD27cw>0+LXf=ctf4OCLu^d` zZSn6-dPzpn!v+lBTZs{CT@sHgk8?TZgFxPx7s0z{FLFDQA#~kHnaiw`Yx2O=`qAk- z1cTrHSHw@lev#<;ocZ8{Mo5#k*&C<&s{?5Xvrc;GQ~TdWV6u?rJDg-$F4w)Hhp4g^ z&DJ^P#uUp70kti4qti(ZxaB{oEkFoc6ae=mnU4N<*JFMcPCBdavpIhpc5eKth4*P5 zLe$AD;@j<@j48U|>um+DrBL(H2;wLfMPxd@-`smfyG0lGwFBb+oj-`}AId@IUp;Na z?2{WC)cBiA6Q* z`eFxLwnw>H3*`M&M7qL3o%I? zdWBpA#IrbA_E>HR%4E)ib?F26yFg-t{P_m*OeK=`ffCf%%2o+=YQaJP-qu=P zAhB64SBouG6RmUsF-p>tho5{$Uy*NsxU+-9USv0wI!)GP6uF49yALKJ{V=n-NO%W2 z%~BoPR}Qe?o?l?_K!-z;ne=>&Vfd_hmOD2;6Gz@+jP1eMT!we0#m6auoIg?KzY)LHJ_E5y61t#bGc2QjBVE1Fa4oH|9YnD z9kvSv3s0K1BjMSjIIi0vZ?#O2;Wm9DX3B4Nc-~33!n9*PlaFax!eK`F5R~8P_J66$rDN0tj;9Vk;VjGO+xwDl5xv85#7jfP3 z11)bCkfbQ3K<$cpylrQ&*%`qEzY*de{pr4IWf?H9Jg0WiZsDSScmf=^YLAS0w$3KI zW#q$@_V4NVQ5)Q2{Ni-<9;|w?+jhU(ivMrb39Kez4AO&|l+gwdP>Umw=@~*!Q6PXS&#nk}7eW{&ix6*n8)-Uqqi+Iu}Gs>HZdL>FL zezWD@mF!^!?5KG34A%3VJ<|8HKs^WJh&By!^0e^r*bV#!z&REnR7F?EW8i1PMS)De zIz0T)dsq)B*(4+3RncYRTT;Eg`ojZ z?ZzfZY4*tIaTJ4xhE@h9SJOf}ID3ktKhl)LAiuX*bFj(HH8 z!5Vc=hA-|uive8~gT^`7_`OM`Ye(WWd5k!&E0ZRCuOAVYeeeHuxHzs)4|i8&tqk3s z>rGhqa#j-o;l|~JI9*b){R1_=2b$VAs;8&r-ME?cf6rW%Igj$e**bC(Rlyh=2SlU2 zK-?HYyF!WqA2Sfkn6H0sl#G3AUtSws20C4dAeIq{zqVhqza*|F>ABJ?c8jwbzQzE| z2(G(718@u{M+X9SlYc~@Q{Qtq>??VAQo#mZfJsySG&^DThUclHEw!FR6txW}<8k+Q zhr$m*H4sRtj||U+XJjptEFE)}J7P$V(0z})oR@SmJQYUGPA9h32QUv$-@Wf9)dSDY z8~dZipS}0X+}D^pp}J66=%hP}bC>JeP~x=)LXKRA4}e{WTd%9@SO>^H@iTYiWqrD@ zI9=aNd1rvO|k7xg8`0{uxIjNg)BiON%>Kmr~=F?FFb^1{)%I6({TIKTo;PXkv zdc%%iSJ%QRq@>4UIM=qZZ@Kt}o)W(~#fDnUJycP%6p#NhR6TUQett2SRH=@f5&L1(~t?W@Q19AndSMwgjeiEr5^ zV~pH6wzGyvxm{d0Qyy)Yu3kn_-#r?Dj!SWpA03v%>`-3!At!VFpnEYm^FgLW7QzW( z6v9zC^lzV!(TDtM!g5zBC0imtiL86=@$LqlC=g7TB98*tZ$H`Le3MS#m2^-N}Za zU+PVi;LG>?a4~OIXV~GM>T3=OYeL^;e82GRMqE>{-rlJbJN=tkCsv5c_od8wO1U&? zSR~xl@>+8deIl?% z8>57O2n1u*U4Rjq^YfX`Om4ja^JMB_ypNoAJEO86y>g-|{2ucnGv?Qk&_JW+Sl zDn0%xIp?N0E*EPz&wH9;`UAzgRHl&sCun$tYOfkFa#h$ou$oe+cp;43ugSc#UMdBJ zYS>?LR-09IR7d> zB*0tHzExqZyoXrPS6|0T0@hA9@$Agcd;ZhUU4ph68gFhl;C4nOGe61VY_G%Va1=4% z#DwlqmY9(QCE~?MOQG&(&-yc+|yG*oCAy{sifyS#&hqHS4 z-4B6k61ZEe6&!&!_s9opw&XIMgor9bQSDublFRNc4zvaJy~z5ni43f!JroJ_8m@nw z*nUh53Oep5d0?Cs|)jZRWwm0!7y5!rs9K0&5pF$GfczdCLK!64l{ORj;cbK%eN0SHk zILd_Ac51`|_M@b|az$2=g1ZKwiHqV9vAt4cnJcUF&5IAJ6f3lTidEH)1bKV2sDH(P z&I0Bya8!5n8(%c2(6fP)K{{|MjgU?xiB93P+i%jedUgI&fM2~rEH;Fse)rpwc3F3u z9rp`&lM0w6o$uwR;}=(?qthS&g3`*Qe^Qmv%Wzfp55kOqywrhhi^=)llFrEmaCG>4y%^Y9ZtdszVBeVN;d#Vj+!mO zL4Um2)q-n6m3L;rNU;>zpPJxTn`yaz7p2p(Cd&`+lOsAG=3k*Cq-GQOk-)6sd*4zk zBrw{~`{TEoKR+3DyqmVk-iehBm@aR7NTiV`R#~kK1DCkP?iP=G`?dB3vkmPNCrY^Q zYSEu742S3ei(x{R)9Q8n4G0hXn|Tz${OD6dFs$lLWAHCz50o18T(A3+Z_j#2jCv!; z&P^-CwnG1vCeqsydf#zPFz>2qmFar-NZrDo1$?7K(uo7sj$*{FyMmw5dE7c`Tj<*` z0>9D>;}@ri0O|XLLP?(wUL8;y0(a9HsLSdD5E-rq>6ZSoD_UlpYQA5}W;#k*8va^% zlh5I~vy_eNiYGs{s2Vz^fbc_Xlxj;XzJ>a9hdgpq3s^*f*w%rnB3@xj>JN2?XH*9#+J@7~iUi5{ApYYxWV-!xFW=Yw_GW&uFhcXMs{{UqK>Diyz(ol#WRTWj`Z^?jw8ESM?K?-qVlwX1 z{6V#(3Si1ULsEd0Fjet7%Kf(8GX4Z-xcvFj5!3;nt5fzDYD<&K3Esd0wA-*lTxS|c z7T`P;;tZ`z9t{HwR>}DsmO1+3HY8mB?d8k#1PNvd%*-TPwfvo+GjCScT6jE~(d6hp z#!TKC`d3Y(YqehyH=7-lrBavmT)QR>5(s`dnC0+CncT-7S(WQFyZ{*pufioe2?L6?mYQP5IN^GlJd6 zm`uPmyiL@Tz90}xQwqT&yK;ZE&agpWw*}M2?V%avr za@UiouS9Q#5c3GeTQPMzDnS%VYMnqWCF||sjCS3d#BI?ry~ESb1j+INsl9vM`A= z?fwHo>6B#?mvi1+-`%4b-9J5A8zZq%pYNl&q#HrNmM4|&Pid;{hH(C;I$2CSHcNro zI4aO1(c_WGl^XQ|vE0AM#aBpk5bYtl^8+6+ z5e*NQ5`TI)?Y>C9+=osY zHUA!oNUg=>qQGmu@ucOzuK24(0>4)aM#udzM2@?=W3TD-8%{}T_GW_2YZWr@O-}zE zZD3(`b+`DC-Cc2V$ritYR&hopgYkW9ioM{wFDmsY_!d->ko&$8^pU{T?KuNvFf(@P z+>W+oZaWe8-vXdmy7~0CcWL0o|>w2D)%kJ2*G$K@}qdfO^~Z7rVrjv2C*ER)N~x=E;m~3o|FLQJ|X9NYNR%7nqZbVZpND)=&02F0cE=0 z^P%)#m%h8BcUk~oa&tM&gsYbE1V_iUu)gl9jb@9xI+&&8-^{%bKW~c zuuO$5NmJGSzQW8I7068|w%N5F#gumhlA?|Xf3PBfqBFa7zTm@qDn!WpBVlte zS~!YEuEdTE+T)sZ=a(eRG!`n4HKb|kRA#Qp%b{1sD3QQNTonJac@DctCKwwXCBuEC&GG6xB(p7gG@1E`WXSn+8PSE zlDSKLw`4>djTDsFu?Arxp2Op2Tp@Uh5-f+_Exp{nW1@AD{9r!&!g!n7ibySc9@a%H zDdr7BHU%W<%{~kDNt$9nKN6fm2ovVPFK--A{9R$#!85c!=k>h|=HSE4&?O9J4#3Jj zQ%DK~PLZKSmW&a}VDCW`%ef(td2K|Ff>^C^w# z?@Xx(82*h7JSFEQ&zs*FALY`X&r7CW09}w^V-L))&4+G1fE?>Oop>~Z)f!mL%>VJV zbGwTl=x>T;{p4%-Jk*9kQ%W4A{0&Fq@#a8s-rj7I+^{nc9ZJh51OY7?Y^d+loT_Ol zNwgUqOHawZRHQxJ&-GSmfceK-PAMcFi2DW=(6?VTQdo|gqz6pZhYn~4?pn`l%{IlTakhWc0X>&}Vj}&8(CrdtHL@%}8 zz?7Cn9wX>rANhq2iCKnp4br@mI&Ersy+NxJmuB*#(_qrp_BLjX!u-azC9*vv=jo zmzPnaMrmujAz%i94zjfC)~%EG-+y1VV(51$QBn$9vty~8^K)B1pT9d|vrL#ZQcEaV zb3CafezN7WX3bLlS18_RWR4i+14$W>0?Udc1rY40tAc z3i*2Rm2x^B3S!IxPlw&@+qdmMD`dcRM_C`DOj8-=vuwgA%o-&-k8KQy^@)rzqoL}& zU0c#;OsDG;;(N@3A^HZv4X=|h3!i`fxlEWaL0*0JRV|&QN}Wo(RC*5VA`4c{y8AGS z<|6b~P*bNB{^JRQvu!#IBFg)|QaEaY(bb2PP-N(2H{S+jrA7T>TpurA z?k%gU@C11p#EQA@5^9?*z z_}(fiDkLExLEl%%4+Isme73$)u33dU%_Q#M>;LgUyckQyEz$cQrt9EspTC>Jl9H07 zuC7i71_nIxw^_`XR}$s|o6(&+ccihgQ3s30#Kfp)jCOloJT`|opcX& zOuPLa8JBno5`;0t7M{%0HtWstn`HIcRpM=Jen{?&U8s+LdQYhrq1+v~D7dzW4%H~w@%e(hIor6WusWhEj+B(jW2DA5l3*Kc5Xt_ zv$M11;>C+9UMwcRq`^OUBqYoQS7mtP#tk`t{=8&nW@^hlz6q+cl}|e*DkegG3IwHQ z9mvp;L3D z^+4aVFJ0c+xlL8b@Td(E9O}?cZ&Ve!P!AnCq#bpT;)#g~_m5>D_qzOPNZ=Z|s;WxT z($bWWAsC=gtzEZ9V&kHfG{2LcBp-jePmBGN)kRWnolmZ_>5B{1a>`mqrB%n|=!smN zx{~tl4%w2pSsNf|*zjI%Q?D06;B=xglN3K5o@7J57B?O#340LuLTKyepkT}dk}zg| zJX&l@jAB^owN%@^1MDj#&|$YKlMq?8+NqBt$gWo&ggx54muc%Up^v=O1oY+QjEtzE(Nui~o(48_!d`oOd!@O#S^Xt!`lw`1r&H$va#;KL`N#^pzqQT+G^W6{ z&VsBeopt7~>&&JTBq*BjIWa7lYc`Db9UUF_*O|e?!CpiF>fqp@9{Loe5GAf}!@|N; zwPH_0!RNc@Hy{w(^}B>PgY|DK0{qq61&JfRfQSsN78JO(wY75f>Q(9M>(faDX8)Ty zP;YPFzO8l@$SV^AxFU>{Qj+3bM|g9?I524Uvv(*1jX*aMj( zIF_g&3S9EB4Dp&h=y-)N^8x# { + organicFollowsAccountsClientColumn.fetcher + .fetch(k) + .map { result => result.v } + } + ) + + /** returns a Seq of ''potential'' content */ + override def apply( + target: TopOrganicFollowsAccountsSource.Target + ): Stitch[Seq[CandidateUser]] = { + if (!target.params(CandidateSourceEnabled)) { + return Stitch.value(Seq[CandidateUser]()) + } + requestsStats.incr() + target.getCountryCode + .orElse(target.geohashAndCountryCode.flatMap(_.countryCode)).map { countryCode => + Stitch + .collect(target + .params(AccountsFilteringAndRankingLogics).map(logic => + cache.readThrough(countryCode.toUpperCase() + "-" + logic))) + .onSuccess(_ => { + successStats.incr() + }) + .onFailure(t => { + debug("candidate source failed identifier = %s".format(identifier), t) + errorStats.incr() + }) + .map(transformOrganicFollowAccountssToCandidateSource) + }.getOrElse { + noCountryCodeStats.incr() + Stitch.value(Seq[CandidateUser]()) + } + } + + private def transformOrganicFollowAccountssToCandidateSource( + organicFollowsAccounts: Seq[Option[OrganicFollowsAccounts]] + ): Seq[CandidateUser] = { + organicFollowsAccounts + .flatMap(opt => + opt + .map(accounts => + accounts.accounts.map(account => + CandidateUser( + id = account.accountId, + score = Some(account.followedCountScore), + ).withCandidateSource(identifier))) + .getOrElse(Seq[CandidateUser]())) + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/triangular_loops/BUILD b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/triangular_loops/BUILD new file mode 100644 index 0000000000..f69443cf66 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/triangular_loops/BUILD @@ -0,0 +1,21 @@ +scala_library( + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/com/google/inject:guice", + "3rdparty/jvm/com/google/inject/extensions:guice-assistedinject", + "3rdparty/jvm/net/codingwell:scala-guice", + "3rdparty/jvm/org/slf4j:slf4j-api", + "finatra/inject/inject-core/src/main/scala", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/base", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/strato", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/constants", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/common", + "strato/config/columns/onboarding/userrecs:userrecs-strato-client", + "strato/src/main/scala/com/twitter/strato/client", + "util/util-slf4j-api/src/main/scala", + ], +) diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/triangular_loops/README.md b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/triangular_loops/README.md new file mode 100644 index 0000000000..dcaec22988 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/triangular_loops/README.md @@ -0,0 +1,5 @@ +# Triangular Loops Candidate Source +Provides account candidates based on the graph structures of the form u -> v -> w -> u, +where the arrow indicates a follow edge. In other words, it looks for triangular loops in the user-user graph. + +If the edge v -> u does not exist in the triangular loop, the Triangular Loops Candidate Source recommends u as a potential outbound mutual follow candidate for v. diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/triangular_loops/TriangularLoopsFSConfig.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/triangular_loops/TriangularLoopsFSConfig.scala new file mode 100644 index 0000000000..444fecd0ea --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/triangular_loops/TriangularLoopsFSConfig.scala @@ -0,0 +1,12 @@ +package com.twitter.follow_recommendations.common.candidate_sources.triangular_loops + +import com.twitter.follow_recommendations.configapi.common.FeatureSwitchConfig +import com.twitter.timelines.configapi.FSName +import com.twitter.timelines.configapi.FSParam +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class TriangularLoopsFSConfig @Inject() () extends FeatureSwitchConfig { + override val booleanFSParams: Seq[FSParam[Boolean] with FSName] = Nil +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/triangular_loops/TriangularLoopsParams.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/triangular_loops/TriangularLoopsParams.scala new file mode 100644 index 0000000000..9fb80235ed --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/triangular_loops/TriangularLoopsParams.scala @@ -0,0 +1,11 @@ +package com.twitter.follow_recommendations.common.candidate_sources.triangular_loops + +import com.twitter.timelines.configapi.FSParam + +object TriangularLoopsParams { + + object KeepOnlyCandidatesWhoFollowTargetUser + extends FSParam[Boolean]( + "triangular_loops_keep_only_candidates_who_follow_target_user", + false) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/triangular_loops/TriangularLoopsSource.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/triangular_loops/TriangularLoopsSource.scala new file mode 100644 index 0000000000..73e187ba46 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/triangular_loops/TriangularLoopsSource.scala @@ -0,0 +1,91 @@ +package com.twitter.follow_recommendations.common.candidate_sources.triangular_loops + +import com.twitter.follow_recommendations.common.models.AccountProof +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.follow_recommendations.common.models.FollowProof +import com.twitter.follow_recommendations.common.models.HasRecentFollowedByUserIds +import com.twitter.follow_recommendations.common.models.Reason +import com.twitter.hermit.model.Algorithm +import com.twitter.product_mixer.core.functional_component.candidate_source.CandidateSource +import com.twitter.product_mixer.core.model.common.identifier.CandidateSourceIdentifier +import com.twitter.product_mixer.core.model.marshalling.request.HasClientContext +import com.twitter.stitch.Stitch +import com.twitter.strato.generated.client.onboarding.userrecs.TriangularLoopsV2OnUserClientColumn +import com.twitter.timelines.configapi.HasParams +import com.twitter.wtf.triangular_loop.thriftscala.Candidates +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class TriangularLoopsSource @Inject() ( + triangularLoopsV2Column: TriangularLoopsV2OnUserClientColumn) + extends CandidateSource[ + HasParams with HasClientContext with HasRecentFollowedByUserIds, + CandidateUser + ] { + + override val identifier: CandidateSourceIdentifier = TriangularLoopsSource.Identifier + + override def apply( + target: HasParams with HasClientContext with HasRecentFollowedByUserIds + ): Stitch[Seq[CandidateUser]] = { + val candidates = target.getOptionalUserId + .map { userId => + val fetcher = triangularLoopsV2Column.fetcher + fetcher + .fetch(userId) + .map { result => + result.v + .map(TriangularLoopsSource.mapCandidatesToCandidateUsers) + .getOrElse(Nil) + } + }.getOrElse(Stitch.Nil) + // Make sure recentFollowedByUserIds is populated within the RequestBuilder before enable it + if (target.params(TriangularLoopsParams.KeepOnlyCandidatesWhoFollowTargetUser)) + filterOutCandidatesNotFollowingTargetUser(candidates, target.recentFollowedByUserIds) + else + candidates + } + + def filterOutCandidatesNotFollowingTargetUser( + candidatesStitch: Stitch[Seq[CandidateUser]], + recentFollowings: Option[Seq[Long]] + ): Stitch[Seq[CandidateUser]] = { + candidatesStitch.map { candidates => + val recentFollowingIdsSet = recentFollowings.getOrElse(Nil).toSet + candidates.filter(candidate => recentFollowingIdsSet.contains(candidate.id)) + } + } +} + +object TriangularLoopsSource { + + val Identifier = CandidateSourceIdentifier(Algorithm.TriangularLoop.toString) + val NumResults = 100 + + def mapCandidatesToCandidateUsers(candidates: Candidates): Seq[CandidateUser] = { + candidates.candidates + .map { candidate => + CandidateUser( + id = candidate.incomingUserId, + score = Some(1.0 / math + .max(1, candidate.numFollowers.getOrElse(0) + candidate.numFollowings.getOrElse(0))), + reason = Some( + Reason( + Some( + AccountProof( + followProof = + if (candidate.socialProofUserIds.isEmpty) None + else + Some( + FollowProof( + candidate.socialProofUserIds, + candidate.numSocialProof.getOrElse(candidate.socialProofUserIds.size))) + ) + ) + ) + ) + ).withCandidateSource(Identifier) + }.sortBy(-_.score.getOrElse(0.0)).take(NumResults) + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/two_hop_random_walk/BUILD b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/two_hop_random_walk/BUILD new file mode 100644 index 0000000000..ad7d6805ee --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/two_hop_random_walk/BUILD @@ -0,0 +1,20 @@ +scala_library( + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/com/google/inject:guice", + "3rdparty/jvm/com/google/inject/extensions:guice-assistedinject", + "3rdparty/jvm/net/codingwell:scala-guice", + "3rdparty/jvm/org/slf4j:slf4j-api", + "finatra/inject/inject-core/src/main/scala", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/base", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/strato", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/constants", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models", + "src/thrift/com/twitter/hermit/candidate:hermit-candidate-scala", + "strato/src/main/scala/com/twitter/strato/client", + "util/util-slf4j-api/src/main/scala", + ], +) diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/two_hop_random_walk/README.md b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/two_hop_random_walk/README.md new file mode 100644 index 0000000000..39c9849264 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/two_hop_random_walk/README.md @@ -0,0 +1,7 @@ +# Two-hop Random Walk +The TwoHopRandomWalk algorithm re-ranks a user's second degree connections based on recent engagement strength. The algorithm works as follows: + +* Given a user `src`, find their top K first degree connections `fd(1)`, `fd(2)`, `fd(3)`,...,`fd(K)`. The ranking is based on real graph weights, which measure the recent engagement strength on the edges. +* For each of the first degree connections `fd(i)`, expand to their top L connections via real graph, `sd(i,1)`, `sd(i,2)`,...,`sd(i,L)`. Note that sd nodes can also be `src`'s first degree nodes. +* Aggregate all the nodes in step 2, filter out the first degree nodes, and calculate the weighted sum for the second degree. +* Re-rank the second degree nodes and select the top M results as the algorithm output. diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/two_hop_random_walk/TwoHopRandomWalkSource.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/two_hop_random_walk/TwoHopRandomWalkSource.scala new file mode 100644 index 0000000000..5ce2fee702 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/two_hop_random_walk/TwoHopRandomWalkSource.scala @@ -0,0 +1,40 @@ +package com.twitter.follow_recommendations.common.candidate_sources.two_hop_random_walk + +import com.twitter.follow_recommendations.common.candidate_sources.base.StratoFetcherWithUnitViewSource +import com.twitter.follow_recommendations.common.constants.GuiceNamedConstants +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.hermit.model.Algorithm +import com.twitter.product_mixer.core.model.common.identifier.CandidateSourceIdentifier +import com.twitter.strato.client.Fetcher +import com.twitter.wtf.candidate.thriftscala.{CandidateSeq => TCandidateSeq} +import javax.inject.Inject +import javax.inject.Named +import javax.inject.Singleton + +@Singleton +class TwoHopRandomWalkSource @Inject() ( + @Named(GuiceNamedConstants.TWO_HOP_RANDOM_WALK_FETCHER) fetcher: Fetcher[ + Long, + Unit, + TCandidateSeq + ]) extends StratoFetcherWithUnitViewSource[Long, TCandidateSeq]( + fetcher, + TwoHopRandomWalkSource.Identifier) { + + override def map(targetUserId: Long, tCandidateSeq: TCandidateSeq): Seq[CandidateUser] = + TwoHopRandomWalkSource.map(targetUserId, tCandidateSeq) + +} + +object TwoHopRandomWalkSource { + def map(targetUserId: Long, tCandidateSeq: TCandidateSeq): Seq[CandidateUser] = { + tCandidateSeq.candidates + .sortBy(-_.score) + .map { tCandidate => + CandidateUser(id = tCandidate.userId, score = Some(tCandidate.score)) + } + } + + val Identifier: CandidateSourceIdentifier = + CandidateSourceIdentifier(Algorithm.TwoHopRandomWalk.toString) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/user_user_graph/BUILD b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/user_user_graph/BUILD new file mode 100644 index 0000000000..63b10be6c9 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/user_user_graph/BUILD @@ -0,0 +1,18 @@ +scala_library( + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/com/google/inject:guice", + "3rdparty/jvm/com/google/inject/extensions:guice-assistedinject", + "3rdparty/jvm/net/codingwell:scala-guice", + "3rdparty/jvm/org/slf4j:slf4j-api", + "finatra/inject/inject-core/src/main/scala", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/strato", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/constants", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/common", + "src/thrift/com/twitter/recos/user_user_graph:user_user_graph-scala", + "util/util-slf4j-api/src/main/scala", + ], +) diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/user_user_graph/README.md b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/user_user_graph/README.md new file mode 100644 index 0000000000..fa510fff7b --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/user_user_graph/README.md @@ -0,0 +1,4 @@ +# User-User Graph Candidate Source +Provides account candidates generated from the User-User Graph (UUG). +## User-User Graph (UUG) +The UUG algorithm reads User-Follow-User engagements that occurred in the past 24-48 hours, and provides accounts that the given user's recent followings have recently followed themselves. The UUG algorithm is implemented using the real-time graph processing library GraphJet. diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/user_user_graph/UserUserGraphCandidateSource.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/user_user_graph/UserUserGraphCandidateSource.scala new file mode 100644 index 0000000000..4571f1471a --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/user_user_graph/UserUserGraphCandidateSource.scala @@ -0,0 +1,125 @@ +package com.twitter.follow_recommendations.common.candidate_sources.user_user_graph + +import com.twitter.finagle.stats.Counter +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.follow_recommendations.common.constants.GuiceNamedConstants +import com.twitter.follow_recommendations.common.models._ +import com.twitter.hermit.model.Algorithm +import com.twitter.product_mixer.core.functional_component.candidate_source.CandidateSource +import com.twitter.product_mixer.core.model.common.identifier.CandidateSourceIdentifier +import com.twitter.product_mixer.core.model.marshalling.request.HasClientContext +import com.twitter.recos.recos_common.thriftscala.UserSocialProofType +import com.twitter.recos.user_user_graph.thriftscala.RecommendUserDisplayLocation +import com.twitter.recos.user_user_graph.thriftscala.RecommendUserRequest +import com.twitter.recos.user_user_graph.thriftscala.RecommendUserResponse +import com.twitter.recos.user_user_graph.thriftscala.RecommendedUser +import com.twitter.stitch.Stitch +import com.twitter.strato.client.Fetcher +import com.twitter.timelines.configapi.HasParams +import javax.inject.Inject +import javax.inject.Named +import javax.inject.Singleton + +@Singleton +class UserUserGraphCandidateSource @Inject() ( + @Named(GuiceNamedConstants.USER_USER_GRAPH_FETCHER) + fetcher: Fetcher[RecommendUserRequest, Unit, RecommendUserResponse], + statsReceiver: StatsReceiver) + extends CandidateSource[ + UserUserGraphCandidateSource.Target, + CandidateUser + ] { + + override val identifier: CandidateSourceIdentifier = UserUserGraphCandidateSource.Identifier + val stats: StatsReceiver = statsReceiver.scope("UserUserGraph") + val requestCounter: Counter = stats.counter("requests") + + override def apply( + target: UserUserGraphCandidateSource.Target + ): Stitch[Seq[CandidateUser]] = { + if (target.params(UserUserGraphParams.UserUserGraphCandidateSourceEnabledInWeightMap)) { + requestCounter.incr() + buildRecommendUserRequest(target) + .map { request => + fetcher + .fetch(request) + .map(_.v) + .map { responseOpt => + responseOpt + .map { response => + response.recommendedUsers + .sortBy(-_.score) + .map(convertToCandidateUsers) + .map(_.withCandidateSource(identifier)) + }.getOrElse(Nil) + } + }.getOrElse(Stitch.Nil) + } else { + Stitch.Nil + } + } + + private[this] def buildRecommendUserRequest( + target: UserUserGraphCandidateSource.Target + ): Option[RecommendUserRequest] = { + (target.getOptionalUserId, target.recentFollowedUserIds) match { + case (Some(userId), Some(recentFollowedUserIds)) => + // use recentFollowedUserIds as seeds for initial experiment + val seedsWithWeights: Map[Long, Double] = recentFollowedUserIds.map { + recentFollowedUserId => + recentFollowedUserId -> UserUserGraphCandidateSource.DefaultSeedWeight + }.toMap + val request = RecommendUserRequest( + requesterId = userId, + displayLocation = UserUserGraphCandidateSource.DisplayLocation, + seedsWithWeights = seedsWithWeights, + excludedUserIds = Some(target.excludedUserIds), + maxNumResults = Some(target.params.getInt(UserUserGraphParams.MaxCandidatesToReturn)), + maxNumSocialProofs = Some(UserUserGraphCandidateSource.MaxNumSocialProofs), + minUserPerSocialProof = Some(UserUserGraphCandidateSource.MinUserPerSocialProof), + socialProofTypes = Some(Seq(UserUserGraphCandidateSource.SocialProofType)) + ) + Some(request) + case _ => None + } + } + + private[this] def convertToCandidateUsers( + recommendedUser: RecommendedUser + ): CandidateUser = { + val socialProofUserIds = + recommendedUser.socialProofs.getOrElse(UserUserGraphCandidateSource.SocialProofType, Nil) + val reasonOpt = if (socialProofUserIds.nonEmpty) { + Some( + Reason( + Some(AccountProof(followProof = + Some(FollowProof(socialProofUserIds, socialProofUserIds.size))))) + ) + } else { + None + } + CandidateUser( + id = recommendedUser.userId, + score = Some(recommendedUser.score), + reason = reasonOpt) + } +} + +object UserUserGraphCandidateSource { + type Target = HasParams + with HasClientContext + with HasRecentFollowedUserIds + with HasExcludedUserIds + + val Identifier: CandidateSourceIdentifier = CandidateSourceIdentifier( + Algorithm.UserUserGraph.toString) + //Use HomeTimeline for experiment + val DisplayLocation: RecommendUserDisplayLocation = RecommendUserDisplayLocation.HomeTimeLine + + //Default params used in MagicRecs + val DefaultSeedWeight: Double = 1.0 + val SocialProofType = UserSocialProofType.Follow + val MaxNumSocialProofs = 10 + val MinUserPerSocialProof: Map[UserSocialProofType, Int] = + Map[UserSocialProofType, Int]((SocialProofType, 2)) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/user_user_graph/UserUserGraphFSConfig.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/user_user_graph/UserUserGraphFSConfig.scala new file mode 100644 index 0000000000..3d3015a362 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/user_user_graph/UserUserGraphFSConfig.scala @@ -0,0 +1,15 @@ +package com.twitter.follow_recommendations.common.candidate_sources.user_user_graph + +import com.twitter.follow_recommendations.configapi.common.FeatureSwitchConfig +import com.twitter.timelines.configapi.FSName +import com.twitter.timelines.configapi.Param +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class UserUserGraphFSConfig @Inject() () extends FeatureSwitchConfig { + override val booleanFSParams: Seq[Param[Boolean] with FSName] = Seq( + UserUserGraphParams.UserUserGraphCandidateSourceEnabledInWeightMap, + UserUserGraphParams.UserUserGraphCandidateSourceEnabledInTransform + ) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/user_user_graph/UserUserGraphParams.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/user_user_graph/UserUserGraphParams.scala new file mode 100644 index 0000000000..d146c3459f --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/user_user_graph/UserUserGraphParams.scala @@ -0,0 +1,19 @@ +package com.twitter.follow_recommendations.common.candidate_sources.user_user_graph + +import com.twitter.timelines.configapi.FSParam +import com.twitter.timelines.configapi.Param + +object UserUserGraphParams { + + // max number of candidates to return in total, 50 is the default param used in MagicRecs + object MaxCandidatesToReturn extends Param[Int](default = 50) + + // whether or not to include UserUserGraph candidate source in the weighted blending step + case object UserUserGraphCandidateSourceEnabledInWeightMap + extends FSParam[Boolean]("user_user_graph_candidate_source_enabled_in_weight_map", true) + + // whether or not to include UserUserGraph candidate source in the final transform step + case object UserUserGraphCandidateSourceEnabledInTransform + extends FSParam[Boolean]("user_user_graph_candidate_source_enabled_in_transform", true) + +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/addressbook/AddressbookClient.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/addressbook/AddressbookClient.scala new file mode 100644 index 0000000000..6fe1cee6f3 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/addressbook/AddressbookClient.scala @@ -0,0 +1,221 @@ +package com.twitter.follow_recommendations.common.clients.addressbook + +import com.twitter.addressbook.datatypes.thriftscala.QueryType +import com.twitter.addressbook.thriftscala.AddressBookGetRequest +import com.twitter.addressbook.thriftscala.AddressBookGetResponse +import com.twitter.addressbook.thriftscala.Addressbook2 +import com.twitter.addressbook.thriftscala.ClientInfo +import com.twitter.finagle.stats.NullStatsReceiver +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.wtf.scalding.jobs.addressbook.thriftscala.STPResultFeature +import com.twitter.follow_recommendations.common.clients.addressbook.models.Contact +import com.twitter.follow_recommendations.common.clients.addressbook.models.EdgeType +import com.twitter.follow_recommendations.common.clients.addressbook.models.QueryOption +import com.twitter.follow_recommendations.common.clients.addressbook.models.RecordIdentifier +import com.twitter.wtf.scalding.jobs.address_book.ABUtil.hashContact +import com.twitter.wtf.scalding.jobs.address_book.ABUtil.normalizeEmail +import com.twitter.wtf.scalding.jobs.address_book.ABUtil.normalizePhoneNumber +import com.twitter.hermit.usercontacts.thriftscala.{UserContacts => tUserContacts} +import com.twitter.stitch.Stitch +import com.twitter.strato.client.Fetcher +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class AddressbookClient @Inject() ( + addressbookService: Addressbook2.MethodPerEndpoint, + statsReceiver: StatsReceiver = NullStatsReceiver) { + + private val stats: StatsReceiver = statsReceiver.scope(this.getClass.getSimpleName) + + private[this] def getResponseFromService( + identifiers: Seq[RecordIdentifier], + batchSize: Int, + edgeType: EdgeType, + maxFetches: Int, + queryOption: Option[QueryOption] + ): Stitch[Seq[AddressBookGetResponse]] = { + Stitch + .collect( + identifiers.map { identifier => + Stitch.callFuture( + addressbookService.get(AddressBookGetRequest( + clientInfo = ClientInfo(None), + identifier = identifier.toThrift, + edgeType = edgeType.toThrift, + queryType = QueryType.UserId, + queryOption = queryOption.map(_.toThrift), + maxFetches = maxFetches, + resultBatchSize = batchSize + ))) + } + ) + } + + private[this] def getContactsResponseFromService( + identifiers: Seq[RecordIdentifier], + batchSize: Int, + edgeType: EdgeType, + maxFetches: Int, + queryOption: Option[QueryOption] + ): Stitch[Seq[AddressBookGetResponse]] = { + Stitch + .collect( + identifiers.map { identifier => + Stitch.callFuture( + addressbookService.get(AddressBookGetRequest( + clientInfo = ClientInfo(None), + identifier = identifier.toThrift, + edgeType = edgeType.toThrift, + queryType = QueryType.Contact, + queryOption = queryOption.map(_.toThrift), + maxFetches = maxFetches, + resultBatchSize = batchSize + ))) + } + ) + } + + /** Mode of addressbook resolving logic + * ManhattanThenABV2: fetching manhattan cached result and backfill with addressbook v2 + * ABV2Only: calling addressbook v2 directly without fetching manhattan cached result + * This can be controlled by passing a fetcher or not. Passing a fetcher will attempt to use it, + * if not then it won't. + */ + def getUsers( + userId: Long, + identifiers: Seq[RecordIdentifier], + batchSize: Int, + edgeType: EdgeType, + fetcherOption: Option[Fetcher[Long, Unit, tUserContacts]] = None, + maxFetches: Int = 1, + queryOption: Option[QueryOption] = None, + ): Stitch[Seq[Long]] = { + fetcherOption match { + case Some(fetcher) => + getUsersFromManhattan(userId, fetcher).flatMap { userContacts => + if (userContacts.isEmpty) { + stats.counter("mhEmptyThenFromAbService").incr() + getResponseFromService(identifiers, batchSize, edgeType, maxFetches, queryOption) + .map(_.flatMap(_.users).flatten.distinct) + } else { + stats.counter("fromManhattan").incr() + Stitch.value(userContacts) + } + } + case None => + stats.counter("fromAbService").incr() + getResponseFromService(identifiers, batchSize, edgeType, maxFetches, queryOption) + .map(_.flatMap(_.users).flatten.distinct) + } + } + + def getHashedContacts( + normalizeFn: String => String, + extractField: String, + )( + userId: Long, + identifiers: Seq[RecordIdentifier], + batchSize: Int, + edgeType: EdgeType, + fetcherOption: Option[Fetcher[String, Unit, STPResultFeature]] = None, + maxFetches: Int = 1, + queryOption: Option[QueryOption] = None, + ): Stitch[Seq[String]] = { + + fetcherOption match { + case Some(fetcher) => + getContactsFromManhattan(userId, fetcher).flatMap { userContacts => + if (userContacts.isEmpty) { + getContactsResponseFromService( + identifiers, + batchSize, + edgeType, + maxFetches, + queryOption) + .map { response => + for { + resp <- response + contacts <- resp.contacts + contactsThrift = contacts.map(Contact.fromThrift) + contactsSet = extractField match { + case "emails" => contactsThrift.flatMap(_.emails.toSeq.flatten) + case "phoneNumbers" => contactsThrift.flatMap(_.phoneNumbers.toSeq.flatten) + } + hashedAndNormalizedContacts = contactsSet.map(c => hashContact(normalizeFn(c))) + } yield hashedAndNormalizedContacts + }.map(_.flatten) + } else { + Stitch.Nil + } + } + case None => { + getContactsResponseFromService(identifiers, batchSize, edgeType, maxFetches, queryOption) + .map { response => + for { + resp <- response + contacts <- resp.contacts + contactsThrift = contacts.map(Contact.fromThrift) + contactsSet = extractField match { + case "emails" => contactsThrift.flatMap(_.emails.toSeq.flatten) + case "phoneNumbers" => contactsThrift.flatMap(_.phoneNumbers.toSeq.flatten) + } + hashedAndNormalizedContacts = contactsSet.map(c => hashContact(normalizeFn(c))) + } yield hashedAndNormalizedContacts + }.map(_.flatten) + } + } + } + + def getEmailContacts = getHashedContacts(normalizeEmail, "emails") _ + def getPhoneContacts = getHashedContacts(normalizePhoneNumber, "phoneNumbers") _ + + private def getUsersFromManhattan( + userId: Long, + fetcher: Fetcher[Long, Unit, tUserContacts], + ): Stitch[Seq[Long]] = fetcher + .fetch(userId) + .map(_.v.map(_.destinationIds).toSeq.flatten.distinct) + + private def getContactsFromManhattan( + userId: Long, + fetcher: Fetcher[String, Unit, STPResultFeature], + ): Stitch[Seq[String]] = fetcher + .fetch(userId.toString) + .map(_.v.map(_.strongTieUserFeature.map(_.destId)).toSeq.flatten.distinct) +} + +object AddressbookClient { + val AddressBook2BatchSize = 500 + + def createQueryOption(edgeType: EdgeType, isPhone: Boolean): Option[QueryOption] = + (edgeType, isPhone) match { + case (EdgeType.Reverse, _) => + None + case (EdgeType.Forward, true) => + Some( + QueryOption( + onlyDiscoverableInExpansion = false, + onlyConfirmedInExpansion = false, + onlyDiscoverableInResult = false, + onlyConfirmedInResult = false, + fetchGlobalApiNamespace = false, + isDebugRequest = false, + resolveEmails = false, + resolvePhoneNumbers = true + )) + case (EdgeType.Forward, false) => + Some( + QueryOption( + onlyDiscoverableInExpansion = false, + onlyConfirmedInExpansion = false, + onlyDiscoverableInResult = false, + onlyConfirmedInResult = false, + fetchGlobalApiNamespace = false, + isDebugRequest = false, + resolveEmails = true, + resolvePhoneNumbers = false + )) + } + +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/addressbook/AddressbookModule.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/addressbook/AddressbookModule.scala new file mode 100644 index 0000000000..97b841dd2a --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/addressbook/AddressbookModule.scala @@ -0,0 +1,10 @@ +package com.twitter.follow_recommendations.common.clients.addressbook + +import com.twitter.addressbook.thriftscala.Addressbook2 +import com.twitter.finatra.mtls.thriftmux.modules.MtlsClient +import com.twitter.follow_recommendations.common.clients.common.BaseClientModule + +object AddressbookModule extends BaseClientModule[Addressbook2.MethodPerEndpoint] with MtlsClient { + override val label = "addressbook" + override val dest = "/s/addressbook/addressbook2" +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/addressbook/BUILD b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/addressbook/BUILD new file mode 100644 index 0000000000..b94cf3efec --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/addressbook/BUILD @@ -0,0 +1,21 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "addressbook/thrift/src/thrift/com/twitter/addressbook:thrift-scala", + "addressbook/thrift/src/thrift/com/twitter/addressbook/datatypes:thrift-scala", + "finatra-internal/mtls-thriftmux/src/main/scala", + "finatra/inject/inject-thrift-client", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/addressbook/models", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/common", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/utils", + "src/scala/com/twitter/wtf/scalding/jobs/address_book:ab_util", + "src/thrift/com/twitter/hermit/usercontacts:hermit-usercontacts-scala", + "src/thrift/com/twitter/wtf/addressbook:addressbook-scala", + "stitch/stitch-core", + "strato/src/main/scala/com/twitter/strato/client", + ], +) diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/addressbook/models/BUILD b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/addressbook/models/BUILD new file mode 100644 index 0000000000..885055e2c8 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/addressbook/models/BUILD @@ -0,0 +1,10 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "addressbook/thrift/src/thrift/com/twitter/addressbook:thrift-scala", + "addressbook/thrift/src/thrift/com/twitter/addressbook/datatypes:thrift-scala", + ], +) diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/addressbook/models/Contact.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/addressbook/models/Contact.scala new file mode 100644 index 0000000000..3ff39fa32d --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/addressbook/models/Contact.scala @@ -0,0 +1,29 @@ +package com.twitter.follow_recommendations.common.clients.addressbook.models + +import com.twitter.addressbook.{thriftscala => t} +import com.twitter.util.Time + +case class Contact( + id: Long, + emails: Option[Set[String]], + phoneNumbers: Option[Set[String]], + firstName: Option[String], + lastName: Option[String], + name: Option[String], + appId: Option[Long], + appIds: Option[Set[Long]], + importedTimestamp: Option[Time]) + +object Contact { + def fromThrift(thriftContact: t.Contact): Contact = Contact( + thriftContact.id, + thriftContact.emails.map(_.toSet), + thriftContact.phoneNumbers.map(_.toSet), + thriftContact.firstName, + thriftContact.lastName, + thriftContact.name, + thriftContact.appId, + thriftContact.appIds.map(_.toSet), + thriftContact.importedTimestamp.map(Time.fromMilliseconds) + ) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/addressbook/models/EdgeType.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/addressbook/models/EdgeType.scala new file mode 100644 index 0000000000..6cfcd65cdb --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/addressbook/models/EdgeType.scala @@ -0,0 +1,16 @@ +package com.twitter.follow_recommendations.common.clients.addressbook.models + +import com.twitter.addressbook.datatypes.{thriftscala => t} + +sealed trait EdgeType { + def toThrift: t.EdgeType +} + +object EdgeType { + case object Forward extends EdgeType { + override val toThrift: t.EdgeType = t.EdgeType.Forward + } + case object Reverse extends EdgeType { + override val toThrift: t.EdgeType = t.EdgeType.Reverse + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/addressbook/models/QueryOption.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/addressbook/models/QueryOption.scala new file mode 100644 index 0000000000..a17c163cbe --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/addressbook/models/QueryOption.scala @@ -0,0 +1,24 @@ +package com.twitter.follow_recommendations.common.clients.addressbook.models + +import com.twitter.addressbook.{thriftscala => t} + +case class QueryOption( + onlyDiscoverableInExpansion: Boolean, + onlyConfirmedInExpansion: Boolean, + onlyDiscoverableInResult: Boolean, + onlyConfirmedInResult: Boolean, + fetchGlobalApiNamespace: Boolean, + isDebugRequest: Boolean, + resolveEmails: Boolean, + resolvePhoneNumbers: Boolean) { + def toThrift: t.QueryOption = t.QueryOption( + onlyDiscoverableInExpansion, + onlyConfirmedInExpansion, + onlyDiscoverableInResult, + onlyConfirmedInResult, + fetchGlobalApiNamespace, + isDebugRequest, + resolveEmails, + resolvePhoneNumbers + ) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/addressbook/models/RecordIdentifier.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/addressbook/models/RecordIdentifier.scala new file mode 100644 index 0000000000..0154c71dce --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/addressbook/models/RecordIdentifier.scala @@ -0,0 +1,10 @@ +package com.twitter.follow_recommendations.common.clients.addressbook.models + +import com.twitter.addressbook.datatypes.{thriftscala => t} + +case class RecordIdentifier( + userId: Option[Long], + email: Option[String], + phoneNumber: Option[String]) { + def toThrift: t.RecordIdentifier = t.RecordIdentifier(userId, email, phoneNumber) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/adserver/AdRequest.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/adserver/AdRequest.scala new file mode 100644 index 0000000000..07cfcdaf5c --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/adserver/AdRequest.scala @@ -0,0 +1,45 @@ +package com.twitter.follow_recommendations.common.clients.adserver + +import com.twitter.adserver.{thriftscala => t} +import com.twitter.follow_recommendations.common.models.DisplayLocation +import com.twitter.product_mixer.core.model.marshalling.request.ClientContext + +case class AdRequest( + clientContext: ClientContext, + displayLocation: DisplayLocation, + isTest: Option[Boolean], + profileUserId: Option[Long]) { + def toThrift: t.AdRequestParams = { + + val request = t.AdRequest( + displayLocation = displayLocation.toAdDisplayLocation.getOrElse( + throw new MissingAdDisplayLocation(displayLocation)), + isTest = isTest, + countImpressionsOnCallback = Some(true), + numOrganicItems = Some(AdRequest.DefaultNumOrganicItems.toShort), + profileUserId = profileUserId + ) + + val clientInfo = t.ClientInfo( + clientId = clientContext.appId.map(_.toInt), + userIp = clientContext.ipAddress, + userId64 = clientContext.userId, + guestId = clientContext.guestId, + userAgent = clientContext.userAgent, + referrer = None, + deviceId = clientContext.deviceId, + languageCode = clientContext.languageCode, + countryCode = clientContext.countryCode + ) + + t.AdRequestParams(request, clientInfo) + } +} + +object AdRequest { + val DefaultNumOrganicItems = 10 +} + +class MissingAdDisplayLocation(displayLocation: DisplayLocation) + extends Exception( + s"Display Location ${displayLocation.toString} has no mapped AdsDisplayLocation set.") diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/adserver/AdserverClient.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/adserver/AdserverClient.scala new file mode 100644 index 0000000000..927c0784ca --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/adserver/AdserverClient.scala @@ -0,0 +1,16 @@ +package com.twitter.follow_recommendations.common.clients.adserver + +import com.twitter.adserver.thriftscala.NewAdServer +import com.twitter.adserver.{thriftscala => t} +import com.twitter.stitch.Stitch +import javax.inject.{Inject, Singleton} + +@Singleton +class AdserverClient @Inject() (adserverService: NewAdServer.MethodPerEndpoint) { + def getAdImpressions(adRequest: AdRequest): Stitch[Seq[t.AdImpression]] = { + Stitch + .callFuture( + adserverService.makeAdRequest(adRequest.toThrift) + ).map(_.impressions) + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/adserver/AdserverModule.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/adserver/AdserverModule.scala new file mode 100644 index 0000000000..9a425c0584 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/adserver/AdserverModule.scala @@ -0,0 +1,15 @@ +package com.twitter.follow_recommendations.common.clients.adserver + +import com.twitter.adserver.thriftscala.NewAdServer +import com.twitter.conversions.DurationOps._ +import com.twitter.finagle.ThriftMux +import com.twitter.finatra.mtls.thriftmux.modules.MtlsClient +import com.twitter.follow_recommendations.common.clients.common.BaseClientModule + +object AdserverModule extends BaseClientModule[NewAdServer.MethodPerEndpoint] with MtlsClient { + override val label = "adserver" + override val dest = "/s/ads/adserver" + + override def configureThriftMuxClient(client: ThriftMux.Client): ThriftMux.Client = + client.withRequestTimeout(500.millis) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/adserver/BUILD b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/adserver/BUILD new file mode 100644 index 0000000000..c90656afeb --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/adserver/BUILD @@ -0,0 +1,14 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "finatra-internal/mtls-thriftmux/src/main/scala", + "finatra/inject/inject-thrift-client", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/common", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models", + "src/thrift/com/twitter/ads/adserver:adserver_rpc-scala", + "stitch/stitch-core", + ], +) diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/cache/BUILD b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/cache/BUILD new file mode 100644 index 0000000000..a32739651f --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/cache/BUILD @@ -0,0 +1,15 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "cache/client", + "finagle/finagle-memcached/src/main/scala", + "finatra/inject/inject-core/src/main/scala", + "finatra/inject/inject-thrift-client/src/main/scala", + "stitch/stitch-core", + "util/util-core:scala", + "util/util-thrift", + ], +) diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/cache/MemcacheClient.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/cache/MemcacheClient.scala new file mode 100644 index 0000000000..d00aef4755 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/cache/MemcacheClient.scala @@ -0,0 +1,121 @@ +package com.twitter.follow_recommendations.common.clients.cache + +import com.twitter.bijection.Bijection +import com.twitter.conversions.DurationOps._ +import com.twitter.finagle.Memcached.Client +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.finagle.util.DefaultTimer +import com.twitter.io.Buf +import com.twitter.stitch.Stitch +import com.twitter.util.Duration +import com.twitter.util.Future +import com.twitter.util.Time +import java.security.MessageDigest + +object MemcacheClient { + def apply[V]( + client: Client, + dest: String, + valueBijection: Bijection[Buf, V], + ttl: Duration, + statsReceiver: StatsReceiver + ): MemcacheClient[V] = { + new MemcacheClient(client, dest, valueBijection, ttl, statsReceiver) + } +} + +class MemcacheClient[V]( + client: Client, + dest: String, + valueBijection: Bijection[Buf, V], + ttl: Duration, + statsReceiver: StatsReceiver) { + val cache = client.newRichClient(dest).adapt[V](valueBijection) + val cacheTtl = Time.fromSeconds(ttl.inSeconds) + + /** + * If cache contains key, return value from cache. Otherwise, run the underlying call + * to fetch the value, store it in cache, and then return the value. + */ + def readThrough( + key: String, + underlyingCall: () => Stitch[V] + ): Stitch[V] = { + val cachedResult: Stitch[Option[V]] = Stitch + .callFuture(getIfPresent(key)) + .within(70.millisecond)(DefaultTimer) + .rescue { + case e: Exception => + statsReceiver.scope("rescued").counter(e.getClass.getSimpleName).incr() + Stitch(None) + } + val resultStitch = cachedResult.map { resultOption => + resultOption match { + case Some(cacheValue) => Stitch.value(cacheValue) + case None => + val underlyingCallStitch = profileStitch( + underlyingCall(), + statsReceiver.scope("underlyingCall") + ) + underlyingCallStitch.map { result => + put(key, result) + result + } + } + }.flatten + // profile the overall Stitch, and return the result + profileStitch(resultStitch, statsReceiver.scope("readThrough")) + } + + def getIfPresent(key: String): Future[Option[V]] = { + cache + .get(hashString(key)) + .onSuccess { + case Some(value) => statsReceiver.counter("cache_hits").incr() + case None => statsReceiver.counter("cache_misses").incr() + } + .onFailure { + case e: Exception => + statsReceiver.counter("cache_misses").incr() + statsReceiver.scope("rescued").counter(e.getClass.getSimpleName).incr() + } + .rescue { + case _ => Future.None + } + } + + def put(key: String, value: V): Future[Unit] = { + cache.set(hashString(key), 0, cacheTtl, value) + } + + /** + * Hash the input key string to a fixed length format using SHA-256 hash function. + */ + def hashString(input: String): String = { + val bytes = MessageDigest.getInstance("SHA-256").digest(input.getBytes("UTF-8")) + bytes.map("%02x".format(_)).mkString + } + + /** + * Helper function for timing a stitch, returning the original stitch. + * + * Defining the profiling function here to keep the dependencies of this class + * generic and easy to export (i.e. copy-and-paste) into other services or packages. + */ + def profileStitch[T](stitch: Stitch[T], stat: StatsReceiver): Stitch[T] = { + Stitch + .time(stitch) + .map { + case (response, stitchRunDuration) => + stat.counter("requests").incr() + stat.stat("latency_ms").add(stitchRunDuration.inMilliseconds) + response + .onSuccess { _ => stat.counter("success").incr() } + .onFailure { e => + stat.counter("failures").incr() + stat.scope("failures").counter(e.getClass.getSimpleName).incr() + } + } + .lowerFromTry + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/cache/MemcacheModule.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/cache/MemcacheModule.scala new file mode 100644 index 0000000000..e85d12f20e --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/cache/MemcacheModule.scala @@ -0,0 +1,30 @@ +package com.twitter.follow_recommendations.common.clients.cache + +import com.google.inject.Provides +import com.twitter.conversions.DurationOps._ +import com.twitter.finagle.Memcached +import com.twitter.finagle.Memcached.Client +import com.twitter.finagle.mtls.client.MtlsStackClient._ +import com.twitter.finagle.mtls.authentication.ServiceIdentifier +import com.twitter.finagle.service.Retries +import com.twitter.finagle.service.RetryPolicy +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.inject.TwitterModule +import javax.inject.Singleton + +object MemcacheModule extends TwitterModule { + @Provides + @Singleton + def provideMemcacheClient( + serviceIdentifier: ServiceIdentifier, + statsReceiver: StatsReceiver, + ): Client = { + Memcached.client + .withMutualTls(serviceIdentifier) + .withStatsReceiver(statsReceiver.scope("twemcache")) + .withTransport.connectTimeout(1.seconds) + .withRequestTimeout(1.seconds) + .withSession.acquisitionTimeout(10.seconds) + .configured(Retries.Policy(RetryPolicy.tries(1))) + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/cache/ThriftBijection.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/cache/ThriftBijection.scala new file mode 100644 index 0000000000..62e36bc336 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/cache/ThriftBijection.scala @@ -0,0 +1,81 @@ +package com.twitter.follow_recommendations.common.clients.cache + +import com.twitter.bijection.Bijection +import com.twitter.io.Buf +import com.twitter.scrooge.CompactThriftSerializer +import com.twitter.scrooge.ThriftEnum +import com.twitter.scrooge.ThriftStruct +import java.nio.ByteBuffer + +abstract class ThriftBijection[T <: ThriftStruct] extends Bijection[Buf, T] { + val serializer: CompactThriftSerializer[T] + + override def apply(b: Buf): T = { + val byteArray = Buf.ByteArray.Owned.extract(b) + serializer.fromBytes(byteArray) + } + + override def invert(a: T): Buf = { + val byteArray = serializer.toBytes(a) + Buf.ByteArray.Owned(byteArray) + } +} + +abstract class ThriftOptionBijection[T <: ThriftStruct] extends Bijection[Buf, Option[T]] { + val serializer: CompactThriftSerializer[T] + + override def apply(b: Buf): Option[T] = { + if (b.isEmpty) { + None + } else { + val byteArray = Buf.ByteArray.Owned.extract(b) + Some(serializer.fromBytes(byteArray)) + } + } + + override def invert(a: Option[T]): Buf = { + a match { + case Some(t) => + val byteArray = serializer.toBytes(t) + Buf.ByteArray.Owned(byteArray) + case None => Buf.Empty + } + } +} + +class ThriftEnumBijection[T <: ThriftEnum](constructor: Int => T) extends Bijection[Buf, T] { + override def apply(b: Buf): T = { + val byteArray = Buf.ByteArray.Owned.extract(b) + val byteBuffer = ByteBuffer.wrap(byteArray) + constructor(byteBuffer.getInt()) + } + + override def invert(a: T): Buf = { + val byteBuffer: ByteBuffer = ByteBuffer.allocate(4) + byteBuffer.putInt(a.getValue) + Buf.ByteArray.Owned(byteBuffer.array()) + } +} + +class ThriftEnumOptionBijection[T <: ThriftEnum](constructor: Int => T) extends Bijection[Buf, Option[T]] { + override def apply(b: Buf): Option[T] = { + if (b.isEmpty) { + None + } else { + val byteArray = Buf.ByteArray.Owned.extract(b) + val byteBuffer = ByteBuffer.wrap(byteArray) + Some(constructor(byteBuffer.getInt())) + } + } + + override def invert(a: Option[T]): Buf = { + a match { + case Some(obj) => { + val byteBuffer: ByteBuffer = ByteBuffer.allocate(4) + byteBuffer.putInt(obj.getValue) + Buf.ByteArray.Owned(byteBuffer.array()) + } + case None => Buf.Empty + } + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/common/BUILD b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/common/BUILD new file mode 100644 index 0000000000..330981d803 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/common/BUILD @@ -0,0 +1,11 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "finatra-internal/mtls-thriftmux/src/main/scala", + "finatra/inject/inject-thrift-client", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/constants", + ], +) diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/common/BaseClientModule.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/common/BaseClientModule.scala new file mode 100644 index 0000000000..43949ea193 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/common/BaseClientModule.scala @@ -0,0 +1,20 @@ +package com.twitter.follow_recommendations.common.clients.common + +import com.twitter.finagle.ThriftMux +import com.twitter.finagle.thrift.Protocols +import com.twitter.follow_recommendations.common.constants.ServiceConstants._ +import com.twitter.inject.thrift.modules.ThriftClientModule +import scala.reflect.ClassTag + +/** + * basic client configurations that we apply for all of our clients go in here + */ +abstract class BaseClientModule[T: ClassTag] extends ThriftClientModule[T] { + def configureThriftMuxClient(client: ThriftMux.Client): ThriftMux.Client = { + client + .withProtocolFactory( + Protocols.binaryFactory( + stringLengthLimit = StringLengthLimit, + containerLengthLimit = ContainerLengthLimit)) + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/deepbirdv2/BUILD b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/deepbirdv2/BUILD new file mode 100644 index 0000000000..daf4c63fa4 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/deepbirdv2/BUILD @@ -0,0 +1,20 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/com/google/inject:guice", + "3rdparty/jvm/com/google/inject/extensions:guice-assistedinject", + "3rdparty/jvm/net/codingwell:scala-guice", + "3rdparty/jvm/org/slf4j:slf4j-api", + "cortex-deepbird/prediction/src/main/scala/com/twitter/cortex/deepbird/prediction", + "cortex-deepbird/thrift/src/main/thrift:thrift-java", + "finatra-internal/mtls-thriftmux/src/main/scala", + "finatra/inject/inject-core/src/main/scala", + "finatra/inject/inject-thrift-client/src/main/scala", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/common", + "src/scala/com/twitter/ml/api/util", + "util/util-core:scala", + ], +) diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/deepbirdv2/DeepBirdV2PredictionServiceClientModule.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/deepbirdv2/DeepBirdV2PredictionServiceClientModule.scala new file mode 100644 index 0000000000..8f707910b7 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/deepbirdv2/DeepBirdV2PredictionServiceClientModule.scala @@ -0,0 +1,67 @@ +package com.twitter.follow_recommendations.common.clients.deepbirdv2 + +import com.google.inject.Provides +import com.google.inject.name.Named +import com.twitter.bijection.scrooge.TBinaryProtocol +import com.twitter.conversions.DurationOps._ +import com.twitter.cortex.deepbird.thriftjava.DeepbirdPredictionService +import com.twitter.finagle.ThriftMux +import com.twitter.finagle.builder.ClientBuilder +import com.twitter.finagle.mtls.authentication.ServiceIdentifier +import com.twitter.finagle.mtls.client.MtlsStackClient._ +import com.twitter.finagle.stats.NullStatsReceiver +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.finagle.thrift.ClientId +import com.twitter.finagle.thrift.RichClientParam +import com.twitter.follow_recommendations.common.constants.GuiceNamedConstants +import com.twitter.inject.TwitterModule + +/** + * Module that provides multiple deepbirdv2 prediction service clients + * We use the java api since data records are native java objects and we want to reduce overhead + * while serializing/deserializing data. + */ +object DeepBirdV2PredictionServiceClientModule extends TwitterModule { + + val RequestTimeout = 300.millis + + private def getDeepbirdPredictionServiceClient( + clientId: ClientId, + label: String, + dest: String, + statsReceiver: StatsReceiver, + serviceIdentifier: ServiceIdentifier + ): DeepbirdPredictionService.ServiceToClient = { + val clientStatsReceiver = statsReceiver.scope("clnt") + val mTlsClient = ThriftMux.client.withClientId(clientId).withMutualTls(serviceIdentifier) + new DeepbirdPredictionService.ServiceToClient( + ClientBuilder() + .name(label) + .stack(mTlsClient) + .dest(dest) + .requestTimeout(RequestTimeout) + .reportHostStats(NullStatsReceiver) + .build(), + RichClientParam( + new TBinaryProtocol.Factory(), + clientStats = clientStatsReceiver + ) + ) + } + + @Provides + @Named(GuiceNamedConstants.WTF_PROD_DEEPBIRDV2_CLIENT) + def providesWtfProdDeepbirdV2PredictionService( + clientId: ClientId, + statsReceiver: StatsReceiver, + serviceIdentifier: ServiceIdentifier + ): DeepbirdPredictionService.ServiceToClient = { + getDeepbirdPredictionServiceClient( + clientId = clientId, + label = "WtfProdDeepbirdV2PredictionService", + dest = "/s/cassowary/deepbirdv2-hermit-wtf", + statsReceiver = statsReceiver, + serviceIdentifier = serviceIdentifier + ) + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/dismiss_store/BUILD b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/dismiss_store/BUILD new file mode 100644 index 0000000000..33d8ba8416 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/dismiss_store/BUILD @@ -0,0 +1,19 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/com/github/nscala_time", + "3rdparty/jvm/com/google/inject:guice", + "3rdparty/jvm/com/google/inject/extensions:guice-assistedinject", + "3rdparty/jvm/net/codingwell:scala-guice", + "3rdparty/jvm/org/slf4j:slf4j-api", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/strato", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/constants", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models", + "follow-recommendations-service/thrift/src/main/thrift:thrift-scala", + "src/thrift/com/twitter/onboarding/relevance/store:store-scala", + "util/util-core:scala", + ], +) diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/dismiss_store/DismissStore.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/dismiss_store/DismissStore.scala new file mode 100644 index 0000000000..1c8c9f8fd8 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/dismiss_store/DismissStore.scala @@ -0,0 +1,60 @@ +package com.twitter.follow_recommendations.common.clients.dismiss_store + +import com.twitter.follow_recommendations.common.constants.GuiceNamedConstants +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.onboarding.relevance.store.thriftscala.WhoToFollowDismissEventDetails +import com.twitter.stitch.Stitch +import com.twitter.strato.catalog.Scan.Slice +import com.twitter.strato.client.Scanner +import com.twitter.util.logging.Logging +import javax.inject.Inject +import javax.inject.Named +import javax.inject.Singleton + +/** + * this store gets the list of dismissed candidates since a certain time + * primarily used for filtering out accounts that a user has explicitly dismissed + * + * we fail open on timeouts, but loudly on other errors + */ +@Singleton +class DismissStore @Inject() ( + @Named(GuiceNamedConstants.DISMISS_STORE_SCANNER) + scanner: Scanner[(Long, Slice[ + (Long, Long) + ]), Unit, (Long, (Long, Long)), WhoToFollowDismissEventDetails], + stats: StatsReceiver) + extends Logging { + + private val MaxCandidatesToReturn = 100 + + // gets a list of dismissed candidates. if numCandidatesToFetchOption is none, we will fetch the default number of candidates + def get( + userId: Long, + negStartTimeMs: Long, + maxCandidatesToFetchOption: Option[Int] + ): Stitch[Seq[Long]] = { + + val maxCandidatesToFetch = maxCandidatesToFetchOption.getOrElse(MaxCandidatesToReturn) + + scanner + .scan( + ( + userId, + Slice( + from = None, + to = Some((negStartTimeMs, Long.MaxValue)), + limit = Some(maxCandidatesToFetch) + ) + ) + ) + .map { + case s: Seq[((Long, (Long, Long)), WhoToFollowDismissEventDetails)] if s.nonEmpty => + s.map { + case ((_: Long, (_: Long, candidateId: Long)), _: WhoToFollowDismissEventDetails) => + candidateId + } + case _ => Nil + } + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/email_storage_service/BUILD b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/email_storage_service/BUILD new file mode 100644 index 0000000000..78a06d536f --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/email_storage_service/BUILD @@ -0,0 +1,14 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "emailstorage/server/src/main/thrift/com/twitter/emailstorage/api:email-storage-service-scala", + "finatra-internal/mtls-thriftmux/src/main/scala", + "finatra/inject/inject-thrift-client", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/common", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models", + "stitch/stitch-core", + ], +) diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/email_storage_service/EmailStorageServiceClient.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/email_storage_service/EmailStorageServiceClient.scala new file mode 100644 index 0000000000..4cf66c6581 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/email_storage_service/EmailStorageServiceClient.scala @@ -0,0 +1,28 @@ +package com.twitter.follow_recommendations.common.clients.email_storage_service + +import com.twitter.cds.contact_consent_state.thriftscala.PurposeOfProcessing +import com.twitter.emailstorage.api.thriftscala.EmailStorageService +import com.twitter.emailstorage.api.thriftscala.GetUsersEmailsRequest +import com.twitter.stitch.Stitch +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class EmailStorageServiceClient @Inject() ( + val emailStorageService: EmailStorageService.MethodPerEndpoint) { + + def getVerifiedEmail( + userId: Long, + purposeOfProcessing: PurposeOfProcessing + ): Stitch[Option[String]] = { + val req = GetUsersEmailsRequest( + userIds = Seq(userId), + clientIdentifier = Some("follow-recommendations-service"), + purposesOfProcessing = Some(Seq(purposeOfProcessing)) + ) + + Stitch.callFuture(emailStorageService.getUsersEmails(req)) map { + _.usersEmails.map(_.confirmedEmail.map(_.email)).head + } + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/email_storage_service/EmailStorageServiceModule.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/email_storage_service/EmailStorageServiceModule.scala new file mode 100644 index 0000000000..0a1be9d1b1 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/email_storage_service/EmailStorageServiceModule.scala @@ -0,0 +1,12 @@ +package com.twitter.follow_recommendations.common.clients.email_storage_service + +import com.twitter.emailstorage.api.thriftscala.EmailStorageService +import com.twitter.finatra.mtls.thriftmux.modules.MtlsClient +import com.twitter.follow_recommendations.common.clients.common.BaseClientModule + +object EmailStorageServiceModule + extends BaseClientModule[EmailStorageService.MethodPerEndpoint] + with MtlsClient { + override val label = "email-storage-service" + override val dest = "/s/email-server/email-server" +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/geoduck/BUILD b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/geoduck/BUILD new file mode 100644 index 0000000000..d67fa01d8c --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/geoduck/BUILD @@ -0,0 +1,22 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/com/github/nscala_time", + "3rdparty/jvm/com/google/inject:guice", + "3rdparty/jvm/com/google/inject/extensions:guice-assistedinject", + "3rdparty/jvm/net/codingwell:scala-guice", + "3rdparty/jvm/org/slf4j:slf4j-api", + "finatra-internal/mtls-thriftmux/src/main/scala", + "finatra/inject/inject-core/src/main/scala", + "finatra/inject/inject-thrift-client/src/main/scala", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/common", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models", + "src/thrift/com/twitter/geoduck:geoduck-scala", + "src/thrift/com/twitter/geoduck:geoduckpartnerplaces-thrift-scala", + "stitch/stitch-core", + "util/util-core:scala", + ], +) diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/geoduck/LocationServiceClient.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/geoduck/LocationServiceClient.scala new file mode 100644 index 0000000000..69207537c3 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/geoduck/LocationServiceClient.scala @@ -0,0 +1,62 @@ +package com.twitter.follow_recommendations.common.clients.geoduck + +import com.twitter.follow_recommendations.common.models.GeohashAndCountryCode +import com.twitter.geoduck.common.thriftscala.LocationSource +import com.twitter.geoduck.common.thriftscala.PlaceQuery +import com.twitter.geoduck.common.thriftscala.TransactionLocation +import com.twitter.geoduck.common.thriftscala.UserLocationRequest +import com.twitter.geoduck.thriftscala.LocationService +import com.twitter.stitch.Stitch +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class LocationServiceClient @Inject() (locationService: LocationService.MethodPerEndpoint) { + def getGeohashAndCountryCode(userId: Long): Stitch[GeohashAndCountryCode] = { + Stitch + .callFuture { + locationService + .userLocation( + UserLocationRequest( + Seq(userId), + Some(PlaceQuery(allPlaceTypes = Some(true))), + simpleReverseGeocode = true)) + .map(_.found.get(userId)).map { transactionLocationOpt => + val geohashOpt = transactionLocationOpt.flatMap(getGeohashFromTransactionLocation) + val countryCodeOpt = + transactionLocationOpt.flatMap(_.simpleRgcResult.flatMap(_.countryCodeAlpha2)) + GeohashAndCountryCode(geohashOpt, countryCodeOpt) + } + } + } + + private[this] def getGeohashFromTransactionLocation( + transactionLocation: TransactionLocation + ): Option[String] = { + transactionLocation.geohash.flatMap { geohash => + val geohashPrefixLength = transactionLocation.locationSource match { + // if location source is logical, keep the first 4 chars in geohash + case Some(LocationSource.Logical) => Some(4) + // if location source is physical, keep the prefix according to accuracy + // accuracy is the accuracy of GPS readings in the unit of meter + case Some(LocationSource.Physical) => + transactionLocation.coordinate.flatMap { coordinate => + coordinate.accuracy match { + case Some(accuracy) if (accuracy < 50) => Some(7) + case Some(accuracy) if (accuracy < 200) => Some(6) + case Some(accuracy) if (accuracy < 1000) => Some(5) + case Some(accuracy) if (accuracy < 50000) => Some(4) + case Some(accuracy) if (accuracy < 100000) => Some(3) + case _ => None + } + } + case Some(LocationSource.Model) => Some(4) + case _ => None + } + geohashPrefixLength match { + case Some(l: Int) => geohash.stringGeohash.map(_.take(l)) + case _ => None + } + } + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/geoduck/LocationServiceModule.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/geoduck/LocationServiceModule.scala new file mode 100644 index 0000000000..a5fc79a801 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/geoduck/LocationServiceModule.scala @@ -0,0 +1,12 @@ +package com.twitter.follow_recommendations.common.clients.geoduck + +import com.twitter.finatra.mtls.thriftmux.modules.MtlsClient +import com.twitter.follow_recommendations.common.clients.common.BaseClientModule +import com.twitter.geoduck.thriftscala.LocationService + +object LocationServiceModule + extends BaseClientModule[LocationService.MethodPerEndpoint] + with MtlsClient { + override val label = "geoduck_locationservice" + override val dest = "/s/geo/geoduck_locationservice" +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/geoduck/ReverseGeocodeClient.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/geoduck/ReverseGeocodeClient.scala new file mode 100644 index 0000000000..5763591283 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/geoduck/ReverseGeocodeClient.scala @@ -0,0 +1,57 @@ +package com.twitter.follow_recommendations.common.clients.geoduck + +import com.twitter.follow_recommendations.common.models.GeohashAndCountryCode +import com.twitter.geoduck.common.thriftscala.Location +import com.twitter.geoduck.common.thriftscala.PlaceQuery +import com.twitter.geoduck.common.thriftscala.ReverseGeocodeIPRequest +import com.twitter.geoduck.service.thriftscala.GeoContext +import com.twitter.geoduck.thriftscala.ReverseGeocoder +import com.twitter.stitch.Stitch +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class ReverseGeocodeClient @Inject() (rgcService: ReverseGeocoder.MethodPerEndpoint) { + def getGeohashAndCountryCode(ipAddress: String): Stitch[GeohashAndCountryCode] = { + Stitch + .callFuture { + rgcService + .reverseGeocodeIp( + ReverseGeocodeIPRequest( + Seq(ipAddress), + PlaceQuery(None), + simpleReverseGeocode = true + ) // note: simpleReverseGeocode means that country code will be included in response + ).map { response => + response.found.get(ipAddress) match { + case Some(location) => getGeohashAndCountryCodeFromLocation(location) + case _ => GeohashAndCountryCode(None, None) + } + } + } + } + + private def getGeohashAndCountryCodeFromLocation(location: Location): GeohashAndCountryCode = { + val countryCode: Option[String] = location.simpleRgcResult.flatMap { _.countryCodeAlpha2 } + + val geohashString: Option[String] = location.geohash.flatMap { hash => + hash.stringGeohash.flatMap { hashString => + Some(ReverseGeocodeClient.truncate(hashString)) + } + } + + GeohashAndCountryCode(geohashString, countryCode) + } + +} + +object ReverseGeocodeClient { + + val DefaultGeoduckIPRequestContext: GeoContext = + GeoContext(allPlaceTypes = true, includeGeohash = true, includeCountryCode = true) + + // All these geohashes are guessed by IP (Logical Location Source). + // So take the four letters to make sure it is consistent with LocationServiceClient + val GeohashLengthAfterTruncation = 4 + def truncate(geohash: String): String = geohash.take(GeohashLengthAfterTruncation) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/geoduck/UserLocationFetcher.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/geoduck/UserLocationFetcher.scala new file mode 100644 index 0000000000..706ae51434 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/geoduck/UserLocationFetcher.scala @@ -0,0 +1,59 @@ +package com.twitter.follow_recommendations.common.clients.geoduck + +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.follow_recommendations.common.models.GeohashAndCountryCode +import com.twitter.stitch.Stitch + +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class UserLocationFetcher @Inject() ( + locationServiceClient: LocationServiceClient, + reverseGeocodeClient: ReverseGeocodeClient, + statsReceiver: StatsReceiver) { + + private val stats: StatsReceiver = statsReceiver.scope("user_location_fetcher") + private val totalRequestsCounter = stats.counter("requests") + private val emptyResponsesCounter = stats.counter("empty") + private val locationServiceExceptionCounter = stats.counter("location_service_exception") + private val reverseGeocodeExceptionCounter = stats.counter("reverse_geocode_exception") + + def getGeohashAndCountryCode( + userId: Option[Long], + ipAddress: Option[String] + ): Stitch[Option[GeohashAndCountryCode]] = { + totalRequestsCounter.incr() + val lscLocationStitch = Stitch + .collect { + userId.map(locationServiceClient.getGeohashAndCountryCode) + }.rescue { + case _: Exception => + locationServiceExceptionCounter.incr() + Stitch.None + } + + val ipLocationStitch = Stitch + .collect { + ipAddress.map(reverseGeocodeClient.getGeohashAndCountryCode) + }.rescue { + case _: Exception => + reverseGeocodeExceptionCounter.incr() + Stitch.None + } + + Stitch.join(lscLocationStitch, ipLocationStitch).map { + case (lscLocation, ipLocation) => { + val geohash = lscLocation.flatMap(_.geohash).orElse(ipLocation.flatMap(_.geohash)) + val countryCode = + lscLocation.flatMap(_.countryCode).orElse(ipLocation.flatMap(_.countryCode)) + (geohash, countryCode) match { + case (None, None) => + emptyResponsesCounter.incr() + None + case _ => Some(GeohashAndCountryCode(geohash, countryCode)) + } + } + } + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/gizmoduck/BUILD b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/gizmoduck/BUILD new file mode 100644 index 0000000000..77cb553c06 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/gizmoduck/BUILD @@ -0,0 +1,21 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/com/github/nscala_time", + "3rdparty/jvm/com/google/inject:guice", + "3rdparty/jvm/com/google/inject/extensions:guice-assistedinject", + "3rdparty/jvm/net/codingwell:scala-guice", + "3rdparty/jvm/org/slf4j:slf4j-api", + "finatra-internal/mtls-thriftmux/src/main/scala", + "finatra/inject/inject-core/src/main/scala", + "finatra/inject/inject-thrift-client/src/main/scala", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/common", + "src/thrift/com/twitter/gizmoduck:thrift-scala", + "stitch/stitch-gizmoduck", + "util/util-core:scala", + ], +) diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/gizmoduck/GizmoduckClient.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/gizmoduck/GizmoduckClient.scala new file mode 100644 index 0000000000..25b5f0d758 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/gizmoduck/GizmoduckClient.scala @@ -0,0 +1,81 @@ +package com.twitter.follow_recommendations.common.clients.gizmoduck + +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.follow_recommendations.common.base.StatsUtil +import com.twitter.gizmoduck.thriftscala.LookupContext +import com.twitter.gizmoduck.thriftscala.PerspectiveEdge +import com.twitter.gizmoduck.thriftscala.QueryFields +import com.twitter.stitch.Stitch +import com.twitter.stitch.gizmoduck.Gizmoduck +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class GizmoduckClient @Inject() (gizmoduckStitchClient: Gizmoduck, statsReceiver: StatsReceiver) { + val stats = statsReceiver.scope("gizmoduck_client") + val getByIdStats = stats.scope("get_by_id") + val getUserById = stats.scope("get_user_by_id") + + def isProtected(userId: Long): Stitch[Boolean] = { + // get latency metrics with StatsUtil.profileStitch when calling .getById + val response = StatsUtil.profileStitch( + gizmoduckStitchClient.getById(userId, Set(QueryFields.Safety)), + getByIdStats + ) + response.map { result => + result.user.flatMap(_.safety).map(_.isProtected).getOrElse(true) + } + } + + def getUserName(userId: Long, forUserId: Long): Stitch[Option[String]] = { + val queryFields = GizmoduckClient.GetUserByIdUserNameQueryFields + val lookupContext = LookupContext( + forUserId = Some(forUserId), + perspectiveEdges = Some(GizmoduckClient.DefaultPerspectiveEdges) + ) + // get latency metrics with StatsUtil.profileStitch when calling .getUserById + val response = StatsUtil.profileStitch( + gizmoduckStitchClient.getUserById(userId, queryFields, lookupContext), + getUserById + ) + response.map(_.profile.map(_.name)) + } +} + +object GizmoduckClient { + // Similar to GizmoduckUserRepository.DefaultPerspectiveEdges + val DefaultPerspectiveEdges: Set[PerspectiveEdge] = + Set( + PerspectiveEdge.Blocking, + PerspectiveEdge.BlockedBy, + PerspectiveEdge.DeviceFollowing, + PerspectiveEdge.FollowRequestSent, + PerspectiveEdge.Following, + PerspectiveEdge.FollowedBy, + PerspectiveEdge.LifelineFollowing, + PerspectiveEdge.LifelineFollowedBy, + PerspectiveEdge.Muting, + PerspectiveEdge.NoRetweetsFrom + ) + + // From GizmoduckUserRepository.DefaultQueryFields + val GetUserByIdQueryFields: Set[QueryFields] = Set( + QueryFields.Account, + QueryFields.Counts, + QueryFields.ExtendedProfile, + QueryFields.Perspective, + QueryFields.Profile, + QueryFields.ProfileDesign, + QueryFields.ProfileLocation, + QueryFields.Safety, + QueryFields.Roles, + QueryFields.Takedowns, + QueryFields.UrlEntities, + QueryFields.DirectMessageView, + QueryFields.MediaView + ) + + val GetUserByIdUserNameQueryFields: Set[QueryFields] = Set( + QueryFields.Profile + ) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/gizmoduck/GizmoduckModule.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/gizmoduck/GizmoduckModule.scala new file mode 100644 index 0000000000..9a5efea06b --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/gizmoduck/GizmoduckModule.scala @@ -0,0 +1,24 @@ +package com.twitter.follow_recommendations.common.clients.gizmoduck + +import com.google.inject.Provides +import com.twitter.finatra.mtls.thriftmux.modules.MtlsClient +import com.twitter.follow_recommendations.common.clients.common.BaseClientModule +import com.twitter.gizmoduck.thriftscala.QueryFields +import com.twitter.gizmoduck.thriftscala.UserService +import com.twitter.stitch.gizmoduck.Gizmoduck +import javax.inject.Singleton + +object GizmoduckModule extends BaseClientModule[UserService.MethodPerEndpoint] with MtlsClient { + override val label = "gizmoduck" + override val dest = "/s/gizmoduck/gizmoduck" + + @Provides + @Singleton + def provideExtraGizmoduckQueryFields: Set[QueryFields] = Set.empty + + @Provides + @Singleton + def providesStitchClient(futureIface: UserService.MethodPerEndpoint): Gizmoduck = { + Gizmoduck(futureIface) + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/graph_feature_service/BUILD b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/graph_feature_service/BUILD new file mode 100644 index 0000000000..ec1139cfe0 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/graph_feature_service/BUILD @@ -0,0 +1,14 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "finatra-internal/mtls-thriftmux/src/main/scala", + "finatra/inject/inject-thrift-client", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/common", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models", + "graph-feature-service/src/main/thrift/com/twitter/graph_feature_service:graph_feature_service_thrift-scala", + "stitch/stitch-core", + ], +) diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/graph_feature_service/GraphFeatureServiceClient.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/graph_feature_service/GraphFeatureServiceClient.scala new file mode 100644 index 0000000000..f333d895f3 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/graph_feature_service/GraphFeatureServiceClient.scala @@ -0,0 +1,50 @@ +package com.twitter.follow_recommendations.common.clients.graph_feature_service + +import com.twitter.follow_recommendations.common.models.FollowProof +import com.twitter.graph_feature_service.thriftscala.PresetFeatureTypes.WtfTwoHop +import com.twitter.graph_feature_service.thriftscala.EdgeType +import com.twitter.graph_feature_service.thriftscala.GfsIntersectionResponse +import com.twitter.graph_feature_service.thriftscala.GfsPresetIntersectionRequest +import com.twitter.graph_feature_service.thriftscala.{Server => GraphFeatureService} +import com.twitter.stitch.Stitch +import javax.inject.{Inject, Singleton} + +@Singleton +class GraphFeatureServiceClient @Inject() ( + graphFeatureService: GraphFeatureService.MethodPerEndpoint) { + + import GraphFeatureServiceClient._ + def getIntersections( + userId: Long, + candidateIds: Seq[Long], + numIntersectionIds: Int + ): Stitch[Map[Long, FollowProof]] = { + Stitch + .callFuture( + graphFeatureService.getPresetIntersection( + GfsPresetIntersectionRequest(userId, candidateIds, WtfTwoHop, Some(numIntersectionIds)) + ) + ).map { + case GfsIntersectionResponse(gfsIntersectionResults) => + (for { + candidateId <- candidateIds + gfsIntersectionResultForCandidate = + gfsIntersectionResults.filter(_.candidateUserId == candidateId) + followProof <- for { + result <- gfsIntersectionResultForCandidate + intersection <- result.intersectionValues + if leftEdgeTypes.contains(intersection.featureType.leftEdgeType) + if rightEdgeTypes.contains(intersection.featureType.rightEdgeType) + intersectionIds <- intersection.intersectionIds.toSeq + } yield FollowProof(intersectionIds, intersection.count.getOrElse(0)) + } yield { + candidateId -> followProof + }).toMap + } + } +} + +object GraphFeatureServiceClient { + val leftEdgeTypes: Set[EdgeType] = Set(EdgeType.Following) + val rightEdgeTypes: Set[EdgeType] = Set(EdgeType.FollowedBy) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/graph_feature_service/GraphFeatureStoreModule.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/graph_feature_service/GraphFeatureStoreModule.scala new file mode 100644 index 0000000000..8d15358cd6 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/graph_feature_service/GraphFeatureStoreModule.scala @@ -0,0 +1,12 @@ +package com.twitter.follow_recommendations.common.clients.graph_feature_service + +import com.twitter.finatra.mtls.thriftmux.modules.MtlsClient +import com.twitter.follow_recommendations.common.clients.common.BaseClientModule +import com.twitter.graph_feature_service.thriftscala.{Server => GraphFeatureService} + +object GraphFeatureStoreModule + extends BaseClientModule[GraphFeatureService.MethodPerEndpoint] + with MtlsClient { + override val label = "graph_feature_service" + override val dest = "/s/cassowary/graph_feature_service-server" +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/impression_store/BUILD b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/impression_store/BUILD new file mode 100644 index 0000000000..7432ce040d --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/impression_store/BUILD @@ -0,0 +1,18 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/com/github/nscala_time", + "3rdparty/jvm/com/google/inject:guice", + "3rdparty/jvm/com/google/inject/extensions:guice-assistedinject", + "3rdparty/jvm/net/codingwell:scala-guice", + "3rdparty/jvm/org/slf4j:slf4j-api", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/strato", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models", + "follow-recommendations-service/thrift/src/main/thrift:thrift-scala", + "stitch/stitch-socialgraph", + "util/util-core:scala", + ], +) diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/impression_store/ImpressionStoreModule.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/impression_store/ImpressionStoreModule.scala new file mode 100644 index 0000000000..35b5d78852 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/impression_store/ImpressionStoreModule.scala @@ -0,0 +1,31 @@ +package com.twitter.follow_recommendations.common.clients.impression_store + +import com.google.inject.Provides +import com.google.inject.Singleton +import com.twitter.follow_recommendations.thriftscala.DisplayLocation +import com.twitter.inject.TwitterModule +import com.twitter.strato.catalog.Scan.Slice +import com.twitter.strato.client.Client +import com.twitter.strato.thrift.ScroogeConvImplicits._ + +object ImpressionStoreModule extends TwitterModule { + + val columnPath: String = "onboarding/userrecs/wtfImpressionCountsStore" + + type PKey = (Long, DisplayLocation) + type LKey = Long + type Value = (Long, Int) + + @Provides + @Singleton + def providesImpressionStore(stratoClient: Client): WtfImpressionStore = { + new WtfImpressionStore( + stratoClient.scanner[ + (PKey, Slice[LKey]), + Unit, + (PKey, LKey), + Value + ](columnPath) + ) + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/impression_store/WtfImpressionStore.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/impression_store/WtfImpressionStore.scala new file mode 100644 index 0000000000..68692dc280 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/impression_store/WtfImpressionStore.scala @@ -0,0 +1,42 @@ +package com.twitter.follow_recommendations.common.clients.impression_store + +import com.twitter.follow_recommendations.common.models.DisplayLocation +import com.twitter.follow_recommendations.common.models.WtfImpression +import com.twitter.follow_recommendations.thriftscala.{DisplayLocation => TDisplayLocation} +import com.twitter.stitch.Stitch +import com.twitter.strato.catalog.Scan.Slice +import com.twitter.strato.client.Scanner +import com.twitter.util.Time +import com.twitter.util.logging.Logging +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class WtfImpressionStore @Inject() ( + scanner: Scanner[ + ((Long, TDisplayLocation), Slice[Long]), + Unit, + ((Long, TDisplayLocation), Long), + (Long, Int) + ]) extends Logging { + def get(userId: Long, dl: DisplayLocation): Stitch[Seq[WtfImpression]] = { + val thriftDl = dl.toThrift + scanner.scan(((userId, thriftDl), Slice.all[Long])).map { impressionsPerDl => + val wtfImpressions = + for { + (((_, _), candidateId), (latestTs, counts)) <- impressionsPerDl + } yield WtfImpression( + candidateId = candidateId, + displayLocation = dl, + latestTime = Time.fromMilliseconds(latestTs), + counts = counts + ) + wtfImpressions + } rescue { + // fail open so that the request can still go through + case ex: Throwable => + logger.warn(s"$dl WtfImpressionsStore warn: " + ex.getMessage) + Stitch.Nil + } + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/interests_service/BUILD b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/interests_service/BUILD new file mode 100644 index 0000000000..0835d840bf --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/interests_service/BUILD @@ -0,0 +1,14 @@ +scala_library( + name = "interests_service", + sources = ["InterestServiceClient.scala"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "frigate/frigate-common/src/main/scala/com/twitter/frigate/common/store/interests", + "interests-service/thrift/src/main/thrift:thrift-scala", + "strato/src/main/scala/com/twitter/strato/catalog", + "strato/src/main/scala/com/twitter/strato/client", + "strato/src/main/scala/com/twitter/strato/data", + "strato/src/main/scala/com/twitter/strato/thrift", + ], +) diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/interests_service/InterestServiceClient.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/interests_service/InterestServiceClient.scala new file mode 100644 index 0000000000..3904cf5155 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/interests_service/InterestServiceClient.scala @@ -0,0 +1,115 @@ +package com.twitter.follow_recommendations.common.clients.interests_service + +import com.google.inject.Inject +import com.google.inject.Singleton +import com.twitter.finagle.stats.NullStatsReceiver +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.frigate.common.store.InterestedInInterestsFetchKey +import com.twitter.inject.Logging +import com.twitter.interests.thriftscala.InterestId +import com.twitter.interests.thriftscala.InterestRelationship +import com.twitter.interests.thriftscala.InterestedInInterestModel +import com.twitter.interests.thriftscala.UserInterest +import com.twitter.interests.thriftscala.UserInterestData +import com.twitter.interests.thriftscala.UserInterestsResponse +import com.twitter.stitch.Stitch +import com.twitter.strato.client.Client +import com.twitter.strato.thrift.ScroogeConvImplicits._ + +@Singleton +class InterestServiceClient @Inject() ( + stratoClient: Client, + statsReceiver: StatsReceiver = NullStatsReceiver) + extends Logging { + + val interestsServiceStratoColumnPath = "interests/interestedInInterests" + val stats = statsReceiver.scope("interest_service_client") + val errorCounter = stats.counter("error") + + private val interestsFetcher = + stratoClient.fetcher[InterestedInInterestsFetchKey, UserInterestsResponse]( + interestsServiceStratoColumnPath, + checkTypes = true + ) + + def fetchUttInterestIds( + userId: Long + ): Stitch[Seq[Long]] = { + fetchInterestRelationships(userId) + .map(_.toSeq.flatten.flatMap(extractUttInterest)) + } + + def extractUttInterest( + interestRelationShip: InterestRelationship + ): Option[Long] = { + interestRelationShip match { + case InterestRelationship.V1(relationshipV1) => + relationshipV1.interestId match { + case InterestId.SemanticCore(semanticCoreInterest) => Some(semanticCoreInterest.id) + case _ => None + } + case _ => None + } + } + + def fetchCustomInterests( + userId: Long + ): Stitch[Seq[String]] = { + fetchInterestRelationships(userId) + .map(_.toSeq.flatten.flatMap(extractCustomInterest)) + } + + def extractCustomInterest( + interestRelationShip: InterestRelationship + ): Option[String] = { + interestRelationShip match { + case InterestRelationship.V1(relationshipV1) => + relationshipV1.interestId match { + case InterestId.FreeForm(freeFormInterest) => Some(freeFormInterest.interest) + case _ => None + } + case _ => None + } + } + + def fetchInterestRelationships( + userId: Long + ): Stitch[Option[Seq[InterestRelationship]]] = { + interestsFetcher + .fetch( + InterestedInInterestsFetchKey( + userId = userId, + labels = None, + None + )) + .map(_.v) + .map { + case Some(response) => + response.interests.interests.map { interests => + interests.collect { + case UserInterest(_, Some(interestData)) => + getInterestRelationship(interestData) + }.flatten + } + case _ => None + } + .rescue { + case e: Throwable => // we are swallowing all errors + logger.warn(s"interests could not be retrieved for user $userId due to ${e.getCause}") + errorCounter.incr + Stitch.None + } + } + + private def getInterestRelationship( + interestData: UserInterestData + ): Seq[InterestRelationship] = { + interestData match { + case UserInterestData.InterestedIn(interestModels) => + interestModels.collect { + case InterestedInInterestModel.ExplicitModel(model) => model + } + case _ => Nil + } + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/phone_storage_service/BUILD b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/phone_storage_service/BUILD new file mode 100644 index 0000000000..37a7f5906e --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/phone_storage_service/BUILD @@ -0,0 +1,14 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "finatra-internal/mtls-thriftmux/src/main/scala", + "finatra/inject/inject-thrift-client", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/common", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models", + "phonestorage/server/src/main/thrift/com/twitter/phonestorage/api:phone-storage-service-scala", + "stitch/stitch-core", + ], +) diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/phone_storage_service/PhoneStorageServiceClient.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/phone_storage_service/PhoneStorageServiceClient.scala new file mode 100644 index 0000000000..46e76b1623 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/phone_storage_service/PhoneStorageServiceClient.scala @@ -0,0 +1,34 @@ +package com.twitter.follow_recommendations.common.clients.phone_storage_service + +import com.twitter.cds.contact_consent_state.thriftscala.PurposeOfProcessing +import com.twitter.phonestorage.api.thriftscala.GetUserPhonesByUsersRequest +import com.twitter.phonestorage.api.thriftscala.PhoneStorageService +import com.twitter.stitch.Stitch +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class PhoneStorageServiceClient @Inject() ( + val phoneStorageService: PhoneStorageService.MethodPerEndpoint) { + + /** + * PSS can potentially return multiple phone records. + * The current implementation of getUserPhonesByUsers returns only a single phone for a single user_id but + * we can trivially support handling multiple in case that changes in the future. + */ + def getPhoneNumbers( + userId: Long, + purposeOfProcessing: PurposeOfProcessing, + forceCarrierLookup: Option[Boolean] = None + ): Stitch[Seq[String]] = { + val req = GetUserPhonesByUsersRequest( + userIds = Seq(userId), + forceCarrierLookup = forceCarrierLookup, + purposesOfProcessing = Some(Seq(purposeOfProcessing)) + ) + + Stitch.callFuture(phoneStorageService.getUserPhonesByUsers(req)) map { + _.userPhones.map(_.phoneNumber) + } + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/phone_storage_service/PhoneStorageServiceModule.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/phone_storage_service/PhoneStorageServiceModule.scala new file mode 100644 index 0000000000..90767a5097 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/phone_storage_service/PhoneStorageServiceModule.scala @@ -0,0 +1,12 @@ +package com.twitter.follow_recommendations.common.clients.phone_storage_service + +import com.twitter.finatra.mtls.thriftmux.modules.MtlsClient +import com.twitter.follow_recommendations.common.clients.common.BaseClientModule +import com.twitter.phonestorage.api.thriftscala.PhoneStorageService + +object PhoneStorageServiceModule + extends BaseClientModule[PhoneStorageService.MethodPerEndpoint] + with MtlsClient { + override val label = "phone-storage-service" + override val dest = "/s/ibis-ds-api/ibis-ds-api:thrift2" +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/real_time_real_graph/BUILD b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/real_time_real_graph/BUILD new file mode 100644 index 0000000000..f711bc3e70 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/real_time_real_graph/BUILD @@ -0,0 +1,20 @@ +scala_library( + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/com/google/inject:guice", + "3rdparty/jvm/com/google/inject/extensions:guice-assistedinject", + "3rdparty/jvm/net/codingwell:scala-guice", + "3rdparty/jvm/org/slf4j:slf4j-api", + "finatra/inject/inject-core/src/main/scala", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/strato", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/constants", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models", + "strato/config/columns/ml/featureStore:featureStore-strato-client", + "strato/config/columns/onboarding/userrecs:userrecs-strato-client", + "strato/src/main/scala/com/twitter/strato/client", + "util/util-slf4j-api/src/main/scala", + ], +) diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/real_time_real_graph/Engagement.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/real_time_real_graph/Engagement.scala new file mode 100644 index 0000000000..98c0555e71 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/real_time_real_graph/Engagement.scala @@ -0,0 +1,14 @@ +package com.twitter.follow_recommendations.common.clients.real_time_real_graph + +sealed trait EngagementType + +// We do not include SoftFollow since it's deprecated +object EngagementType { + object Click extends EngagementType + object Like extends EngagementType + object Mention extends EngagementType + object Retweet extends EngagementType + object ProfileView extends EngagementType +} + +case class Engagement(engagementType: EngagementType, timestamp: Long) diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/real_time_real_graph/EngagementScorer.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/real_time_real_graph/EngagementScorer.scala new file mode 100644 index 0000000000..c41fcc98ef --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/real_time_real_graph/EngagementScorer.scala @@ -0,0 +1,58 @@ +package com.twitter.follow_recommendations.common.clients.real_time_real_graph + +import com.twitter.conversions.DurationOps._ +import com.twitter.util.Time + +object EngagementScorer { + private[real_time_real_graph] val MemoryDecayHalfLife = 24.hour + private val ScoringFunctionBase = 0.5 + + def apply( + engagements: Map[Long, Seq[Engagement]], + engagementScoreMap: Map[EngagementType, Double], + minScore: Double = 0.0 + ): Seq[(Long, Double, Seq[EngagementType])] = { + val now = Time.now + engagements + .mapValues { engags => + val totalScore = engags.map { engagement => score(engagement, now, engagementScoreMap) }.sum + val engagementProof = getEngagementProof(engags, engagementScoreMap) + (totalScore, engagementProof) + } + .collect { case (uid, (score, proof)) if score > minScore => (uid, score, proof) } + .toSeq + .sortBy(-_._2) + } + + /** + * The engagement score is the base score decayed via timestamp, loosely model the human memory forgetting + * curve, see https://en.wikipedia.org/wiki/Forgetting_curve + */ + private[real_time_real_graph] def score( + engagement: Engagement, + now: Time, + engagementScoreMap: Map[EngagementType, Double] + ): Double = { + val timeLapse = math.max(now.inMillis - engagement.timestamp, 0) + val engagementScore = engagementScoreMap.getOrElse(engagement.engagementType, 0.0) + engagementScore * math.pow( + ScoringFunctionBase, + timeLapse.toDouble / MemoryDecayHalfLife.inMillis) + } + + private def getEngagementProof( + engagements: Seq[Engagement], + engagementScoreMap: Map[EngagementType, Double] + ): Seq[EngagementType] = { + + val filteredEngagement = engagements + .collectFirst { + case engagement + if engagement.engagementType != EngagementType.Click + && engagementScoreMap.get(engagement.engagementType).exists(_ > 0.0) => + engagement.engagementType + } + + Seq(filteredEngagement.getOrElse(EngagementType.Click)) + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/real_time_real_graph/RealTimeRealGraphClient.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/real_time_real_graph/RealTimeRealGraphClient.scala new file mode 100644 index 0000000000..16fedc8a0a --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/real_time_real_graph/RealTimeRealGraphClient.scala @@ -0,0 +1,128 @@ +package com.twitter.follow_recommendations.common.clients.real_time_real_graph + +import com.google.inject.Inject +import com.google.inject.Singleton +import com.twitter.conversions.DurationOps._ +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.snowflake.id.SnowflakeId +import com.twitter.stitch.Stitch +import com.twitter.strato.generated.client.ml.featureStore.TimelinesUserVertexOnUserClientColumn +import com.twitter.strato.generated.client.onboarding.userrecs.RealGraphScoresMhOnUserClientColumn +import com.twitter.util.Duration +import com.twitter.util.Time +import com.twitter.wtf.real_time_interaction_graph.thriftscala._ + +@Singleton +class RealTimeRealGraphClient @Inject() ( + timelinesUserVertexOnUserClientColumn: TimelinesUserVertexOnUserClientColumn, + realGraphScoresMhOnUserClientColumn: RealGraphScoresMhOnUserClientColumn) { + + def mapUserVertexToEngagementAndFilter(userVertex: UserVertex): Map[Long, Seq[Engagement]] = { + val minTimestamp = (Time.now - RealTimeRealGraphClient.MaxEngagementAge).inMillis + userVertex.outgoingInteractionMap.mapValues { interactions => + interactions + .flatMap { interaction => RealTimeRealGraphClient.toEngagement(interaction) }.filter( + _.timestamp >= minTimestamp) + }.toMap + } + + def getRecentProfileViewEngagements(userId: Long): Stitch[Map[Long, Seq[Engagement]]] = { + timelinesUserVertexOnUserClientColumn.fetcher + .fetch(userId).map(_.v).map { input => + input + .map { userVertex => + val targetToEngagements = mapUserVertexToEngagementAndFilter(userVertex) + targetToEngagements.mapValues { engagements => + engagements.filter(engagement => + engagement.engagementType == EngagementType.ProfileView) + } + }.getOrElse(Map.empty) + } + } + + def getUsersRecentlyEngagedWith( + userId: Long, + engagementScoreMap: Map[EngagementType, Double], + includeDirectFollowCandidates: Boolean, + includeNonDirectFollowCandidates: Boolean + ): Stitch[Seq[CandidateUser]] = { + val isNewUser = + SnowflakeId.timeFromIdOpt(userId).exists { signupTime => + (Time.now - signupTime) < RealTimeRealGraphClient.MaxNewUserAge + } + val updatedEngagementScoreMap = + if (isNewUser) + engagementScoreMap + (EngagementType.ProfileView -> RealTimeRealGraphClient.ProfileViewScore) + else engagementScoreMap + + Stitch + .join( + timelinesUserVertexOnUserClientColumn.fetcher.fetch(userId).map(_.v), + realGraphScoresMhOnUserClientColumn.fetcher.fetch(userId).map(_.v)).map { + case (Some(userVertex), Some(neighbors)) => + val engagements = mapUserVertexToEngagementAndFilter(userVertex) + + val candidatesAndScores: Seq[(Long, Double, Seq[EngagementType])] = + EngagementScorer.apply(engagements, engagementScoreMap = updatedEngagementScoreMap) + + val directNeighbors = neighbors.candidates.map(_._1).toSet + val (directFollows, nonDirectFollows) = candidatesAndScores + .partition { + case (id, _, _) => directNeighbors.contains(id) + } + + val candidates = + (if (includeNonDirectFollowCandidates) nonDirectFollows else Seq.empty) ++ + (if (includeDirectFollowCandidates) + directFollows.take(RealTimeRealGraphClient.MaxNumDirectFollow) + else Seq.empty) + + candidates.map { + case (id, score, proof) => + CandidateUser(id, Some(score)) + } + + case _ => Nil + } + } + + def getRealGraphWeights(userId: Long): Stitch[Map[Long, Double]] = + realGraphScoresMhOnUserClientColumn.fetcher + .fetch(userId) + .map( + _.v + .map(_.candidates.map(candidate => (candidate.userId, candidate.score)).toMap) + .getOrElse(Map.empty[Long, Double])) +} + +object RealTimeRealGraphClient { + private def toEngagement(interaction: Interaction): Option[Engagement] = { + // We do not include SoftFollow since it's deprecated + interaction match { + case Interaction.Retweet(Retweet(timestamp)) => + Some(Engagement(EngagementType.Retweet, timestamp)) + case Interaction.Favorite(Favorite(timestamp)) => + Some(Engagement(EngagementType.Like, timestamp)) + case Interaction.Click(Click(timestamp)) => Some(Engagement(EngagementType.Click, timestamp)) + case Interaction.Mention(Mention(timestamp)) => + Some(Engagement(EngagementType.Mention, timestamp)) + case Interaction.ProfileView(ProfileView(timestamp)) => + Some(Engagement(EngagementType.ProfileView, timestamp)) + case _ => None + } + } + + val MaxNumDirectFollow = 50 + val MaxEngagementAge: Duration = 14.days + val MaxNewUserAge: Duration = 30.days + val ProfileViewScore = 0.4 + val EngagementScoreMap = Map( + EngagementType.Like -> 1.0, + EngagementType.Retweet -> 1.0, + EngagementType.Mention -> 1.0 + ) + val StrongEngagementScoreMap = Map( + EngagementType.Like -> 1.0, + EngagementType.Retweet -> 1.0, + ) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/socialgraph/BUILD b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/socialgraph/BUILD new file mode 100644 index 0000000000..58b5001fd8 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/socialgraph/BUILD @@ -0,0 +1,26 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/com/github/nscala_time", + "3rdparty/jvm/com/google/inject:guice", + "3rdparty/jvm/com/google/inject/extensions:guice-assistedinject", + "3rdparty/jvm/net/codingwell:scala-guice", + "3rdparty/jvm/org/slf4j:slf4j-api", + "escherbird/src/scala/com/twitter/escherbird/util/stitchcache", + "finatra-internal/mtls-thriftmux/src/main/scala", + "finatra/inject/inject-core/src/main/scala", + "finatra/inject/inject-thrift-client/src/main/scala", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/common", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models", + "socialgraph/server/src/main/scala/com/twitter/socialgraph/util", + "src/thrift/com/twitter/socialgraph:thrift-scala", + "stitch/stitch-socialgraph", + "strato/config/columns/onboarding/socialGraphService:socialGraphService-strato-client", + "strato/src/main/scala/com/twitter/strato/client", + "util/util-core:scala", + ], +) diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/socialgraph/SocialGraphClient.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/socialgraph/SocialGraphClient.scala new file mode 100644 index 0000000000..3ad90b5eda --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/socialgraph/SocialGraphClient.scala @@ -0,0 +1,421 @@ +package com.twitter.follow_recommendations.common.clients.socialgraph + +import com.twitter.escherbird.util.stitchcache.StitchCache +import com.twitter.finagle.stats.NullStatsReceiver +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.follow_recommendations.common.base.StatsUtil +import com.twitter.follow_recommendations.common.models.FollowProof +import com.twitter.follow_recommendations.common.models.UserIdWithTimestamp +import com.twitter.inject.Logging +import com.twitter.socialgraph.thriftscala.EdgesRequest +import com.twitter.socialgraph.thriftscala.IdsRequest +import com.twitter.socialgraph.thriftscala.IdsResult +import com.twitter.socialgraph.thriftscala.LookupContext +import com.twitter.socialgraph.thriftscala.OverCapacity +import com.twitter.socialgraph.thriftscala.PageRequest +import com.twitter.socialgraph.thriftscala.RelationshipType +import com.twitter.socialgraph.thriftscala.SrcRelationship +import com.twitter.socialgraph.util.ByteBufferUtil +import com.twitter.stitch.Stitch +import com.twitter.stitch.socialgraph.SocialGraph +import com.twitter.strato.client.Fetcher +import com.twitter.strato.generated.client.onboarding.socialGraphService.IdsClientColumn +import com.twitter.util.Duration +import com.twitter.util.Time +import java.nio.ByteBuffer +import javax.inject.Inject +import javax.inject.Singleton + +case class RecentEdgesQuery( + userId: Long, + relations: Seq[RelationshipType], + // prefer to default value to better utilize the caching function of stitch + count: Option[Int] = Some(SocialGraphClient.MaxQuerySize), + performUnion: Boolean = true, + recentEdgesWindowOpt: Option[Duration] = None, + targets: Option[Seq[Long]] = None) + +case class EdgeRequestQuery( + userId: Long, + relation: RelationshipType, + count: Option[Int] = Some(SocialGraphClient.MaxQuerySize), + performUnion: Boolean = true, + recentEdgesWindowOpt: Option[Duration] = None, + targets: Option[Seq[Long]] = None) + +@Singleton +class SocialGraphClient @Inject() ( + socialGraph: SocialGraph, + idsClientColumn: IdsClientColumn, + statsReceiver: StatsReceiver = NullStatsReceiver) + extends Logging { + + private val stats = statsReceiver.scope(this.getClass.getSimpleName) + private val cacheStats = stats.scope("cache") + private val getIntersectionsStats = stats.scope("getIntersections") + private val getIntersectionsFromCachedColumnStats = + stats.scope("getIntersectionsFromCachedColumn") + private val getRecentEdgesStats = stats.scope("getRecentEdges") + private val getRecentEdgesCachedStats = stats.scope("getRecentEdgesCached") + private val getRecentEdgesFromCachedColumnStats = stats.scope("getRecentEdgesFromCachedColumn") + private val getRecentEdgesCachedInternalStats = stats.scope("getRecentEdgesCachedInternal") + private val getRecentEdgesWithTimeStats = stats.scope("getRecentEdgesWithTime") + + val sgsIdsFetcher: Fetcher[IdsRequest, Unit, IdsResult] = idsClientColumn.fetcher + + private val recentEdgesCache = StitchCache[RecentEdgesQuery, Seq[Long]]( + maxCacheSize = SocialGraphClient.MaxCacheSize, + ttl = SocialGraphClient.CacheTTL, + statsReceiver = cacheStats, + underlyingCall = getRecentEdges + ) + + def getRecentEdgesCached( + rq: RecentEdgesQuery, + useCachedStratoColumn: Boolean = true + ): Stitch[Seq[Long]] = { + getRecentEdgesCachedStats.counter("requests").incr() + if (useCachedStratoColumn) { + getRecentEdgesFromCachedColumn(rq) + } else { + StatsUtil.profileStitch( + getRecentEdgesCachedInternal(rq), + getRecentEdgesCachedInternalStats + ) + } + } + + def getRecentEdgesCachedInternal(rq: RecentEdgesQuery): Stitch[Seq[Long]] = { + recentEdgesCache.readThrough(rq) + } + + def getRecentEdgesFromCachedColumn(rq: RecentEdgesQuery): Stitch[Seq[Long]] = { + val pageRequest = rq.recentEdgesWindowOpt match { + case Some(recentEdgesWindow) => + PageRequest( + count = rq.count, + cursor = Some(getEdgeCursor(recentEdgesWindow)), + selectAll = Some(true) + ) + case _ => PageRequest(count = rq.count) + } + val idsRequest = IdsRequest( + rq.relations.map { relationshipType => + SrcRelationship( + source = rq.userId, + relationshipType = relationshipType, + targets = rq.targets + ) + }, + pageRequest = Some(pageRequest), + context = Some(LookupContext(performUnion = Some(rq.performUnion))) + ) + + val socialGraphStitch = sgsIdsFetcher + .fetch(idsRequest, Unit) + .map(_.v) + .map { result => + result + .map { idResult => + val userIds: Seq[Long] = idResult.ids + getRecentEdgesFromCachedColumnStats.stat("num_edges").add(userIds.size) + userIds + }.getOrElse(Seq.empty) + } + .rescue { + case e: Exception => + stats.counter(e.getClass.getSimpleName).incr() + Stitch.Nil + } + + StatsUtil.profileStitch( + socialGraphStitch, + getRecentEdgesFromCachedColumnStats + ) + } + + def getRecentEdges(rq: RecentEdgesQuery): Stitch[Seq[Long]] = { + val pageRequest = rq.recentEdgesWindowOpt match { + case Some(recentEdgesWindow) => + PageRequest( + count = rq.count, + cursor = Some(getEdgeCursor(recentEdgesWindow)), + selectAll = Some(true) + ) + case _ => PageRequest(count = rq.count) + } + val socialGraphStitch = socialGraph + .ids( + IdsRequest( + rq.relations.map { relationshipType => + SrcRelationship( + source = rq.userId, + relationshipType = relationshipType, + targets = rq.targets + ) + }, + pageRequest = Some(pageRequest), + context = Some(LookupContext(performUnion = Some(rq.performUnion))) + ) + ) + .map { idsResult => + val userIds: Seq[Long] = idsResult.ids + getRecentEdgesStats.stat("num_edges").add(userIds.size) + userIds + } + .rescue { + case e: OverCapacity => + stats.counter(e.getClass.getSimpleName).incr() + logger.warn("SGS Over Capacity", e) + Stitch.Nil + } + StatsUtil.profileStitch( + socialGraphStitch, + getRecentEdgesStats + ) + } + + // This method return recent edges of (userId, timeInMs) + def getRecentEdgesWithTime(rq: EdgeRequestQuery): Stitch[Seq[UserIdWithTimestamp]] = { + val pageRequest = rq.recentEdgesWindowOpt match { + case Some(recentEdgesWindow) => + PageRequest( + count = rq.count, + cursor = Some(getEdgeCursor(recentEdgesWindow)), + selectAll = Some(true) + ) + case _ => PageRequest(count = rq.count) + } + + val socialGraphStitch = socialGraph + .edges( + EdgesRequest( + SrcRelationship( + source = rq.userId, + relationshipType = rq.relation, + targets = rq.targets + ), + pageRequest = Some(pageRequest), + context = Some(LookupContext(performUnion = Some(rq.performUnion))) + ) + ) + .map { edgesResult => + val userIds = edgesResult.edges.map { socialEdge => + UserIdWithTimestamp(socialEdge.target, socialEdge.updatedAt) + } + getRecentEdgesWithTimeStats.stat("num_edges").add(userIds.size) + userIds + } + .rescue { + case e: OverCapacity => + stats.counter(e.getClass.getSimpleName).incr() + logger.warn("SGS Over Capacity", e) + Stitch.Nil + } + StatsUtil.profileStitch( + socialGraphStitch, + getRecentEdgesWithTimeStats + ) + } + + // This method returns the cursor for a time duration, such that all the edges returned by SGS will be created + // in the range (now-window, now) + def getEdgeCursor(window: Duration): ByteBuffer = { + val cursorInLong = (-(Time.now - window).inMilliseconds) << 20 + ByteBufferUtil.fromLong(cursorInLong) + } + + // notice that this is more expensive but more realtime than the GFS one + def getIntersections( + userId: Long, + candidateIds: Seq[Long], + numIntersectionIds: Int + ): Stitch[Map[Long, FollowProof]] = { + val socialGraphStitch: Stitch[Map[Long, FollowProof]] = Stitch + .collect(candidateIds.map { candidateId => + socialGraph + .ids( + IdsRequest( + Seq( + SrcRelationship(userId, RelationshipType.Following), + SrcRelationship(candidateId, RelationshipType.FollowedBy) + ), + pageRequest = Some(PageRequest(count = Some(numIntersectionIds))) + ) + ).map { idsResult => + getIntersectionsStats.stat("num_edges").add(idsResult.ids.size) + (candidateId -> FollowProof(idsResult.ids, idsResult.ids.size)) + } + }).map(_.toMap) + .rescue { + case e: OverCapacity => + stats.counter(e.getClass.getSimpleName).incr() + logger.warn("social graph over capacity in hydrating social proof", e) + Stitch.value(Map.empty) + } + StatsUtil.profileStitch( + socialGraphStitch, + getIntersectionsStats + ) + } + + def getIntersectionsFromCachedColumn( + userId: Long, + candidateIds: Seq[Long], + numIntersectionIds: Int + ): Stitch[Map[Long, FollowProof]] = { + val socialGraphStitch: Stitch[Map[Long, FollowProof]] = Stitch + .collect(candidateIds.map { candidateId => + val idsRequest = IdsRequest( + Seq( + SrcRelationship(userId, RelationshipType.Following), + SrcRelationship(candidateId, RelationshipType.FollowedBy) + ), + pageRequest = Some(PageRequest(count = Some(numIntersectionIds))) + ) + + sgsIdsFetcher + .fetch(idsRequest, Unit) + .map(_.v) + .map { resultOpt => + resultOpt.map { idsResult => + getIntersectionsFromCachedColumnStats.stat("num_edges").add(idsResult.ids.size) + candidateId -> FollowProof(idsResult.ids, idsResult.ids.size) + } + } + }).map(_.flatten.toMap) + .rescue { + case e: Exception => + stats.counter(e.getClass.getSimpleName).incr() + Stitch.value(Map.empty) + } + StatsUtil.profileStitch( + socialGraphStitch, + getIntersectionsFromCachedColumnStats + ) + } + + def getInvalidRelationshipUserIds( + userId: Long, + maxNumRelationship: Int = SocialGraphClient.MaxNumInvalidRelationship + ): Stitch[Seq[Long]] = { + getRecentEdges( + RecentEdgesQuery( + userId, + SocialGraphClient.InvalidRelationshipTypes, + Some(maxNumRelationship) + ) + ) + } + + def getInvalidRelationshipUserIdsFromCachedColumn( + userId: Long, + maxNumRelationship: Int = SocialGraphClient.MaxNumInvalidRelationship + ): Stitch[Seq[Long]] = { + getRecentEdgesFromCachedColumn( + RecentEdgesQuery( + userId, + SocialGraphClient.InvalidRelationshipTypes, + Some(maxNumRelationship) + ) + ) + } + + def getRecentFollowedUserIds(userId: Long): Stitch[Seq[Long]] = { + getRecentEdges( + RecentEdgesQuery( + userId, + Seq(RelationshipType.Following) + ) + ) + } + + def getRecentFollowedUserIdsFromCachedColumn(userId: Long): Stitch[Seq[Long]] = { + getRecentEdgesFromCachedColumn( + RecentEdgesQuery( + userId, + Seq(RelationshipType.Following) + ) + ) + } + + def getRecentFollowedUserIdsWithTime(userId: Long): Stitch[Seq[UserIdWithTimestamp]] = { + getRecentEdgesWithTime( + EdgeRequestQuery( + userId, + RelationshipType.Following + ) + ) + } + + def getRecentFollowedByUserIds(userId: Long): Stitch[Seq[Long]] = { + getRecentEdges( + RecentEdgesQuery( + userId, + Seq(RelationshipType.FollowedBy) + ) + ) + } + + def getRecentFollowedByUserIdsFromCachedColumn(userId: Long): Stitch[Seq[Long]] = { + getRecentEdgesFromCachedColumn( + RecentEdgesQuery( + userId, + Seq(RelationshipType.FollowedBy) + ) + ) + } + + def getRecentFollowedUserIdsWithTimeWindow( + userId: Long, + timeWindow: Duration + ): Stitch[Seq[Long]] = { + getRecentEdges( + RecentEdgesQuery( + userId, + Seq(RelationshipType.Following), + recentEdgesWindowOpt = Some(timeWindow) + ) + ) + } +} + +object SocialGraphClient { + + val MaxQuerySize: Int = 500 + val MaxCacheSize: Int = 5000000 + // Ref: src/thrift/com/twitter/socialgraph/social_graph_service.thrift + val MaxNumInvalidRelationship: Int = 5000 + val CacheTTL: Duration = Duration.fromHours(24) + + val InvalidRelationshipTypes: Seq[RelationshipType] = Seq( + RelationshipType.HideRecommendations, + RelationshipType.Blocking, + RelationshipType.BlockedBy, + RelationshipType.Muting, + RelationshipType.MutedBy, + RelationshipType.ReportedAsSpam, + RelationshipType.ReportedAsSpamBy, + RelationshipType.ReportedAsAbuse, + RelationshipType.ReportedAsAbuseBy, + RelationshipType.FollowRequestOutgoing, + RelationshipType.Following, + RelationshipType.UsedToFollow, + ) + + /** + * + * Whether to call SGS to validate each candidate based on the number of invalid relationship users + * prefetched during request building step. This aims to not omit any invalid candidates that are + * not filtered out in previous steps. + * If the number is 0, this might be a fail-opened SGS call. + * If the number is larger or equal to 5000, this could hit SGS page size limit. + * Both cases account for a small percentage of the total traffic (<5%). + * + * @param numInvalidRelationshipUsers number of invalid relationship users fetched from getInvalidRelationshipUserIds + * @return whether to enable post-ranker SGS predicate + */ + def enablePostRankerSgsPredicate(numInvalidRelationshipUsers: Int): Boolean = { + numInvalidRelationshipUsers == 0 || numInvalidRelationshipUsers >= MaxNumInvalidRelationship + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/socialgraph/SocialGraphModule.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/socialgraph/SocialGraphModule.scala new file mode 100644 index 0000000000..5a97448d1a --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/socialgraph/SocialGraphModule.scala @@ -0,0 +1,25 @@ +package com.twitter.follow_recommendations.common.clients.socialgraph + +import com.google.inject.Provides +import com.twitter.finagle.ThriftMux +import com.twitter.finatra.mtls.thriftmux.modules.MtlsClient +import com.twitter.follow_recommendations.common.clients.common.BaseClientModule +import com.twitter.socialgraph.thriftscala.SocialGraphService +import com.twitter.stitch.socialgraph.SocialGraph +import javax.inject.Singleton + +object SocialGraphModule + extends BaseClientModule[SocialGraphService.MethodPerEndpoint] + with MtlsClient { + override val label = "social-graph-service" + override val dest = "/s/socialgraph/socialgraph" + + override def configureThriftMuxClient(client: ThriftMux.Client): ThriftMux.Client = + client.withSessionQualifier.noFailFast + + @Provides + @Singleton + def providesStitchClient(futureIface: SocialGraphService.MethodPerEndpoint): SocialGraph = { + SocialGraph(futureIface) + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/strato/BUILD b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/strato/BUILD new file mode 100644 index 0000000000..3661887c83 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/strato/BUILD @@ -0,0 +1,30 @@ +scala_library( + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/com/google/inject:guice", + "3rdparty/jvm/com/google/inject/extensions:guice-assistedinject", + "3rdparty/jvm/net/codingwell:scala-guice", + "3rdparty/jvm/org/slf4j:slf4j-api", + "finatra/inject/inject-core/src/main/scala", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/constants", + "src/scala/com/twitter/onboarding/relevance/candidate_generation/utt/models", + "src/thrift/com/twitter/core_workflows/user_model:user_model-scala", + "src/thrift/com/twitter/frigate/data_pipeline:frigate-user-history-thrift-scala", + "src/thrift/com/twitter/hermit/candidate:hermit-candidate-scala", + "src/thrift/com/twitter/hermit/pop_geo:hermit-pop-geo-scala", + "src/thrift/com/twitter/onboarding/relevance/relatable_accounts:relatable_accounts-scala", + "src/thrift/com/twitter/onboarding/relevance/store:store-scala", + "src/thrift/com/twitter/recos/user_user_graph:user_user_graph-scala", + "src/thrift/com/twitter/search/account_search/extended_network:extended_network_users-scala", + "src/thrift/com/twitter/service/metastore/gen:thrift-scala", + "src/thrift/com/twitter/wtf/candidate:wtf-candidate-scala", + "src/thrift/com/twitter/wtf/ml:wtf-ml-output-thrift-scala", + "src/thrift/com/twitter/wtf/real_time_interaction_graph:wtf-real_time_interaction_graph-thrift-scala", + "src/thrift/com/twitter/wtf/triangular_loop:triangular_loop-scala", + "strato/src/main/scala/com/twitter/strato/client", + "util/util-slf4j-api/src/main/scala", + ], +) diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/strato/StratoClientModule.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/strato/StratoClientModule.scala new file mode 100644 index 0000000000..4046ac7545 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/strato/StratoClientModule.scala @@ -0,0 +1,249 @@ +package com.twitter.follow_recommendations.common.clients.strato + +import com.google.inject.name.Named +import com.google.inject.Provides +import com.google.inject.Singleton +import com.twitter.conversions.DurationOps._ +import com.twitter.core_workflows.user_model.thriftscala.CondensedUserState +import com.twitter.search.account_search.extended_network.thriftscala.ExtendedNetworkUserKey +import com.twitter.search.account_search.extended_network.thriftscala.ExtendedNetworkUserVal +import com.twitter.finagle.ThriftMux +import com.twitter.finagle.mtls.authentication.ServiceIdentifier +import com.twitter.finagle.thrift.Protocols +import com.twitter.follow_recommendations.common.constants.GuiceNamedConstants +import com.twitter.follow_recommendations.common.constants.ServiceConstants._ +import com.twitter.frigate.data_pipeline.candidate_generation.thriftscala.LatestEvents +import com.twitter.hermit.candidate.thriftscala.{Candidates => HermitCandidates} +import com.twitter.hermit.pop_geo.thriftscala.PopUsersInPlace +import com.twitter.onboarding.relevance.relatable_accounts.thriftscala.RelatableAccounts +import com.twitter.inject.TwitterModule +import com.twitter.onboarding.relevance.candidates.thriftscala.InterestBasedUserRecommendations +import com.twitter.onboarding.relevance.candidates.thriftscala.UTTInterest +import com.twitter.onboarding.relevance.store.thriftscala.WhoToFollowDismissEventDetails +import com.twitter.recos.user_user_graph.thriftscala.RecommendUserRequest +import com.twitter.recos.user_user_graph.thriftscala.RecommendUserResponse +import com.twitter.service.metastore.gen.thriftscala.UserRecommendabilityFeatures +import com.twitter.strato.catalog.Scan.Slice +import com.twitter.strato.client.Strato.{Client => StratoClient} +import com.twitter.strato.client.Client +import com.twitter.strato.client.Fetcher +import com.twitter.strato.client.Scanner +import com.twitter.strato.thrift.ScroogeConvImplicits._ +import com.twitter.wtf.candidate.thriftscala.CandidateSeq +import com.twitter.wtf.ml.thriftscala.CandidateFeatures +import com.twitter.wtf.real_time_interaction_graph.thriftscala.Interaction +import com.twitter.wtf.triangular_loop.thriftscala.{Candidates => TriangularLoopCandidates} +import com.twitter.strato.opcontext.Attribution._ + +object StratoClientModule extends TwitterModule { + + // column paths + val CosineFollowPath = "recommendations/similarity/similarUsersByFollowGraph.User" + val CosineListPath = "recommendations/similarity/similarUsersByListGraph.User" + val CuratedCandidatesPath = "onboarding/curatedAccounts" + val CuratedFilteredAccountsPath = "onboarding/filteredAccountsFromRecommendations" + val PopUsersInPlacePath = "onboarding/userrecs/popUsersInPlace" + val ProfileSidebarBlacklistPath = "recommendations/hermit/profile-sidebar-blacklist" + val RealTimeInteractionsPath = "hmli/realTimeInteractions" + val SimsPath = "recommendations/similarity/similarUsersBySims.User" + val DBV2SimsPath = "onboarding/userrecs/newSims.User" + val TriangularLoopsPath = "onboarding/userrecs/triangularLoops.User" + val TwoHopRandomWalkPath = "onboarding/userrecs/twoHopRandomWalk.User" + val UserRecommendabilityPath = "onboarding/userRecommendabilityWithLongKeys.User" + val UTTAccountRecommendationsPath = "onboarding/userrecs/utt_account_recommendations" + val UttSeedAccountsRecommendationPath = "onboarding/userrecs/utt_seed_accounts" + val UserStatePath = "onboarding/userState.User" + val WTFPostNuxFeaturesPath = "ml/featureStore/onboarding/wtfPostNuxFeatures.User" + val ElectionCandidatesPath = "onboarding/electionAccounts" + val UserUserGraphPath = "recommendations/userUserGraph" + val WtfDissmissEventsPath = "onboarding/wtfDismissEvents" + val RelatableAccountsPath = "onboarding/userrecs/relatableAccounts" + val ExtendedNetworkCandidatesPath = "search/account_search/extendedNetworkCandidatesMH" + val LabeledNotificationPath = "frigate/magicrecs/labeledPushRecsAggregated.User" + + @Provides + @Singleton + def stratoClient(serviceIdentifier: ServiceIdentifier): Client = { + val timeoutBudget = 500.milliseconds + StratoClient( + ThriftMux.client + .withRequestTimeout(timeoutBudget) + .withProtocolFactory(Protocols.binaryFactory( + stringLengthLimit = StringLengthLimit, + containerLengthLimit = ContainerLengthLimit))) + .withMutualTls(serviceIdentifier) + .build() + } + + // add strato putters, fetchers, scanners below: + @Provides + @Singleton + @Named(GuiceNamedConstants.COSINE_FOLLOW_FETCHER) + def cosineFollowFetcher(stratoClient: Client): Fetcher[Long, Unit, HermitCandidates] = + stratoClient.fetcher[Long, Unit, HermitCandidates](CosineFollowPath) + + @Provides + @Singleton + @Named(GuiceNamedConstants.COSINE_LIST_FETCHER) + def cosineListFetcher(stratoClient: Client): Fetcher[Long, Unit, HermitCandidates] = + stratoClient.fetcher[Long, Unit, HermitCandidates](CosineListPath) + + @Provides + @Singleton + @Named(GuiceNamedConstants.CURATED_COMPETITOR_ACCOUNTS_FETCHER) + def curatedBlacklistedAccountsFetcher(stratoClient: Client): Fetcher[String, Unit, Seq[Long]] = + stratoClient.fetcher[String, Unit, Seq[Long]](CuratedFilteredAccountsPath) + + @Provides + @Singleton + @Named(GuiceNamedConstants.CURATED_CANDIDATES_FETCHER) + def curatedCandidatesFetcher(stratoClient: Client): Fetcher[String, Unit, Seq[Long]] = + stratoClient.fetcher[String, Unit, Seq[Long]](CuratedCandidatesPath) + + @Provides + @Singleton + @Named(GuiceNamedConstants.POP_USERS_IN_PLACE_FETCHER) + def popUsersInPlaceFetcher(stratoClient: Client): Fetcher[String, Unit, PopUsersInPlace] = + stratoClient.fetcher[String, Unit, PopUsersInPlace](PopUsersInPlacePath) + + @Provides + @Singleton + @Named(GuiceNamedConstants.RELATABLE_ACCOUNTS_FETCHER) + def relatableAccountsFetcher(stratoClient: Client): Fetcher[String, Unit, RelatableAccounts] = + stratoClient.fetcher[String, Unit, RelatableAccounts](RelatableAccountsPath) + + @Provides + @Singleton + @Named(GuiceNamedConstants.PROFILE_SIDEBAR_BLACKLIST_SCANNER) + def profileSidebarBlacklistScanner( + stratoClient: Client + ): Scanner[(Long, Slice[Long]), Unit, (Long, Long), Unit] = + stratoClient.scanner[(Long, Slice[Long]), Unit, (Long, Long), Unit](ProfileSidebarBlacklistPath) + + @Provides + @Singleton + @Named(GuiceNamedConstants.REAL_TIME_INTERACTIONS_FETCHER) + def realTimeInteractionsFetcher( + stratoClient: Client + ): Fetcher[(Long, Long), Unit, Seq[Interaction]] = + stratoClient.fetcher[(Long, Long), Unit, Seq[Interaction]](RealTimeInteractionsPath) + + @Provides + @Singleton + @Named(GuiceNamedConstants.SIMS_FETCHER) + def simsFetcher(stratoClient: Client): Fetcher[Long, Unit, HermitCandidates] = + stratoClient.fetcher[Long, Unit, HermitCandidates](SimsPath) + + @Provides + @Singleton + @Named(GuiceNamedConstants.DBV2_SIMS_FETCHER) + def dbv2SimsFetcher(stratoClient: Client): Fetcher[Long, Unit, HermitCandidates] = + stratoClient.fetcher[Long, Unit, HermitCandidates](DBV2SimsPath) + + @Provides + @Singleton + @Named(GuiceNamedConstants.TRIANGULAR_LOOPS_FETCHER) + def triangularLoopsFetcher(stratoClient: Client): Fetcher[Long, Unit, TriangularLoopCandidates] = + stratoClient.fetcher[Long, Unit, TriangularLoopCandidates](TriangularLoopsPath) + + @Provides + @Singleton + @Named(GuiceNamedConstants.TWO_HOP_RANDOM_WALK_FETCHER) + def twoHopRandomWalkFetcher(stratoClient: Client): Fetcher[Long, Unit, CandidateSeq] = + stratoClient.fetcher[Long, Unit, CandidateSeq](TwoHopRandomWalkPath) + + @Provides + @Singleton + @Named(GuiceNamedConstants.USER_RECOMMENDABILITY_FETCHER) + def userRecommendabilityFetcher( + stratoClient: Client + ): Fetcher[Long, Unit, UserRecommendabilityFeatures] = + stratoClient.fetcher[Long, Unit, UserRecommendabilityFeatures](UserRecommendabilityPath) + + @Provides + @Singleton + @Named(GuiceNamedConstants.USER_STATE_FETCHER) + def userStateFetcher(stratoClient: Client): Fetcher[Long, Unit, CondensedUserState] = + stratoClient.fetcher[Long, Unit, CondensedUserState](UserStatePath) + + @Provides + @Singleton + @Named(GuiceNamedConstants.UTT_ACCOUNT_RECOMMENDATIONS_FETCHER) + def uttAccountRecommendationsFetcher( + stratoClient: Client + ): Fetcher[UTTInterest, Unit, InterestBasedUserRecommendations] = + stratoClient.fetcher[UTTInterest, Unit, InterestBasedUserRecommendations]( + UTTAccountRecommendationsPath) + + @Provides + @Singleton + @Named(GuiceNamedConstants.UTT_SEED_ACCOUNTS_FETCHER) + def uttSeedAccountRecommendationsFetcher( + stratoClient: Client + ): Fetcher[UTTInterest, Unit, InterestBasedUserRecommendations] = + stratoClient.fetcher[UTTInterest, Unit, InterestBasedUserRecommendations]( + UttSeedAccountsRecommendationPath) + + @Provides + @Singleton + @Named(GuiceNamedConstants.ELECTION_CANDIDATES_FETCHER) + def electionCandidatesFetcher(stratoClient: Client): Fetcher[String, Unit, Seq[Long]] = + stratoClient.fetcher[String, Unit, Seq[Long]](ElectionCandidatesPath) + + @Provides + @Singleton + @Named(GuiceNamedConstants.USER_USER_GRAPH_FETCHER) + def userUserGraphFetcher( + stratoClient: Client + ): Fetcher[RecommendUserRequest, Unit, RecommendUserResponse] = + stratoClient.fetcher[RecommendUserRequest, Unit, RecommendUserResponse](UserUserGraphPath) + + @Provides + @Singleton + @Named(GuiceNamedConstants.POST_NUX_WTF_FEATURES_FETCHER) + def wtfPostNuxFeaturesFetcher(stratoClient: Client): Fetcher[Long, Unit, CandidateFeatures] = { + val attribution = ManhattanAppId("starbuck", "wtf_starbuck") + stratoClient + .fetcher[Long, Unit, CandidateFeatures](WTFPostNuxFeaturesPath) + .withAttribution(attribution) + } + + @Provides + @Singleton + @Named(GuiceNamedConstants.EXTENDED_NETWORK) + def extendedNetworkFetcher( + stratoClient: Client + ): Fetcher[ExtendedNetworkUserKey, Unit, ExtendedNetworkUserVal] = { + stratoClient + .fetcher[ExtendedNetworkUserKey, Unit, ExtendedNetworkUserVal](ExtendedNetworkCandidatesPath) + } + + @Provides + @Singleton + @Named(GuiceNamedConstants.DISMISS_STORE_SCANNER) + def dismissStoreScanner( + stratoClient: Client + ): Scanner[ + (Long, Slice[(Long, Long)]), + Unit, + (Long, (Long, Long)), + WhoToFollowDismissEventDetails + ] = + stratoClient.scanner[ + (Long, Slice[(Long, Long)]), // PKEY: userId, LKEY: (-ts, candidateId) + Unit, + (Long, (Long, Long)), + WhoToFollowDismissEventDetails + ](WtfDissmissEventsPath) + + @Provides + @Singleton + @Named(GuiceNamedConstants.LABELED_NOTIFICATION_FETCHER) + def labeledNotificationFetcher( + stratoClient: Client + ): Fetcher[Long, Unit, LatestEvents] = { + stratoClient + .fetcher[Long, Unit, LatestEvents](LabeledNotificationPath) + } + +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/user_state/BUILD b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/user_state/BUILD new file mode 100644 index 0000000000..16a82a3025 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/user_state/BUILD @@ -0,0 +1,17 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/com/google/inject:guice", + "3rdparty/jvm/net/codingwell:scala-guice", + "3rdparty/jvm/org/slf4j:slf4j-api", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/cache", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/strato", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/deciders", + "stitch/stitch-core", + "strato/src/main/scala/com/twitter/strato/client", + "user-signal-service/thrift/src/main/thrift:thrift-scala", + ], +) diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/user_state/UserStateClient.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/user_state/UserStateClient.scala new file mode 100644 index 0000000000..fe8101261c --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/user_state/UserStateClient.scala @@ -0,0 +1,83 @@ +package com.twitter.follow_recommendations.common.clients.user_state + +import com.google.inject.name.Named +import com.twitter.conversions.DurationOps._ +import com.twitter.core_workflows.user_model.thriftscala.CondensedUserState +import com.twitter.core_workflows.user_model.thriftscala.UserState +import com.twitter.decider.Decider +import com.twitter.decider.RandomRecipient +import com.twitter.finagle.Memcached.Client +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.finagle.util.DefaultTimer +import com.twitter.follow_recommendations.common.base.StatsUtil +import com.twitter.follow_recommendations.common.clients.cache.MemcacheClient +import com.twitter.follow_recommendations.common.clients.cache.ThriftEnumOptionBijection +import com.twitter.follow_recommendations.common.constants.GuiceNamedConstants +import com.twitter.follow_recommendations.configapi.deciders.DeciderKey +import com.twitter.stitch.Stitch +import com.twitter.strato.client.Fetcher +import com.twitter.util.Duration +import javax.inject.Inject +import javax.inject.Singleton +import java.lang.{Long => JLong} + +@Singleton +class UserStateClient @Inject() ( + @Named(GuiceNamedConstants.USER_STATE_FETCHER) userStateFetcher: Fetcher[ + Long, + Unit, + CondensedUserState + ], + client: Client, + statsReceiver: StatsReceiver, + decider: Decider = Decider.False) { + + private val stats: StatsReceiver = statsReceiver.scope("user_state_client") + + // client to memcache cluster + val bijection = new ThriftEnumOptionBijection[UserState](UserState.apply) + val memcacheClient = MemcacheClient[Option[UserState]]( + client = client, + dest = "/s/cache/follow_recos_service:twemcaches", + valueBijection = bijection, + ttl = UserStateClient.CacheTTL, + statsReceiver = stats.scope("twemcache") + ) + + def getUserState(userId: Long): Stitch[Option[UserState]] = { + val deciderKey: String = DeciderKey.EnableDistributedCaching.toString + val enableDistributedCaching: Boolean = decider.isAvailable(deciderKey, Some(RandomRecipient)) + val userStateStitch: Stitch[Option[UserState]] = + enableDistributedCaching match { + // read from memcache + case true => memcacheClient.readThrough( + // add a key prefix to address cache key collisions + key = "UserStateClient" + userId.toString, + underlyingCall = () => fetchUserState(userId) + ) + case false => fetchUserState(userId) + } + val userStateStitchWithTimeout: Stitch[Option[UserState]] = + userStateStitch + // set a 150ms timeout limit for user state fetches + .within(150.milliseconds)(DefaultTimer) + .rescue { + case e: Exception => + stats.scope("rescued").counter(e.getClass.getSimpleName).incr() + Stitch(None) + } + // profile the latency of stitch call and return the result + StatsUtil.profileStitch( + userStateStitchWithTimeout, + stats.scope("getUserState") + ) + } + + def fetchUserState(userId: JLong): Stitch[Option[UserState]] = { + userStateFetcher.fetch(userId).map(_.v.flatMap(_.userState)) + } +} + +object UserStateClient { + val CacheTTL: Duration = Duration.fromHours(6) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/constants/BUILD b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/constants/BUILD new file mode 100644 index 0000000000..dc9335d5b1 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/constants/BUILD @@ -0,0 +1,9 @@ +scala_library( + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models", + "util/util-core/src/main/scala/com/twitter/conversions", + ], +) diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/constants/CandidateAlgorithmTypeConstants.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/constants/CandidateAlgorithmTypeConstants.scala new file mode 100644 index 0000000000..7c7892c022 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/constants/CandidateAlgorithmTypeConstants.scala @@ -0,0 +1,91 @@ +package com.twitter.follow_recommendations.common.constants + +import com.twitter.hermit.constants.AlgorithmFeedbackTokens.AlgorithmToFeedbackTokenMap +import com.twitter.hermit.model.Algorithm._ +import com.twitter.follow_recommendations.common.models.AlgorithmType + +object CandidateAlgorithmTypeConstants { + + /** + * Each algorithm is based on one, or more, of the 4 types of information we have on users, + * described in [[AlgorithmType]]. Assignment of algorithms to these categories are based on + */ + private val AlgorithmIdToType: Map[String, Set[AlgorithmType.Value]] = Map( + // Activity Algorithms: + AlgorithmToFeedbackTokenMap(NewFollowingSimilarUser).toString -> Set(AlgorithmType.Activity), + AlgorithmToFeedbackTokenMap(Sims).toString -> Set(AlgorithmType.Activity), + AlgorithmToFeedbackTokenMap(NewFollowingSimilarUserSalsa).toString -> Set( + AlgorithmType.Activity), + AlgorithmToFeedbackTokenMap(RecentEngagementNonDirectFollow).toString -> Set( + AlgorithmType.Activity), + AlgorithmToFeedbackTokenMap(RecentEngagementSimilarUser).toString -> Set( + AlgorithmType.Activity), + AlgorithmToFeedbackTokenMap(RecentEngagementSarusOcCur).toString -> Set(AlgorithmType.Activity), + AlgorithmToFeedbackTokenMap(RecentSearchBasedRec).toString -> Set(AlgorithmType.Activity), + AlgorithmToFeedbackTokenMap(TwistlyTweetAuthors).toString -> Set(AlgorithmType.Activity), + AlgorithmToFeedbackTokenMap(Follow2VecNearestNeighbors).toString -> Set(AlgorithmType.Activity), + AlgorithmToFeedbackTokenMap(EmailTweetClick).toString -> Set(AlgorithmType.Activity), + AlgorithmToFeedbackTokenMap(RepeatedProfileVisits).toString -> Set(AlgorithmType.Activity), + AlgorithmToFeedbackTokenMap(GoodTweetClickEngagements).toString -> Set(AlgorithmType.Activity), + AlgorithmToFeedbackTokenMap(TweetShareEngagements).toString -> Set(AlgorithmType.Activity), + AlgorithmToFeedbackTokenMap(TweetSharerToShareRecipientEngagements).toString -> Set( + AlgorithmType.Activity), + AlgorithmToFeedbackTokenMap(TweetAuthorToShareRecipientEngagements).toString -> Set( + AlgorithmType.Activity), + AlgorithmToFeedbackTokenMap(LinearRegressionFollow2VecNearestNeighbors).toString -> Set( + AlgorithmType.Activity), + AlgorithmToFeedbackTokenMap(NUXLOHistory).toString -> Set(AlgorithmType.Activity), + AlgorithmToFeedbackTokenMap(TrafficAttributionAccounts).toString -> Set(AlgorithmType.Activity), + AlgorithmToFeedbackTokenMap(RealGraphOonV2).toString -> Set(AlgorithmType.Activity), + AlgorithmToFeedbackTokenMap(MagicRecsRecentEngagements).toString -> Set(AlgorithmType.Activity), + AlgorithmToFeedbackTokenMap(NotificationEngagement).toString -> Set(AlgorithmType.Activity), + // Social Algorithms: + AlgorithmToFeedbackTokenMap(TwoHopRandomWalk).toString -> Set(AlgorithmType.Social), + AlgorithmToFeedbackTokenMap(RealTimeMutualFollow).toString -> Set(AlgorithmType.Social), + AlgorithmToFeedbackTokenMap(ForwardPhoneBook).toString -> Set(AlgorithmType.Social), + AlgorithmToFeedbackTokenMap(ForwardEmailBook).toString -> Set(AlgorithmType.Social), + AlgorithmToFeedbackTokenMap(NewFollowingNewFollowingExpansion).toString -> Set( + AlgorithmType.Social), + AlgorithmToFeedbackTokenMap(NewFollowingSarusCoOccurSocialProof).toString -> Set( + AlgorithmType.Social), + AlgorithmToFeedbackTokenMap(ReverseEmailBookIbis).toString -> Set(AlgorithmType.Social), + AlgorithmToFeedbackTokenMap(ReversePhoneBook).toString -> Set(AlgorithmType.Social), + AlgorithmToFeedbackTokenMap(StrongTiePredictionRec).toString -> Set(AlgorithmType.Social), + AlgorithmToFeedbackTokenMap(StrongTiePredictionRecWithSocialProof).toString -> Set( + AlgorithmType.Social), + AlgorithmToFeedbackTokenMap(OnlineStrongTiePredictionRec).toString -> Set(AlgorithmType.Social), + AlgorithmToFeedbackTokenMap(OnlineStrongTiePredictionRecNoCaching).toString -> Set( + AlgorithmType.Social), + AlgorithmToFeedbackTokenMap(TriangularLoop).toString -> Set(AlgorithmType.Social), + AlgorithmToFeedbackTokenMap(StrongTiePredictionPmi).toString -> Set(AlgorithmType.Social), + AlgorithmToFeedbackTokenMap(OnlineStrongTiePredictionRAB).toString -> Set(AlgorithmType.Social), + // Geo Algorithms: + AlgorithmToFeedbackTokenMap(PopCountryBackFill).toString -> Set(AlgorithmType.Geo), + AlgorithmToFeedbackTokenMap(PopCountry).toString -> Set(AlgorithmType.Geo), + AlgorithmToFeedbackTokenMap(PopGeohash).toString -> Set(AlgorithmType.Geo), +// AlgorithmToFeedbackTokenMap(PopGeohashRealGraph).toString -> Set(AlgorithmType.Geo), + AlgorithmToFeedbackTokenMap(EngagedFollowerRatio).toString -> Set(AlgorithmType.Geo), + AlgorithmToFeedbackTokenMap(CrowdSearchAccounts).toString -> Set(AlgorithmType.Geo), + AlgorithmToFeedbackTokenMap(OrganicFollowAccounts).toString -> Set(AlgorithmType.Geo), + AlgorithmToFeedbackTokenMap(PopGeohashQualityFollow).toString -> Set(AlgorithmType.Geo), + AlgorithmToFeedbackTokenMap(PPMILocaleFollow).toString -> Set(AlgorithmType.Geo), + // Interest Algorithms: + AlgorithmToFeedbackTokenMap(TttInterest).toString -> Set(AlgorithmType.Interest), + AlgorithmToFeedbackTokenMap(UttInterestRelatedUsers).toString -> Set(AlgorithmType.Interest), + AlgorithmToFeedbackTokenMap(UttSeedAccounts).toString -> Set(AlgorithmType.Interest), + AlgorithmToFeedbackTokenMap(UttProducerExpansion).toString -> Set(AlgorithmType.Interest), + // Hybrid (more than one type) Algorithms: + AlgorithmToFeedbackTokenMap(UttProducerOfflineMbcgV1).toString -> Set( + AlgorithmType.Interest, + AlgorithmType.Geo), + AlgorithmToFeedbackTokenMap(CuratedAccounts).toString -> Set( + AlgorithmType.Interest, + AlgorithmType.Geo), + AlgorithmToFeedbackTokenMap(UserUserGraph).toString -> Set( + AlgorithmType.Social, + AlgorithmType.Activity), + ) + def getAlgorithmTypes(algoId: String): Set[String] = { + AlgorithmIdToType.get(algoId).map(_.map(_.toString)).getOrElse(Set.empty) + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/constants/GuiceNamedConstants.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/constants/GuiceNamedConstants.scala new file mode 100644 index 0000000000..d3d61fa430 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/constants/GuiceNamedConstants.scala @@ -0,0 +1,43 @@ +package com.twitter.follow_recommendations.common.constants + +object GuiceNamedConstants { + final val PRODUCER_SIDE_FEATURE_SWITCHES = "PRODUCER_SIDE_FEATURE_SWITCHES" + final val CLIENT_EVENT_LOGGER = "CLIENT_EVENT_LOGGER" + final val COSINE_FOLLOW_FETCHER = "cosine_follow_fetcher" + final val COSINE_LIST_FETCHER = "cosine_list_fetcher" + final val CURATED_CANDIDATES_FETCHER = "curated_candidates_fetcher" + final val CURATED_COMPETITOR_ACCOUNTS_FETCHER = "curated_competitor_accounts_fetcher" + final val POP_USERS_IN_PLACE_FETCHER = "pop_users_in_place_fetcher" + final val PROFILE_SIDEBAR_BLACKLIST_SCANNER = "profile_sidebar_blacklist_scanner" + final val REQUEST_LOGGER = "REQUEST_LOGGER" + final val FLOW_LOGGER = "FLOW_LOGGER" + final val REAL_TIME_INTERACTIONS_FETCHER = "real_time_interactions_fetcher" + final val SIMS_FETCHER = "sims_fetcher" + final val DBV2_SIMS_FETCHER = "dbv2_sims_fetcher" + + final val TRIANGULAR_LOOPS_FETCHER = "triangular_loops_fetcher" + final val TWO_HOP_RANDOM_WALK_FETCHER = "two_hop_random_walk_fetcher" + final val USER_RECOMMENDABILITY_FETCHER = "user_recommendability_fetcher" + final val USER_STATE_FETCHER = "user_state_fetcher" + final val UTT_ACCOUNT_RECOMMENDATIONS_FETCHER = "utt_account_recomendations_fetcher" + final val UTT_SEED_ACCOUNTS_FETCHER = "utt_seed_accounts_fetcher" + + final val ELECTION_CANDIDATES_FETCHER = "election_candidates_fetcher" + final val POST_NUX_WTF_FEATURES_FETCHER = "post_nux_wtf_features_fetcher" + + final val USER_USER_GRAPH_FETCHER = "user_user_graph_fetcher" + final val DISMISS_STORE_SCANNER = "dismiss_store_scanner" + final val LABELED_NOTIFICATION_FETCHER = "labeled_notification_scanner" + + final val STP_EP_SCORER = "stp_ep_scorer" + final val STP_DBV2_SCORER = "stp_dbv2_scorer" + final val STP_RAB_DBV2_SCORER = "stp_rab_dbv2_scorer" + + final val EXTENDED_NETWORK = "extended_network_candidates" + + // scoring client constants + final val WTF_PROD_DEEPBIRDV2_CLIENT = "wtf_prod_deepbirdv2_client" + + // ann clients + final val RELATABLE_ACCOUNTS_FETCHER = "relatable_accounts_fetcher" +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/constants/ServiceConstants.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/constants/ServiceConstants.scala new file mode 100644 index 0000000000..6aade704c5 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/constants/ServiceConstants.scala @@ -0,0 +1,15 @@ +package com.twitter.follow_recommendations.common.constants + +import com.twitter.conversions.StorageUnitOps._ + +object ServiceConstants { + + /** thrift client response size limits + * these were estimated using monitoring dashboard + * 3MB network usage per second / 25 rps ~ 120KB/req << 1MB + * we give some buffer here in case some requests require more data than others + */ + val StringLengthLimit: Long = + 10.megabyte.inBytes + val ContainerLengthLimit: Long = 1.megabyte.inBytes +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/adapters/BUILD b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/adapters/BUILD new file mode 100644 index 0000000000..b4afee5909 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/adapters/BUILD @@ -0,0 +1,82 @@ +scala_library( + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + ":candidate-algorithm-adapter", + ":client-context-adapter", + ":post-nux-algorithm-adapter", + ":pre-fetched-feature-adapter", + ], +) + +target( + name = "common", + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/org/slf4j:slf4j-api", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/strato", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/constants", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/common", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/utils", + "src/java/com/twitter/ml/api:api-base", + "src/scala/com/twitter/ml/api/util", + "src/scala/com/twitter/onboarding/relevance/util/metadata", + "util/util-slf4j-api/src/main/scala", + ], +) + +scala_library( + name = "candidate-algorithm-adapter", + sources = [ + "CandidateAlgorithmAdapter.scala", + ], + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + ":common", + "hermit/hermit-core/src/main/scala/com/twitter/hermit/constants", + ], +) + +scala_library( + name = "client-context-adapter", + sources = [ + "ClientContextAdapter.scala", + ], + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + ":common", + "snowflake/src/main/scala/com/twitter/snowflake/id", + ], +) + +scala_library( + name = "post-nux-algorithm-adapter", + sources = [ + "PostNuxAlgorithmAdapter.scala", + ], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + ":common", + "src/scala/com/twitter/ml/featurestore/catalog/features/customer_journey:post-nux-algorithm-aggregate", + ], +) + +scala_library( + name = "pre-fetched-feature-adapter", + sources = [ + "PreFetchedFeatureAdapter.scala", + ], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + ":common", + ], +) diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/adapters/CandidateAlgorithmAdapter.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/adapters/CandidateAlgorithmAdapter.scala new file mode 100644 index 0000000000..7a487a95bb --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/adapters/CandidateAlgorithmAdapter.scala @@ -0,0 +1,72 @@ +package com.twitter.follow_recommendations.common.feature_hydration.adapters + +import com.twitter.follow_recommendations.common.models.UserCandidateSourceDetails +import com.twitter.hermit.constants.AlgorithmFeedbackTokens.AlgorithmToFeedbackTokenMap +import com.twitter.hermit.model.Algorithm +import com.twitter.hermit.model.Algorithm.Algorithm +import com.twitter.hermit.model.Algorithm.UttProducerOfflineMbcgV1 +import com.twitter.hermit.model.Algorithm.UttProducerOnlineMbcgV1 +import com.twitter.ml.api.DataRecord +import com.twitter.ml.api.Feature.SparseBinary +import com.twitter.ml.api.Feature.SparseContinuous +import com.twitter.ml.api.FeatureContext +import com.twitter.ml.api.IRecordOneToOneAdapter +import com.twitter.ml.api.util.FDsl._ + +object CandidateAlgorithmAdapter + extends IRecordOneToOneAdapter[Option[UserCandidateSourceDetails]] { + + val CANDIDATE_ALGORITHMS: SparseBinary = new SparseBinary("candidate.source.algorithm_ids") + val CANDIDATE_SOURCE_SCORES: SparseContinuous = + new SparseContinuous("candidate.source.scores") + val CANDIDATE_SOURCE_RANKS: SparseContinuous = + new SparseContinuous("candidate.source.ranks") + + override val getFeatureContext: FeatureContext = new FeatureContext( + CANDIDATE_ALGORITHMS, + CANDIDATE_SOURCE_SCORES, + CANDIDATE_SOURCE_RANKS + ) + + /** list of candidate source remaps to avoid creating different features for experimental sources. + * the LHS should contain the experimental source, and the RHS should contain the prod source. + */ + def remapCandidateSource(a: Algorithm): Algorithm = a match { + case UttProducerOnlineMbcgV1 => UttProducerOfflineMbcgV1 + case _ => a + } + + // add the list of algorithm feedback tokens (integers) as a sparse binary feature + override def adaptToDataRecord( + userCandidateSourceDetailsOpt: Option[UserCandidateSourceDetails] + ): DataRecord = { + val dr = new DataRecord() + userCandidateSourceDetailsOpt.foreach { userCandidateSourceDetails => + val scoreMap = for { + (source, scoreOpt) <- userCandidateSourceDetails.candidateSourceScores + score <- scoreOpt + algo <- Algorithm.withNameOpt(source.name) + algoId <- AlgorithmToFeedbackTokenMap.get(remapCandidateSource(algo)) + } yield algoId.toString -> score + val rankMap = for { + (source, rank) <- userCandidateSourceDetails.candidateSourceRanks + algo <- Algorithm.withNameOpt(source.name) + algoId <- AlgorithmToFeedbackTokenMap.get(remapCandidateSource(algo)) + } yield algoId.toString -> rank.toDouble + + val algoIds = scoreMap.keys.toSet ++ rankMap.keys.toSet + + // hydrate if not empty + if (rankMap.nonEmpty) { + dr.setFeatureValue(CANDIDATE_SOURCE_RANKS, rankMap) + } + if (scoreMap.nonEmpty) { + dr.setFeatureValue(CANDIDATE_SOURCE_SCORES, scoreMap) + } + if (algoIds.nonEmpty) { + dr.setFeatureValue(CANDIDATE_ALGORITHMS, algoIds) + } + } + dr + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/adapters/ClientContextAdapter.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/adapters/ClientContextAdapter.scala new file mode 100644 index 0000000000..9aa4bdb0d2 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/adapters/ClientContextAdapter.scala @@ -0,0 +1,79 @@ +package com.twitter.follow_recommendations.common.feature_hydration.adapters + +import com.twitter.follow_recommendations.common.models.DisplayLocation +import com.twitter.ml.api.Feature.Binary +import com.twitter.ml.api.Feature.Continuous +import com.twitter.ml.api.Feature.Discrete +import com.twitter.ml.api.Feature.Text +import com.twitter.ml.api.util.FDsl._ +import com.twitter.ml.api.DataRecord +import com.twitter.ml.api.FeatureContext +import com.twitter.ml.api.IRecordOneToOneAdapter +import com.twitter.onboarding.relevance.util.metadata.LanguageUtil +import com.twitter.product_mixer.core.model.marshalling.request.ClientContext +import com.twitter.snowflake.id.SnowflakeId + +object ClientContextAdapter extends IRecordOneToOneAdapter[(ClientContext, DisplayLocation)] { + + // we name features with `user.account` for relatively static user-related features + val USER_COUNTRY: Text = new Text("user.account.country") + val USER_LANGUAGE: Text = new Text("user.account.language") + // we name features with `user.context` for more dynamic user-related features + val USER_LANGUAGE_PREFIX: Text = new Text("user.context.language_prefix") + val USER_CLIENT: Discrete = new Discrete("user.context.client") + val USER_AGE: Continuous = new Continuous("user.context.age") + val USER_IS_RECENT: Binary = new Binary("user.is.recent") + // we name features with `meta` for meta info about the WTF recommendation request + val META_DISPLAY_LOCATION: Text = new Text("meta.display_location") + val META_POSITION: Discrete = new Discrete("meta.position") + // This indicates whether a data point is from a random serving policy + val META_IS_RANDOM: Binary = new Binary("prediction.engine.is_random") + + val RECENT_WIN_IN_DAYS: Int = 30 + val GOAL_META_POSITION: Long = 1L + val GOAL_META_IS_RANDOM: Boolean = true + + override val getFeatureContext: FeatureContext = new FeatureContext( + USER_COUNTRY, + USER_LANGUAGE, + USER_AGE, + USER_LANGUAGE_PREFIX, + USER_CLIENT, + USER_IS_RECENT, + META_DISPLAY_LOCATION, + META_POSITION, + META_IS_RANDOM + ) + + /** + * we only want to set the relevant fields iff they exist to eliminate redundant information + * we do some simple normalization on the language code + * we set META_POSITION to 1 always + * we set META_IS_RANDOM to true always to simulate a random serving distribution + * @param record ClientContext and DisplayLocation from the request + */ + override def adaptToDataRecord(target: (ClientContext, DisplayLocation)): DataRecord = { + val dr = new DataRecord() + val cc = target._1 + val dl = target._2 + cc.countryCode.foreach(countryCode => dr.setFeatureValue(USER_COUNTRY, countryCode)) + cc.languageCode.foreach(rawLanguageCode => { + val userLanguage = LanguageUtil.simplifyLanguage(rawLanguageCode) + val userLanguagePrefix = userLanguage.take(2) + dr.setFeatureValue(USER_LANGUAGE, userLanguage) + dr.setFeatureValue(USER_LANGUAGE_PREFIX, userLanguagePrefix) + }) + cc.appId.foreach(appId => dr.setFeatureValue(USER_CLIENT, appId)) + cc.userId.foreach(id => + SnowflakeId.timeFromIdOpt(id).map { signupTime => + val userAge = signupTime.untilNow.inMillis.toDouble + dr.setFeatureValue(USER_AGE, userAge) + dr.setFeatureValue(USER_IS_RECENT, signupTime.untilNow.inDays <= RECENT_WIN_IN_DAYS) + signupTime.untilNow.inDays + }) + dr.setFeatureValue(META_DISPLAY_LOCATION, dl.toFsName) + dr.setFeatureValue(META_POSITION, GOAL_META_POSITION) + dr.setFeatureValue(META_IS_RANDOM, GOAL_META_IS_RANDOM) + dr + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/adapters/PostNuxAlgorithmAdapter.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/adapters/PostNuxAlgorithmAdapter.scala new file mode 100644 index 0000000000..e8fe745a02 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/adapters/PostNuxAlgorithmAdapter.scala @@ -0,0 +1,151 @@ +package com.twitter.follow_recommendations.common.feature_hydration.adapters + +import com.twitter.ml.api.DataRecord +import com.twitter.ml.api.Feature +import com.twitter.ml.api.Feature.Continuous +import com.twitter.ml.api.FeatureContext +import com.twitter.ml.api.IRecordOneToOneAdapter +import com.twitter.ml.api.util.FDsl._ +import com.twitter.ml.featurestore.catalog.features.customer_journey.PostNuxAlgorithmFeatures +import com.twitter.ml.featurestore.catalog.features.customer_journey.PostNuxAlgorithmIdAggregateFeatureGroup +import com.twitter.ml.featurestore.catalog.features.customer_journey.PostNuxAlgorithmTypeAggregateFeatureGroup +import scala.collection.JavaConverters._ + +object PostNuxAlgorithmIdAdapter extends PostNuxAlgorithmAdapter { + override val PostNuxAlgorithmFeatureGroup: PostNuxAlgorithmFeatures = + PostNuxAlgorithmIdAggregateFeatureGroup + + // To keep the length of feature names reasonable, we remove the prefix added by FeatureStore. + override val FeatureStorePrefix: String = + "wtf_algorithm_id.customer_journey.post_nux_algorithm_id_aggregate_feature_group." +} + +object PostNuxAlgorithmTypeAdapter extends PostNuxAlgorithmAdapter { + override val PostNuxAlgorithmFeatureGroup: PostNuxAlgorithmFeatures = + PostNuxAlgorithmTypeAggregateFeatureGroup + + // To keep the length of feature names reasonable, we remove the prefix added by FeatureStore. + override val FeatureStorePrefix: String = + "wtf_algorithm_type.customer_journey.post_nux_algorithm_type_aggregate_feature_group." +} + +trait PostNuxAlgorithmAdapter extends IRecordOneToOneAdapter[DataRecord] { + + val PostNuxAlgorithmFeatureGroup: PostNuxAlgorithmFeatures + + // The string that is attached to the feature name when it is fetched from feature store. + val FeatureStorePrefix: String + + /** + * + * This stores transformed aggregate features for PostNux algorithm aggregate features. The + * transformation here is log-ratio, where ratio is the raw value divided by # of impressions. + */ + case class TransformedAlgorithmFeatures( + ratioLog: Continuous) { + def getFeatures: Seq[Continuous] = Seq(ratioLog) + } + + private def applyFeatureStorePrefix(feature: Continuous) = new Continuous( + s"$FeatureStorePrefix${feature.getFeatureName}") + + // The list of input features WITH the prefix assigned to them by FeatureStore. + lazy val allInputFeatures: Seq[Seq[Continuous]] = Seq( + PostNuxAlgorithmFeatureGroup.Aggregate7DayFeatures.map(applyFeatureStorePrefix), + PostNuxAlgorithmFeatureGroup.Aggregate30DayFeatures.map(applyFeatureStorePrefix) + ) + + // This is a list of the features WITHOUT the prefix assigned to them by FeatureStore. + lazy val outputBaseFeatureNames: Seq[Seq[Continuous]] = Seq( + PostNuxAlgorithmFeatureGroup.Aggregate7DayFeatures, + PostNuxAlgorithmFeatureGroup.Aggregate30DayFeatures + ) + + // We use backend impression to calculate ratio values. + lazy val ratioDenominators: Seq[Continuous] = Seq( + applyFeatureStorePrefix(PostNuxAlgorithmFeatureGroup.BackendImpressions7Days), + applyFeatureStorePrefix(PostNuxAlgorithmFeatureGroup.BackendImpressions30Days) + ) + + /** + * A mapping from an original feature's ID to the corresponding set of transformed features. + * This is used to compute the transformed features for each of the original ones. + */ + private lazy val TransformedFeaturesMap: Map[Continuous, TransformedAlgorithmFeatures] = + outputBaseFeatureNames.flatten.map { feature => + ( + // The input feature would have the FeatureStore prefix attached to it. + new Continuous(s"$FeatureStorePrefix${feature.getFeatureName}"), + // We don't keep the FeatureStore prefix to keep the length of feature names reasonable. + TransformedAlgorithmFeatures( + new Continuous(s"${feature.getFeatureName}-ratio-log") + )) + }.toMap + + /** + * Given a denominator, number of impressions, this function returns another function that adds + * transformed features (log1p and ratio) of an input feature to a DataRecord. + */ + private def addTransformedFeaturesToDataRecordFunc( + originalDr: DataRecord, + numImpressions: Double, + ): (DataRecord, Continuous) => DataRecord = { (record: DataRecord, feature: Continuous) => + { + Option(originalDr.getFeatureValue(feature)) foreach { featureValue => + TransformedFeaturesMap.get(feature).foreach { transformedFeatures => + record.setFeatureValue( + transformedFeatures.ratioLog, + // We don't use log1p here since the values are ratios and adding 1 to the _ratio_ would + // lead to logarithm of values between 1 and 2, essentially making all values the same. + math.log((featureValue + 1) / numImpressions) + ) + } + } + record + } + } + + /** + * @param record: The input record whose PostNuxAlgorithm aggregates are to be transformed. + * @return the input [[DataRecord]] with transformed aggregates added. + */ + override def adaptToDataRecord(record: DataRecord): DataRecord = { + if (record.continuousFeatures == null) { + // There are no base features available, and hence no transformations. + record + } else { + + /** + * The `foldLeft` below goes through pairs of (1) Feature groups, such as those calculated over + * 7 days or 30 days, and (2) the number of impressions for each of these groups, which is the + * denominator when ratio is calculated. + */ + ratioDenominators + .zip(allInputFeatures).foldLeft( /* initial empty DataRecord */ record)( + ( + /* DataRecord with transformed features up to here */ transformedRecord, + /* A tuple with the denominator (#impressions) and features to be transformed */ numImpressionsAndFeatures + ) => { + val (numImpressionsFeature, features) = numImpressionsAndFeatures + Option(record.getFeatureValue(numImpressionsFeature)) match { + case Some(numImpressions) if numImpressions > 0.0 => + /** + * With the number of impressions fixed, we generate a function that adds log-ratio + * for each feature in the current [[DataRecord]]. The `foldLeft` goes through all + * such features and applies that function while updating the kept DataRecord. + */ + features.foldLeft(transformedRecord)( + addTransformedFeaturesToDataRecordFunc(record, numImpressions)) + case _ => + transformedRecord + } + }) + } + } + + def getFeatures: Seq[Feature[_]] = TransformedFeaturesMap.values.flatMap(_.getFeatures).toSeq + + override def getFeatureContext: FeatureContext = + new FeatureContext() + .addFeatures(this.getFeatures.asJava) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/adapters/PreFetchedFeatureAdapter.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/adapters/PreFetchedFeatureAdapter.scala new file mode 100644 index 0000000000..f24ed1efec --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/adapters/PreFetchedFeatureAdapter.scala @@ -0,0 +1,91 @@ +package com.twitter.follow_recommendations.common.feature_hydration.adapters + +import com.twitter.follow_recommendations.common.feature_hydration.common.HasPreFetchedFeature +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.ml.api.Feature.Continuous +import com.twitter.ml.api.util.FDsl._ +import com.twitter.ml.api.DataRecord +import com.twitter.ml.api.FeatureContext +import com.twitter.ml.api.IRecordOneToOneAdapter +import com.twitter.util.Time + +/** + * This adapter mimics UserRecentWTFImpressionsAndFollowsAdapter (for user) and + * RecentWTFImpressionsFeatureAdapter (for candidate) for extracting recent impression + * and follow features. This adapter extracts user, candidate, and pair-wise features. + */ +object PreFetchedFeatureAdapter + extends IRecordOneToOneAdapter[ + (HasPreFetchedFeature, CandidateUser) + ] { + + // impression features + val USER_NUM_RECENT_IMPRESSIONS: Continuous = new Continuous( + "user.prefetch.num_recent_impressions" + ) + val USER_LAST_IMPRESSION_DURATION: Continuous = new Continuous( + "user.prefetch.last_impression_duration" + ) + val CANDIDATE_NUM_RECENT_IMPRESSIONS: Continuous = new Continuous( + "user-candidate.prefetch.num_recent_impressions" + ) + val CANDIDATE_LAST_IMPRESSION_DURATION: Continuous = new Continuous( + "user-candidate.prefetch.last_impression_duration" + ) + // follow features + val USER_NUM_RECENT_FOLLOWERS: Continuous = new Continuous( + "user.prefetch.num_recent_followers" + ) + val USER_NUM_RECENT_FOLLOWED_BY: Continuous = new Continuous( + "user.prefetch.num_recent_followed_by" + ) + val USER_NUM_RECENT_MUTUAL_FOLLOWS: Continuous = new Continuous( + "user.prefetch.num_recent_mutual_follows" + ) + // impression + follow features + val USER_NUM_RECENT_FOLLOWED_IMPRESSIONS: Continuous = new Continuous( + "user.prefetch.num_recent_followed_impression" + ) + val USER_LAST_FOLLOWED_IMPRESSION_DURATION: Continuous = new Continuous( + "user.prefetch.last_followed_impression_duration" + ) + + override def adaptToDataRecord( + record: (HasPreFetchedFeature, CandidateUser) + ): DataRecord = { + val (target, candidate) = record + val dr = new DataRecord() + val t = Time.now + // set impression features for user, optionally for candidate + dr.setFeatureValue(USER_NUM_RECENT_IMPRESSIONS, target.numWtfImpressions.toDouble) + dr.setFeatureValue( + USER_LAST_IMPRESSION_DURATION, + (t - target.latestImpressionTime).inMillis.toDouble) + target.getCandidateImpressionCounts(candidate.id).foreach { counts => + dr.setFeatureValue(CANDIDATE_NUM_RECENT_IMPRESSIONS, counts.toDouble) + } + target.getCandidateLatestTime(candidate.id).foreach { latestTime: Time => + dr.setFeatureValue(CANDIDATE_LAST_IMPRESSION_DURATION, (t - latestTime).inMillis.toDouble) + } + // set recent follow features for user + dr.setFeatureValue(USER_NUM_RECENT_FOLLOWERS, target.numRecentFollowedUserIds.toDouble) + dr.setFeatureValue(USER_NUM_RECENT_FOLLOWED_BY, target.numRecentFollowedByUserIds.toDouble) + dr.setFeatureValue(USER_NUM_RECENT_MUTUAL_FOLLOWS, target.numRecentMutualFollows.toDouble) + dr.setFeatureValue(USER_NUM_RECENT_FOLLOWED_IMPRESSIONS, target.numFollowedImpressions.toDouble) + dr.setFeatureValue( + USER_LAST_FOLLOWED_IMPRESSION_DURATION, + target.lastFollowedImpressionDurationMs.getOrElse(Long.MaxValue).toDouble) + dr + } + override def getFeatureContext: FeatureContext = new FeatureContext( + USER_NUM_RECENT_IMPRESSIONS, + USER_LAST_IMPRESSION_DURATION, + CANDIDATE_NUM_RECENT_IMPRESSIONS, + CANDIDATE_LAST_IMPRESSION_DURATION, + USER_NUM_RECENT_FOLLOWERS, + USER_NUM_RECENT_FOLLOWED_BY, + USER_NUM_RECENT_MUTUAL_FOLLOWS, + USER_NUM_RECENT_FOLLOWED_IMPRESSIONS, + USER_LAST_FOLLOWED_IMPRESSION_DURATION, + ) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/common/BUILD b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/common/BUILD new file mode 100644 index 0000000000..93ddb11914 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/common/BUILD @@ -0,0 +1,18 @@ +scala_library( + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/com/google/inject:guice", + "3rdparty/jvm/com/google/inject/extensions:guice-assistedinject", + "3rdparty/jvm/net/codingwell:scala-guice", + "3rdparty/jvm/org/slf4j:slf4j-api", + "finatra/inject/inject-core/src/main/scala", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/constants", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/utils", + "src/java/com/twitter/ml/api:api-base", + "util/util-slf4j-api/src/main/scala", + ], +) diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/common/FeatureSource.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/common/FeatureSource.scala new file mode 100644 index 0000000000..9d43f0c13d --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/common/FeatureSource.scala @@ -0,0 +1,23 @@ +package com.twitter.follow_recommendations.common.feature_hydration.common + +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.follow_recommendations.common.models.HasDisplayLocation +import com.twitter.follow_recommendations.common.models.HasSimilarToContext +import com.twitter.ml.api.DataRecord +import com.twitter.ml.api.FeatureContext +import com.twitter.product_mixer.core.model.marshalling.request.HasClientContext +import com.twitter.stitch.Stitch +import com.twitter.timelines.configapi.HasParams + +trait FeatureSource { + def id: FeatureSourceId + def featureContext: FeatureContext + def hydrateFeatures( + target: HasClientContext + with HasPreFetchedFeature + with HasParams + with HasSimilarToContext + with HasDisplayLocation, + candidates: Seq[CandidateUser] + ): Stitch[Map[CandidateUser, DataRecord]] +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/common/FeatureSourceId.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/common/FeatureSourceId.scala new file mode 100644 index 0000000000..66b8831200 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/common/FeatureSourceId.scala @@ -0,0 +1,19 @@ +package com.twitter.follow_recommendations.common.feature_hydration.common + +sealed trait FeatureSourceId + +object FeatureSourceId { + object CandidateAlgorithmSourceId extends FeatureSourceId + object ClientContextSourceId extends FeatureSourceId + object FeatureStoreSourceId extends FeatureSourceId + object FeatureStoreTimelinesAuthorSourceId extends FeatureSourceId + object FeatureStoreGizmoduckSourceId extends FeatureSourceId + object FeatureStoreUserMetricCountsSourceId extends FeatureSourceId + object FeatureStoreNotificationSourceId extends FeatureSourceId + + object FeatureStorePrecomputedNotificationSourceId extends FeatureSourceId + object FeatureStorePostNuxAlgorithmSourceId extends FeatureSourceId + @deprecated object StratoFeatureHydrationSourceId extends FeatureSourceId + object PreFetchedFeatureSourceId extends FeatureSourceId + object UserScoringFeatureSourceId extends FeatureSourceId +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/common/HasPreFetchedFeature.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/common/HasPreFetchedFeature.scala new file mode 100644 index 0000000000..8f4bae8879 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/common/HasPreFetchedFeature.scala @@ -0,0 +1,25 @@ +package com.twitter.follow_recommendations.common.feature_hydration.common + +import com.twitter.follow_recommendations.common.models.HasMutualFollowedUserIds +import com.twitter.follow_recommendations.common.models.HasWtfImpressions +import com.twitter.follow_recommendations.common.models.WtfImpression +import com.twitter.util.Time + +trait HasPreFetchedFeature extends HasMutualFollowedUserIds with HasWtfImpressions { + + lazy val followedImpressions: Seq[WtfImpression] = { + for { + wtfImprList <- wtfImpressions.toSeq + wtfImpr <- wtfImprList + if recentFollowedUserIds.exists(_.contains(wtfImpr.candidateId)) + } yield wtfImpr + } + + lazy val numFollowedImpressions: Int = followedImpressions.size + + lazy val lastFollowedImpressionDurationMs: Option[Long] = { + if (followedImpressions.nonEmpty) { + Some((Time.now - followedImpressions.map(_.latestTime).max).inMillis) + } else None + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/sources/BUILD b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/sources/BUILD new file mode 100644 index 0000000000..c0538240f2 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/sources/BUILD @@ -0,0 +1,59 @@ +scala_library( + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/com/google/inject:guice", + "3rdparty/jvm/com/google/inject/extensions:guice-assistedinject", + "3rdparty/jvm/net/codingwell:scala-guice", + "3rdparty/jvm/org/slf4j:slf4j-api", + "escherbird/src/scala/com/twitter/escherbird/util/stitchcache", + "finatra/inject/inject-core/src/main/scala", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/strato", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/constants", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/adapters", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/common", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/utils", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/common", + "hermit/hermit-core/src/main/scala/com/twitter/hermit/constants", + "src/java/com/twitter/ml/api:api-base", + "src/scala/com/twitter/ml/featurestore/catalog/datasets/core:socialgraph", + "src/scala/com/twitter/ml/featurestore/catalog/datasets/core:usersource", + "src/scala/com/twitter/ml/featurestore/catalog/datasets/onboarding:mc-user-counting", + "src/scala/com/twitter/ml/featurestore/catalog/datasets/onboarding:user-wtf-algorithm-aggregate", + "src/scala/com/twitter/ml/featurestore/catalog/datasets/onboarding:wtf-impression", + "src/scala/com/twitter/ml/featurestore/catalog/datasets/onboarding:wtf-post-nux", + "src/scala/com/twitter/ml/featurestore/catalog/datasets/onboarding:wtf-user-algorithm-aggregate", + "src/scala/com/twitter/ml/featurestore/catalog/datasets/timelines:timelines-author-features", + "src/scala/com/twitter/ml/featurestore/catalog/entities/core", + "src/scala/com/twitter/ml/featurestore/catalog/entities/onboarding", + "src/scala/com/twitter/ml/featurestore/catalog/features/core:socialgraph", + "src/scala/com/twitter/ml/featurestore/catalog/features/core:user", + "src/scala/com/twitter/ml/featurestore/catalog/features/interests_discovery:user-topic-relationships", + "src/scala/com/twitter/ml/featurestore/catalog/features/magicrecs:non-mr-notif-summmaries", + "src/scala/com/twitter/ml/featurestore/catalog/features/magicrecs:non-mr-notif-summmary-aggregates", + "src/scala/com/twitter/ml/featurestore/catalog/features/magicrecs:nonmr-ntab-summaries", + "src/scala/com/twitter/ml/featurestore/catalog/features/onboarding:mc-user-counting", + "src/scala/com/twitter/ml/featurestore/catalog/features/onboarding:post-nux-offline", + "src/scala/com/twitter/ml/featurestore/catalog/features/onboarding:post-nux-offline-edge", + "src/scala/com/twitter/ml/featurestore/catalog/features/onboarding:ratio", + "src/scala/com/twitter/ml/featurestore/catalog/features/onboarding:simcluster-user-interested-in-candidate-known-for", + "src/scala/com/twitter/ml/featurestore/catalog/features/onboarding:user-wtf-algorithm-aggregate", + "src/scala/com/twitter/ml/featurestore/catalog/features/onboarding:wtf-impression", + "src/scala/com/twitter/ml/featurestore/catalog/features/onboarding:wtf-user-algorithm-aggregate", + "src/scala/com/twitter/ml/featurestore/catalog/features/rux:user-resurrection", + "src/scala/com/twitter/ml/featurestore/catalog/features/timelines:aggregate", + "src/scala/com/twitter/ml/featurestore/lib", + "src/scala/com/twitter/ml/featurestore/lib/dynamic", + "src/scala/com/twitter/ml/featurestore/lib/embedding", + "src/scala/com/twitter/ml/featurestore/lib/feature", + "src/scala/com/twitter/ml/featurestore/lib/online", + "src/scala/com/twitter/ml/featurestore/lib/params", + "src/scala/com/twitter/onboarding/relevance/adapters/features/featurestore", + "strato/config/columns/ml/featureStore:featureStore-strato-client", + "strato/config/columns/ml/featureStore/onboarding:onboarding-strato-client", + "util/util-slf4j-api/src/main/scala", + ], +) diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/sources/CandidateAlgorithmSource.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/sources/CandidateAlgorithmSource.scala new file mode 100644 index 0000000000..0838fb98d6 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/sources/CandidateAlgorithmSource.scala @@ -0,0 +1,73 @@ +package com.twitter.follow_recommendations.common.feature_hydration.sources + +import com.google.inject.Inject +import com.google.inject.Provides +import com.google.inject.Singleton +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.follow_recommendations.common.feature_hydration.adapters.CandidateAlgorithmAdapter +import com.twitter.follow_recommendations.common.feature_hydration.common.FeatureSource +import com.twitter.follow_recommendations.common.feature_hydration.common.FeatureSourceId +import com.twitter.follow_recommendations.common.feature_hydration.common.HasPreFetchedFeature +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.follow_recommendations.common.models.HasDisplayLocation +import com.twitter.follow_recommendations.common.models.HasSimilarToContext +import com.twitter.ml.api.DataRecord +import com.twitter.ml.api.FeatureContext +import com.twitter.product_mixer.core.model.marshalling.request.HasClientContext +import com.twitter.stitch.Stitch +import com.twitter.timelines.configapi.HasParams + +/** + * This source only takes features from the candidate's source, + * which is all the information we have about the candidate pre-feature-hydration + */ + +@Provides +@Singleton +class CandidateAlgorithmSource @Inject() (stats: StatsReceiver) extends FeatureSource { + + override val id: FeatureSourceId = FeatureSourceId.CandidateAlgorithmSourceId + + override val featureContext: FeatureContext = CandidateAlgorithmAdapter.getFeatureContext + + override def hydrateFeatures( + t: HasClientContext + with HasPreFetchedFeature + with HasParams + with HasSimilarToContext + with HasDisplayLocation, // we don't use the target here + candidates: Seq[CandidateUser] + ): Stitch[Map[CandidateUser, DataRecord]] = { + val featureHydrationStats = stats.scope("candidate_alg_source") + val hasSourceDetailsStat = featureHydrationStats.counter("has_source_details") + val noSourceDetailsStat = featureHydrationStats.counter("no_source_details") + val noSourceRankStat = featureHydrationStats.counter("no_source_rank") + val hasSourceRankStat = featureHydrationStats.counter("has_source_rank") + val noSourceScoreStat = featureHydrationStats.counter("no_source_score") + val hasSourceScoreStat = featureHydrationStats.counter("has_source_score") + + val candidatesToAlgoMap = for { + candidate <- candidates + } yield { + if (candidate.userCandidateSourceDetails.nonEmpty) { + hasSourceDetailsStat.incr() + candidate.userCandidateSourceDetails.foreach { details => + if (details.candidateSourceRanks.isEmpty) { + noSourceRankStat.incr() + } else { + hasSourceRankStat.incr() + } + if (details.candidateSourceScores.isEmpty) { + noSourceScoreStat.incr() + } else { + hasSourceScoreStat.incr() + } + } + } else { + noSourceDetailsStat.incr() + } + candidate -> CandidateAlgorithmAdapter.adaptToDataRecord(candidate.userCandidateSourceDetails) + } + Stitch.value(candidatesToAlgoMap.toMap) + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/sources/ClientContextSource.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/sources/ClientContextSource.scala new file mode 100644 index 0000000000..718502d44d --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/sources/ClientContextSource.scala @@ -0,0 +1,43 @@ +package com.twitter.follow_recommendations.common.feature_hydration.sources + +import com.google.inject.Provides +import com.google.inject.Singleton +import com.twitter.follow_recommendations.common.feature_hydration.adapters.ClientContextAdapter +import com.twitter.follow_recommendations.common.feature_hydration.common.FeatureSource +import com.twitter.follow_recommendations.common.feature_hydration.common.FeatureSourceId +import com.twitter.follow_recommendations.common.feature_hydration.common.HasPreFetchedFeature +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.follow_recommendations.common.models.HasDisplayLocation +import com.twitter.follow_recommendations.common.models.HasSimilarToContext +import com.twitter.ml.api.DataRecord +import com.twitter.ml.api.FeatureContext +import com.twitter.product_mixer.core.model.marshalling.request.HasClientContext +import com.twitter.stitch.Stitch +import com.twitter.timelines.configapi.HasParams + +/** + * This source only takes features from the request (e.g. client context, WTF display location) + * No external calls are made. + */ +@Provides +@Singleton +class ClientContextSource() extends FeatureSource { + + override val id: FeatureSourceId = FeatureSourceId.ClientContextSourceId + + override val featureContext: FeatureContext = ClientContextAdapter.getFeatureContext + + override def hydrateFeatures( + t: HasClientContext + with HasPreFetchedFeature + with HasParams + with HasSimilarToContext + with HasDisplayLocation, + candidates: Seq[CandidateUser] + ): Stitch[Map[CandidateUser, DataRecord]] = { + Stitch.value( + candidates + .map(_ -> ((t.clientContext, t.displayLocation))).toMap.mapValues( + ClientContextAdapter.adaptToDataRecord)) + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/sources/FeatureHydrationSourcesFSConfig.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/sources/FeatureHydrationSourcesFSConfig.scala new file mode 100644 index 0000000000..f78ec17cc8 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/sources/FeatureHydrationSourcesFSConfig.scala @@ -0,0 +1,42 @@ +package com.twitter.follow_recommendations.common.feature_hydration.sources + +import com.twitter.follow_recommendations.configapi.common.FeatureSwitchConfig +import com.twitter.timelines.configapi.FSBoundedParam +import com.twitter.timelines.configapi.FSName +import com.twitter.timelines.configapi.HasDurationConversion +import com.twitter.timelines.configapi.Param +import com.twitter.util.Duration +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class FeatureHydrationSourcesFSConfig @Inject() () extends FeatureSwitchConfig { + override val booleanFSParams: Seq[Param[Boolean] with FSName] = Seq( + FeatureStoreSourceParams.EnableAlgorithmAggregateFeatures, + FeatureStoreSourceParams.EnableAuthorTopicAggregateFeatures, + FeatureStoreSourceParams.EnableCandidateClientFeatures, + FeatureStoreSourceParams.EnableCandidatePrecomputedNotificationFeatures, + FeatureStoreSourceParams.EnableCandidateUserAuthorRealTimeAggregateFeatures, + FeatureStoreSourceParams.EnableCandidateUserFeatures, + FeatureStoreSourceParams.EnableCandidateUserResurrectionFeatures, + FeatureStoreSourceParams.EnableCandidateUserTimelinesAuthorAggregateFeatures, + FeatureStoreSourceParams.EnableSeparateClientForTimelinesAuthors, + FeatureStoreSourceParams.EnableSeparateClientForGizmoduck, + FeatureStoreSourceParams.EnableSeparateClientForMetricCenterUserCounting, + FeatureStoreSourceParams.EnableSeparateClientForNotifications, + FeatureStoreSourceParams.EnableSimilarToUserFeatures, + FeatureStoreSourceParams.EnableTargetUserFeatures, + FeatureStoreSourceParams.EnableTargetUserResurrectionFeatures, + FeatureStoreSourceParams.EnableTargetUserWtfImpressionFeatures, + FeatureStoreSourceParams.EnableTopicAggregateFeatures, + FeatureStoreSourceParams.EnableUserCandidateEdgeFeatures, + FeatureStoreSourceParams.EnableUserCandidateWtfImpressionCandidateFeatures, + FeatureStoreSourceParams.EnableUserClientFeatures, + FeatureStoreSourceParams.EnableUserTopicFeatures, + FeatureStoreSourceParams.EnableUserWtfAlgEdgeFeatures, + ) + + override val durationFSParams: Seq[FSBoundedParam[Duration] with HasDurationConversion] = Seq( + FeatureStoreSourceParams.GlobalFetchTimeout + ) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/sources/FeatureHydrationSourcesFeatureSwitchKeys.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/sources/FeatureHydrationSourcesFeatureSwitchKeys.scala new file mode 100644 index 0000000000..fb22329276 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/sources/FeatureHydrationSourcesFeatureSwitchKeys.scala @@ -0,0 +1,42 @@ +package com.twitter.follow_recommendations.common.feature_hydration.sources + +object FeatureHydrationSourcesFeatureSwitchKeys { + val EnableAlgorithmAggregateFeatures = "feature_store_source_enable_algorithm_aggregate_features" + val EnableAuthorTopicAggregateFeatures = + "feature_store_source_enable_author_topic_aggregate_features" + val EnableCandidateClientFeatures = "feature_store_source_enable_candidate_client_features" + val EnableCandidateNotificationFeatures = + "feature_store_source_enable_candidate_notification_features" + val EnableCandidatePrecomputedNotificationFeatures = + "feature_store_source_enable_candidate_precomputed_notification_features" + val EnableCandidateUserFeatures = "feature_store_source_enable_candidate_user_features" + val EnableCandidateUserAuthorRealTimeAggregateFeatures = + "feature_store_source_enable_candidate_user_author_rta_features" + val EnableCandidateUserResurrectionFeatures = + "feature_store_source_enable_candidate_user_resurrection_features" + val EnableCandidateUserTimelinesAuthorAggregateFeatures = + "feature_store_source_enable_candidate_user_timelines_author_aggregate_features" + val EnableSimilarToUserFeatures = "feature_store_source_enable_similar_to_user_features" + val EnableTargetUserFeatures = "feature_store_source_enable_target_user_features" + val EnableTargetUserUserAuthorUserStateRealTimeAggregatesFeature = + "feature_store_source_enable_target_user_user_author_user_state_rta_features" + val EnableTargetUserResurrectionFeatures = + "feature_store_source_enable_target_user_resurrection_features" + val EnableTargetUserWtfImpressionFeatures = + "feature_store_source_enable_target_user_wtf_impression_features" + val EnableTopicAggregateFeatures = "feature_store_source_enable_topic_aggregate_features" + val EnableUserCandidateEdgeFeatures = "feature_store_source_enable_user_candidate_edge_features" + val EnableUserCandidateWtfImpressionCandidateFeatures = + "feature_store_source_enable_user_candidate_wtf_impression_features" + val EnableUserClientFeatures = "feature_store_source_enable_user_client_features" + val EnableUserNotificationFeatures = "feature_store_source_enable_user_notification_features" + val EnableUserTopicFeatures = "feature_store_source_enable_user_topic_features" + val EnableUserWtfAlgEdgeFeatures = "feature_store_source_enable_user_wtf_alg_edge_features" + val FeatureHydrationTimeout = "feature_store_source_hydration_timeout_in_millis" + val UseSeparateClientForTimelinesAuthor = + "feature_store_source_separate_client_for_timelines_author_data" + val UseSeparateClientMetricCenterUserCounting = + "feature_store_source_separate_client_for_mc_user_counting_data" + val UseSeparateClientForNotifications = "feature_store_source_separate_client_for_notifications" + val UseSeparateClientForGizmoduck = "feature_store_source_separate_client_for_gizmoduck" +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/sources/FeatureStoreFeatures.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/sources/FeatureStoreFeatures.scala new file mode 100644 index 0000000000..6c849f7a78 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/sources/FeatureStoreFeatures.scala @@ -0,0 +1,342 @@ +package com.twitter.follow_recommendations.common.feature_hydration.sources + +import com.twitter.ml.api.DataRecord +import com.twitter.ml.api.FeatureContext +import com.twitter.ml.featurestore.catalog.entities.core.{Author => AuthorEntity} +import com.twitter.ml.featurestore.catalog.entities.core.{AuthorTopic => AuthorTopicEntity} +import com.twitter.ml.featurestore.catalog.entities.core.{CandidateUser => CandidateUserEntity} +import com.twitter.ml.featurestore.catalog.entities.core.{Topic => TopicEntity} +import com.twitter.ml.featurestore.catalog.entities.core.{User => UserEntity} +import com.twitter.ml.featurestore.catalog.entities.core.{UserCandidate => UserCandidateEntity} +import com.twitter.ml.featurestore.catalog.entities.onboarding.UserWtfAlgorithmEntity +import com.twitter.ml.featurestore.catalog.entities.onboarding.{ + WtfAlgorithm => WtfAlgorithmIdEntity +} +import com.twitter.ml.featurestore.catalog.entities.onboarding.{ + WtfAlgorithmType => WtfAlgorithmTypeEntity +} +import com.twitter.ml.featurestore.catalog.features.core.UserClients.FullPrimaryClientVersion +import com.twitter.ml.featurestore.catalog.features.core.UserClients.NumClients +import com.twitter.ml.featurestore.catalog.features.core.UserClients.PrimaryClient +import com.twitter.ml.featurestore.catalog.features.core.UserClients.PrimaryClientVersion +import com.twitter.ml.featurestore.catalog.features.core.UserClients.PrimaryDeviceManufacturer +import com.twitter.ml.featurestore.catalog.features.core.UserClients.PrimaryMobileSdkVersion +import com.twitter.ml.featurestore.catalog.features.core.UserClients.SecondaryClient +import com.twitter.ml.featurestore.catalog.features.core.UserCounts.Favorites +import com.twitter.ml.featurestore.catalog.features.core.UserCounts.Followers +import com.twitter.ml.featurestore.catalog.features.core.UserCounts.Following +import com.twitter.ml.featurestore.catalog.features.core.UserCounts.Tweets +import com.twitter.ml.featurestore.catalog.features.customer_journey.PostNuxAlgorithmIdAggregateFeatureGroup +import com.twitter.ml.featurestore.catalog.features.customer_journey.PostNuxAlgorithmTypeAggregateFeatureGroup +import com.twitter.ml.featurestore.catalog.features.customer_journey.{Utils => FeatureGroupUtils} +import com.twitter.ml.featurestore.catalog.features.interests_discovery.UserTopicRelationships.FollowedTopics +import com.twitter.ml.featurestore.catalog.features.onboarding.MetricCenterUserCounts.NumFavorites +import com.twitter.ml.featurestore.catalog.features.onboarding.MetricCenterUserCounts.NumFavoritesReceived +import com.twitter.ml.featurestore.catalog.features.onboarding.MetricCenterUserCounts.NumFollowBacks +import com.twitter.ml.featurestore.catalog.features.onboarding.MetricCenterUserCounts.NumFollows +import com.twitter.ml.featurestore.catalog.features.onboarding.MetricCenterUserCounts.NumFollowsReceived +import com.twitter.ml.featurestore.catalog.features.onboarding.MetricCenterUserCounts.NumLoginDays +import com.twitter.ml.featurestore.catalog.features.onboarding.MetricCenterUserCounts.NumLoginTweetImpressions +import com.twitter.ml.featurestore.catalog.features.onboarding.MetricCenterUserCounts.NumMuteBacks +import com.twitter.ml.featurestore.catalog.features.onboarding.MetricCenterUserCounts.NumMuted +import com.twitter.ml.featurestore.catalog.features.onboarding.MetricCenterUserCounts.NumOriginalTweets +import com.twitter.ml.featurestore.catalog.features.onboarding.MetricCenterUserCounts.NumQualityFollowReceived +import com.twitter.ml.featurestore.catalog.features.onboarding.MetricCenterUserCounts.NumQuoteRetweets +import com.twitter.ml.featurestore.catalog.features.onboarding.MetricCenterUserCounts.NumQuoteRetweetsReceived +import com.twitter.ml.featurestore.catalog.features.onboarding.MetricCenterUserCounts.NumReplies +import com.twitter.ml.featurestore.catalog.features.onboarding.MetricCenterUserCounts.NumRepliesReceived +import com.twitter.ml.featurestore.catalog.features.onboarding.MetricCenterUserCounts.NumRetweets +import com.twitter.ml.featurestore.catalog.features.onboarding.MetricCenterUserCounts.NumRetweetsReceived +import com.twitter.ml.featurestore.catalog.features.onboarding.MetricCenterUserCounts.NumSpamBlocked +import com.twitter.ml.featurestore.catalog.features.onboarding.MetricCenterUserCounts.NumSpamBlockedBacks +import com.twitter.ml.featurestore.catalog.features.onboarding.MetricCenterUserCounts.NumTweetImpressions +import com.twitter.ml.featurestore.catalog.features.onboarding.MetricCenterUserCounts.NumTweets +import com.twitter.ml.featurestore.catalog.features.onboarding.MetricCenterUserCounts.NumUnfollowBacks +import com.twitter.ml.featurestore.catalog.features.onboarding.MetricCenterUserCounts.NumUnfollows +import com.twitter.ml.featurestore.catalog.features.onboarding.MetricCenterUserCounts.NumUserActiveMinutes +import com.twitter.ml.featurestore.catalog.features.onboarding.MetricCenterUserCounts.NumWasMutualFollowed +import com.twitter.ml.featurestore.catalog.features.onboarding.MetricCenterUserCounts.NumWasMutualUnfollowed +import com.twitter.ml.featurestore.catalog.features.onboarding.MetricCenterUserCounts.NumWasUnfollowed +import com.twitter.ml.featurestore.catalog.features.onboarding.PostNuxOffline.Country +import com.twitter.ml.featurestore.catalog.features.onboarding.PostNuxOffline.FollowersOverFollowingRatio +import com.twitter.ml.featurestore.catalog.features.onboarding.PostNuxOffline.Language +import com.twitter.ml.featurestore.catalog.features.onboarding.PostNuxOffline.MutualFollowsOverFollowersRatio +import com.twitter.ml.featurestore.catalog.features.onboarding.PostNuxOffline.MutualFollowsOverFollowingRatio +import com.twitter.ml.featurestore.catalog.features.onboarding.PostNuxOffline.NumFollowers +import com.twitter.ml.featurestore.catalog.features.onboarding.PostNuxOffline.NumFollowings +import com.twitter.ml.featurestore.catalog.features.onboarding.PostNuxOffline.NumMutualFollows +import com.twitter.ml.featurestore.catalog.features.onboarding.PostNuxOffline.TweepCred +import com.twitter.ml.featurestore.catalog.features.onboarding.PostNuxOffline.UserState +import com.twitter.ml.featurestore.catalog.features.onboarding.PostNuxOfflineEdge.HaveSameCountry +import com.twitter.ml.featurestore.catalog.features.onboarding.PostNuxOfflineEdge.HaveSameLanguage +import com.twitter.ml.featurestore.catalog.features.onboarding.PostNuxOfflineEdge.HaveSameUserState +import com.twitter.ml.featurestore.catalog.features.onboarding.PostNuxOfflineEdge.NumFollowersGap +import com.twitter.ml.featurestore.catalog.features.onboarding.PostNuxOfflineEdge.NumFollowingsGap +import com.twitter.ml.featurestore.catalog.features.onboarding.PostNuxOfflineEdge.NumMutualFollowsGap +import com.twitter.ml.featurestore.catalog.features.onboarding.PostNuxOfflineEdge.TweepCredGap +import com.twitter.ml.featurestore.catalog.features.onboarding.Ratio.FollowersFollowings +import com.twitter.ml.featurestore.catalog.features.onboarding.Ratio.MutualFollowsFollowing +import com.twitter.ml.featurestore.catalog.features.onboarding.SimclusterUserInterestedInCandidateKnownFor.HasIntersection +import com.twitter.ml.featurestore.catalog.features.onboarding.SimclusterUserInterestedInCandidateKnownFor.IntersectionCandidateKnownForScore +import com.twitter.ml.featurestore.catalog.features.onboarding.SimclusterUserInterestedInCandidateKnownFor.IntersectionClusterIds +import com.twitter.ml.featurestore.catalog.features.onboarding.SimclusterUserInterestedInCandidateKnownFor.IntersectionUserFavCandidateKnownForScore +import com.twitter.ml.featurestore.catalog.features.onboarding.SimclusterUserInterestedInCandidateKnownFor.IntersectionUserFavScore +import com.twitter.ml.featurestore.catalog.features.onboarding.SimclusterUserInterestedInCandidateKnownFor.IntersectionUserFollowCandidateKnownForScore +import com.twitter.ml.featurestore.catalog.features.onboarding.SimclusterUserInterestedInCandidateKnownFor.IntersectionUserFollowScore +import com.twitter.ml.featurestore.catalog.features.onboarding.UserWtfAlgorithmAggregate +import com.twitter.ml.featurestore.catalog.features.onboarding.WhoToFollowImpression.HomeTimelineWtfCandidateCounts +import com.twitter.ml.featurestore.catalog.features.onboarding.WhoToFollowImpression.HomeTimelineWtfCandidateImpressionCounts +import com.twitter.ml.featurestore.catalog.features.onboarding.WhoToFollowImpression.HomeTimelineWtfCandidateImpressionLatestTimestamp +import com.twitter.ml.featurestore.catalog.features.onboarding.WhoToFollowImpression.HomeTimelineWtfLatestTimestamp +import com.twitter.ml.featurestore.catalog.features.onboarding.WtfUserAlgorithmAggregate.FollowRate +import com.twitter.ml.featurestore.catalog.features.onboarding.WtfUserAlgorithmAggregate.Follows +import com.twitter.ml.featurestore.catalog.features.onboarding.WtfUserAlgorithmAggregate.FollowsTweetFavRate +import com.twitter.ml.featurestore.catalog.features.onboarding.WtfUserAlgorithmAggregate.FollowsTweetReplies +import com.twitter.ml.featurestore.catalog.features.onboarding.WtfUserAlgorithmAggregate.FollowsTweetReplyRate +import com.twitter.ml.featurestore.catalog.features.onboarding.WtfUserAlgorithmAggregate.FollowsTweetRetweetRate +import com.twitter.ml.featurestore.catalog.features.onboarding.WtfUserAlgorithmAggregate.FollowsTweetRetweets +import com.twitter.ml.featurestore.catalog.features.onboarding.WtfUserAlgorithmAggregate.FollowsWithTweetFavs +import com.twitter.ml.featurestore.catalog.features.onboarding.WtfUserAlgorithmAggregate.FollowsWithTweetImpressions +import com.twitter.ml.featurestore.catalog.features.onboarding.WtfUserAlgorithmAggregate.HasAnyEngagements +import com.twitter.ml.featurestore.catalog.features.onboarding.WtfUserAlgorithmAggregate.HasForwardEngagements +import com.twitter.ml.featurestore.catalog.features.onboarding.WtfUserAlgorithmAggregate.HasReverseEngagements +import com.twitter.ml.featurestore.catalog.features.onboarding.WtfUserAlgorithmAggregate.Impressions +import com.twitter.ml.featurestore.catalog.features.rux.UserResurrection.DaysSinceRecentResurrection +import com.twitter.ml.featurestore.catalog.features.timelines.AuthorTopicAggregates +import com.twitter.ml.featurestore.catalog.features.timelines.EngagementsReceivedByAuthorRealTimeAggregates +import com.twitter.ml.featurestore.catalog.features.timelines.NegativeEngagementsReceivedByAuthorRealTimeAggregates +import com.twitter.ml.featurestore.catalog.features.timelines.OriginalAuthorAggregates +import com.twitter.ml.featurestore.catalog.features.timelines.TopicEngagementRealTimeAggregates +import com.twitter.ml.featurestore.catalog.features.timelines.TopicEngagementUserStateRealTimeAggregates +import com.twitter.ml.featurestore.catalog.features.timelines.TopicNegativeEngagementUserStateRealTimeAggregates +import com.twitter.ml.featurestore.catalog.features.timelines.UserEngagementAuthorUserStateRealTimeAggregates +import com.twitter.ml.featurestore.catalog.features.timelines.UserNegativeEngagementAuthorUserStateRealTimeAggregates +import com.twitter.ml.featurestore.lib.EntityId +import com.twitter.ml.featurestore.lib.UserId +import com.twitter.ml.featurestore.lib.feature.BoundFeature +import com.twitter.ml.featurestore.lib.feature.Feature + +object FeatureStoreFeatures { + import FeatureStoreRawFeatures._ + ///////////////////////////// Target user features //////////////////////// + val targetUserFeatures: Set[BoundFeature[_ <: EntityId, _]] = + (userKeyedFeatures ++ userAlgorithmAggregateFeatures).map(_.bind(UserEntity)) + + val targetUserResurrectionFeatures: Set[BoundFeature[_ <: EntityId, _]] = + userResurrectionFeatures.map(_.bind(UserEntity)) + val targetUserWtfImpressionFeatures: Set[BoundFeature[_ <: EntityId, _]] = + wtfImpressionUserFeatures.map(_.bind(UserEntity)) + val targetUserUserAuthorUserStateRealTimeAggregatesFeature: Set[BoundFeature[_ <: EntityId, _]] = + userAuthorUserStateRealTimeAggregatesFeature.map(_.bind(UserEntity)) + + val targetUserStatusFeatures: Set[BoundFeature[_ <: EntityId, _]] = + userStatusFeatures.map(_.bind(UserEntity).logarithm1p) + val targetUserMetricCountFeatures: Set[BoundFeature[_ <: EntityId, _]] = + mcFeatures.map(_.bind(UserEntity).logarithm1p) + + val targetUserClientFeatures: Set[BoundFeature[_ <: EntityId, _]] = + clientFeatures.map(_.bind(UserEntity)) + + ///////////////////////////// Candidate user features //////////////////////// + val candidateUserFeatures: Set[BoundFeature[_ <: EntityId, _]] = + userKeyedFeatures.map(_.bind(CandidateUserEntity)) + val candidateUserAuthorRealTimeAggregateFeatures: Set[BoundFeature[_ <: EntityId, _]] = + authorAggregateFeatures.map(_.bind(CandidateUserEntity)) + val candidateUserResurrectionFeatures: Set[BoundFeature[_ <: EntityId, _]] = + userResurrectionFeatures.map(_.bind(CandidateUserEntity)) + + val candidateUserStatusFeatures: Set[BoundFeature[_ <: EntityId, _]] = + userStatusFeatures.map(_.bind(CandidateUserEntity).logarithm1p) + val candidateUserTimelinesAuthorAggregateFeatures: Set[BoundFeature[_ <: EntityId, _]] = + Set(timelinesAuthorAggregateFeatures.bind(CandidateUserEntity)) + val candidateUserMetricCountFeatures: Set[BoundFeature[_ <: EntityId, _]] = + mcFeatures.map(_.bind(CandidateUserEntity).logarithm1p) + + val candidateUserClientFeatures: Set[BoundFeature[_ <: EntityId, _]] = + clientFeatures.map(_.bind(CandidateUserEntity)) + + val similarToUserFeatures: Set[BoundFeature[_ <: EntityId, _]] = + (userKeyedFeatures ++ authorAggregateFeatures).map(_.bind(AuthorEntity)) + + val similarToUserStatusFeatures: Set[BoundFeature[_ <: EntityId, _]] = + userStatusFeatures.map(_.bind(AuthorEntity).logarithm1p) + val similarToUserTimelinesAuthorAggregateFeatures: Set[BoundFeature[_ <: EntityId, _]] = + Set(timelinesAuthorAggregateFeatures.bind(AuthorEntity)) + val similarToUserMetricCountFeatures: Set[BoundFeature[_ <: EntityId, _]] = + mcFeatures.map(_.bind(AuthorEntity).logarithm1p) + + val userCandidateEdgeFeatures: Set[BoundFeature[_ <: EntityId, _]] = + (simclusterUVIntersectionFeatures ++ userCandidatePostNuxEdgeFeatures).map( + _.bind(UserCandidateEntity)) + val userCandidateWtfImpressionCandidateFeatures: Set[BoundFeature[_ <: EntityId, _]] = + wtfImpressionCandidateFeatures.map(_.bind(UserCandidateEntity)) + + /** + * Aggregate features based on candidate source algorithms. + */ + val postNuxAlgorithmIdAggregateFeatures: Set[BoundFeature[_ <: EntityId, _]] = + Set(PostNuxAlgorithmIdAggregateFeatureGroup.FeaturesAsDataRecord) + .map(_.bind(WtfAlgorithmIdEntity)) + + /** + * Aggregate features based on candidate source algorithm types. There are 4 at the moment: + * Geo, Social, Activity and Interest. + */ + val postNuxAlgorithmTypeAggregateFeatures: Set[BoundFeature[_ <: EntityId, _]] = + Set(PostNuxAlgorithmTypeAggregateFeatureGroup.FeaturesAsDataRecord) + .map(_.bind(WtfAlgorithmTypeEntity)) + + // user wtf-Algorithm features + val userWtfAlgorithmEdgeFeatures: Set[BoundFeature[_ <: EntityId, _]] = + FeatureGroupUtils.getTimelinesAggregationFrameworkCombinedFeatures( + UserWtfAlgorithmAggregate, + UserWtfAlgorithmEntity, + FeatureGroupUtils.getMaxSumAvgAggregate(UserWtfAlgorithmAggregate) + ) + + /** + * We have to add the max/sum/avg-aggregated features to the set of all features so that we can + * register them using FRS's [[FrsFeatureJsonExporter]]. + * + * Any additional such aggregated features that are included in [[FeatureStoreSource]] client + * should be registered here as well. + */ + val maxSumAvgAggregatedFeatureContext: FeatureContext = new FeatureContext() + .addFeatures( + UserWtfAlgorithmAggregate.getSecondaryAggregatedFeatureContext + ) + + // topic features + val topicAggregateFeatures: Set[BoundFeature[_ <: EntityId, _]] = Set( + TopicEngagementRealTimeAggregates.FeaturesAsDataRecord, + TopicNegativeEngagementUserStateRealTimeAggregates.FeaturesAsDataRecord, + TopicEngagementUserStateRealTimeAggregates.FeaturesAsDataRecord + ).map(_.bind(TopicEntity)) + val userTopicFeatures: Set[BoundFeature[_ <: EntityId, _]] = Set(FollowedTopics.bind(UserEntity)) + val authorTopicFeatures: Set[BoundFeature[_ <: EntityId, _]] = Set( + AuthorTopicAggregates.FeaturesAsDataRecord.bind(AuthorTopicEntity)) + val topicFeatures = topicAggregateFeatures ++ userTopicFeatures ++ authorTopicFeatures + +} + +object FeatureStoreRawFeatures { + val mcFeatures = Set( + NumTweets, + NumRetweets, + NumOriginalTweets, + NumRetweetsReceived, + NumFavoritesReceived, + NumRepliesReceived, + NumQuoteRetweetsReceived, + NumFollowsReceived, + NumFollowBacks, + NumFollows, + NumUnfollows, + NumUnfollowBacks, + NumQualityFollowReceived, + NumQuoteRetweets, + NumFavorites, + NumReplies, + NumLoginTweetImpressions, + NumTweetImpressions, + NumLoginDays, + NumUserActiveMinutes, + NumMuted, + NumSpamBlocked, + NumMuteBacks, + NumSpamBlockedBacks, + NumWasMutualFollowed, + NumWasMutualUnfollowed, + NumWasUnfollowed + ) + // based off usersource, and each feature represents the cumulative 'sent' counts + val userStatusFeatures = Set( + Favorites, + Followers, + Following, + Tweets + ) + // ratio features created from combining other features + val userRatioFeatures = Set(MutualFollowsFollowing, FollowersFollowings) + // features related to user login history + val userResurrectionFeatures: Set[Feature[UserId, Int]] = Set( + DaysSinceRecentResurrection + ) + + // real-time aggregate features borrowed from timelines + val authorAggregateFeatures = Set( + EngagementsReceivedByAuthorRealTimeAggregates.FeaturesAsDataRecord, + NegativeEngagementsReceivedByAuthorRealTimeAggregates.FeaturesAsDataRecord, + ) + + val timelinesAuthorAggregateFeatures = OriginalAuthorAggregates.FeaturesAsDataRecord + + val userAuthorUserStateRealTimeAggregatesFeature: Set[Feature[UserId, DataRecord]] = Set( + UserEngagementAuthorUserStateRealTimeAggregates.FeaturesAsDataRecord, + UserNegativeEngagementAuthorUserStateRealTimeAggregates.FeaturesAsDataRecord + ) + // post nux per-user offline features + val userOfflineFeatures = Set( + NumFollowings, + NumFollowers, + NumMutualFollows, + TweepCred, + UserState, + Language, + Country, + MutualFollowsOverFollowingRatio, + MutualFollowsOverFollowersRatio, + FollowersOverFollowingRatio, + ) + // matched post nux offline features between user and candidate + val userCandidatePostNuxEdgeFeatures = Set( + HaveSameUserState, + HaveSameLanguage, + HaveSameCountry, + NumFollowingsGap, + NumFollowersGap, + NumMutualFollowsGap, + TweepCredGap, + ) + // user algorithm aggregate features + val userAlgorithmAggregateFeatures = Set( + Impressions, + Follows, + FollowRate, + FollowsWithTweetImpressions, + FollowsWithTweetFavs, + FollowsTweetFavRate, + FollowsTweetReplies, + FollowsTweetReplyRate, + FollowsTweetRetweets, + FollowsTweetRetweetRate, + HasForwardEngagements, + HasReverseEngagements, + HasAnyEngagements, + ) + val userKeyedFeatures = userRatioFeatures ++ userOfflineFeatures + val wtfImpressionUserFeatures = + Set(HomeTimelineWtfCandidateCounts, HomeTimelineWtfLatestTimestamp) + val wtfImpressionCandidateFeatures = + Set(HomeTimelineWtfCandidateImpressionCounts, HomeTimelineWtfCandidateImpressionLatestTimestamp) + val simclusterUVIntersectionFeatures = Set( + IntersectionClusterIds, + HasIntersection, + IntersectionUserFollowScore, + IntersectionUserFavScore, + IntersectionCandidateKnownForScore, + IntersectionUserFollowCandidateKnownForScore, + IntersectionUserFavCandidateKnownForScore + ) + + // Client features + val clientFeatures = Set( + NumClients, + PrimaryClient, + PrimaryClientVersion, + FullPrimaryClientVersion, + PrimaryDeviceManufacturer, + PrimaryMobileSdkVersion, + SecondaryClient + ) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/sources/FeatureStoreGizmoduckSource.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/sources/FeatureStoreGizmoduckSource.scala new file mode 100644 index 0000000000..bb6b588570 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/sources/FeatureStoreGizmoduckSource.scala @@ -0,0 +1,188 @@ +package com.twitter.follow_recommendations.common.feature_hydration.sources + +import com.github.benmanes.caffeine.cache.Caffeine +import com.google.inject.Inject +import com.twitter.finagle.TimeoutException +import com.twitter.finagle.mtls.authentication.ServiceIdentifier +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.follow_recommendations.common.feature_hydration.common.FeatureSource +import com.twitter.follow_recommendations.common.feature_hydration.common.FeatureSourceId +import com.twitter.follow_recommendations.common.feature_hydration.common.HasPreFetchedFeature +import com.twitter.follow_recommendations.common.feature_hydration.sources.Utils.adaptAdditionalFeaturesToDataRecord +import com.twitter.follow_recommendations.common.feature_hydration.sources.Utils.randomizedTTL +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.follow_recommendations.common.models.HasSimilarToContext +import com.twitter.ml.api.DataRecord +import com.twitter.ml.api.FeatureContext +import com.twitter.ml.api.IRecordOneToOneAdapter +import com.twitter.ml.featurestore.catalog.datasets.core.UsersourceEntityDataset +import com.twitter.ml.featurestore.catalog.entities.core.{Author => AuthorEntity} +import com.twitter.ml.featurestore.catalog.entities.core.{AuthorTopic => AuthorTopicEntity} +import com.twitter.ml.featurestore.catalog.entities.core.{CandidateUser => CandidateUserEntity} +import com.twitter.ml.featurestore.catalog.entities.core.{User => UserEntity} +import com.twitter.ml.featurestore.lib.EdgeEntityId +import com.twitter.ml.featurestore.lib.EntityId +import com.twitter.ml.featurestore.lib.TopicId +import com.twitter.ml.featurestore.lib.UserId +import com.twitter.ml.featurestore.lib.data.PredictionRecord +import com.twitter.ml.featurestore.lib.data.PredictionRecordAdapter +import com.twitter.ml.featurestore.lib.dataset.DatasetId +import com.twitter.ml.featurestore.lib.dataset.online.Hydrator.HydrationResponse +import com.twitter.ml.featurestore.lib.dataset.online.OnlineAccessDataset +import com.twitter.ml.featurestore.lib.dynamic.ClientConfig +import com.twitter.ml.featurestore.lib.dynamic.DynamicFeatureStoreClient +import com.twitter.ml.featurestore.lib.dynamic.DynamicHydrationConfig +import com.twitter.ml.featurestore.lib.dynamic.FeatureStoreParamsConfig +import com.twitter.ml.featurestore.lib.dynamic.GatedFeatures +import com.twitter.ml.featurestore.lib.feature.BoundFeature +import com.twitter.ml.featurestore.lib.feature.BoundFeatureSet +import com.twitter.ml.featurestore.lib.online.DatasetValuesCache +import com.twitter.ml.featurestore.lib.online.FeatureStoreRequest +import com.twitter.ml.featurestore.lib.online.OnlineFeatureGenerationStats +import com.twitter.stitch.Stitch +import com.twitter.timelines.configapi.HasParams +import java.util.concurrent.TimeUnit +import com.twitter.conversions.DurationOps._ +import com.twitter.follow_recommendations.common.models.HasDisplayLocation +import com.twitter.product_mixer.core.model.marshalling.request.HasClientContext + +class FeatureStoreGizmoduckSource @Inject() ( + serviceIdentifier: ServiceIdentifier, + stats: StatsReceiver) + extends FeatureSource { + import FeatureStoreGizmoduckSource._ + + val backupSourceStats = stats.scope("feature_store_hydration_gizmoduck") + val adapterStats = backupSourceStats.scope("adapters") + override def id: FeatureSourceId = FeatureSourceId.FeatureStoreGizmoduckSourceId + override def featureContext: FeatureContext = getFeatureContext + + val clientConfig: ClientConfig[HasParams] = ClientConfig( + dynamicHydrationConfig = dynamicHydrationConfig, + featureStoreParamsConfig = + FeatureStoreParamsConfig(FeatureStoreParameters.featureStoreParams, Map.empty), + /** + * The smaller one between `timeoutProvider` and `FeatureStoreSourceParams.GlobalFetchTimeout` + * used below takes effect. + */ + timeoutProvider = Function.const(800.millis), + serviceIdentifier = serviceIdentifier + ) + + private val datasetsToCache = Set( + UsersourceEntityDataset + ).asInstanceOf[Set[OnlineAccessDataset[_ <: EntityId, _]]] + + private val datasetValuesCache: DatasetValuesCache = + DatasetValuesCache( + Caffeine + .newBuilder() + .expireAfterWrite(randomizedTTL(12.hours.inSeconds), TimeUnit.SECONDS) + .maximumSize(DefaultCacheMaxKeys) + .build[(_ <: EntityId, DatasetId), Stitch[HydrationResponse[_]]] + .asMap, + datasetsToCache, + DatasetCacheScope + ) + + private val dynamicFeatureStoreClient = DynamicFeatureStoreClient( + clientConfig, + backupSourceStats, + Set(datasetValuesCache) + ) + + private val adapter: IRecordOneToOneAdapter[PredictionRecord] = + PredictionRecordAdapter.oneToOne( + BoundFeatureSet(allFeatures), + OnlineFeatureGenerationStats(backupSourceStats) + ) + + override def hydrateFeatures( + target: HasClientContext + with HasPreFetchedFeature + with HasParams + with HasSimilarToContext + with HasDisplayLocation, + candidates: Seq[CandidateUser] + ): Stitch[Map[CandidateUser, DataRecord]] = { + target.getOptionalUserId + .map { targetUserId => + val featureRequests = candidates.map { candidate => + val userEntityId = UserEntity.withId(UserId(targetUserId)) + val candidateEntityId = CandidateUserEntity.withId(UserId(candidate.id)) + val similarToUserId = target.similarToUserIds.map(id => AuthorEntity.withId(UserId(id))) + val topicProof = candidate.reason.flatMap(_.accountProof.flatMap(_.topicProof)) + val authorTopicEntity = if (topicProof.isDefined) { + backupSourceStats.counter("candidates_with_topic_proof").incr() + Set( + AuthorTopicEntity.withId( + EdgeEntityId(UserId(candidate.id), TopicId(topicProof.get.topicId)))) + } else Nil + + val entities = + Seq(userEntityId, candidateEntityId) ++ similarToUserId ++ authorTopicEntity + FeatureStoreRequest(entities) + } + + val predictionRecordsFut = dynamicFeatureStoreClient(featureRequests, target) + val candidateFeatureMap = predictionRecordsFut.map { predictionRecords => + // we can zip predictionRecords with candidates as the order is preserved in the client + candidates + .zip(predictionRecords).map { + case (candidate, predictionRecord) => + candidate -> adaptAdditionalFeaturesToDataRecord( + adapter.adaptToDataRecord(predictionRecord), + adapterStats, + FeatureStoreSource.featureAdapters) + }.toMap + } + Stitch + .callFuture(candidateFeatureMap) + .within(target.params(FeatureStoreSourceParams.GlobalFetchTimeout))( + com.twitter.finagle.util.DefaultTimer) + .rescue { + case _: TimeoutException => + Stitch.value(Map.empty[CandidateUser, DataRecord]) + } + }.getOrElse(Stitch.value(Map.empty[CandidateUser, DataRecord])) + } +} + +object FeatureStoreGizmoduckSource { + private val DatasetCacheScope = "feature_store_local_cache_gizmoduck" + private val DefaultCacheMaxKeys = 20000 + + val allFeatures: Set[BoundFeature[_ <: EntityId, _]] = + FeatureStoreFeatures.candidateUserStatusFeatures ++ + FeatureStoreFeatures.similarToUserStatusFeatures ++ + FeatureStoreFeatures.targetUserStatusFeatures + + val getFeatureContext: FeatureContext = + BoundFeatureSet(allFeatures).toFeatureContext + + val dynamicHydrationConfig: DynamicHydrationConfig[HasParams] = + DynamicHydrationConfig( + Set( + GatedFeatures( + boundFeatureSet = BoundFeatureSet(FeatureStoreFeatures.targetUserStatusFeatures), + gate = HasParams + .paramGate(FeatureStoreSourceParams.EnableSeparateClientForGizmoduck) & + HasParams.paramGate(FeatureStoreSourceParams.EnableTargetUserFeatures) + ), + GatedFeatures( + boundFeatureSet = BoundFeatureSet(FeatureStoreFeatures.candidateUserStatusFeatures), + gate = + HasParams + .paramGate(FeatureStoreSourceParams.EnableSeparateClientForGizmoduck) & + HasParams.paramGate(FeatureStoreSourceParams.EnableCandidateUserFeatures) + ), + GatedFeatures( + boundFeatureSet = BoundFeatureSet(FeatureStoreFeatures.similarToUserStatusFeatures), + gate = + HasParams + .paramGate(FeatureStoreSourceParams.EnableSeparateClientForGizmoduck) & + HasParams.paramGate(FeatureStoreSourceParams.EnableSimilarToUserFeatures) + ), + )) + +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/sources/FeatureStoreParameters.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/sources/FeatureStoreParameters.scala new file mode 100644 index 0000000000..468a15ff65 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/sources/FeatureStoreParameters.scala @@ -0,0 +1,79 @@ +package com.twitter.follow_recommendations.common.feature_hydration.sources + +import com.twitter.conversions.DurationOps._ +import com.twitter.ml.featurestore.catalog.datasets.core.UserMobileSdkDataset +import com.twitter.ml.featurestore.catalog.datasets.core.UsersourceEntityDataset +import com.twitter.ml.featurestore.catalog.datasets.customer_journey.PostNuxAlgorithmIdAggregateDataset +import com.twitter.ml.featurestore.catalog.datasets.customer_journey.PostNuxAlgorithmTypeAggregateDataset +import com.twitter.ml.featurestore.catalog.datasets.magicrecs.NotificationSummariesEntityDataset +import com.twitter.ml.featurestore.catalog.datasets.onboarding.MetricCenterUserCountingFeaturesDataset +import com.twitter.ml.featurestore.catalog.datasets.onboarding.UserWtfAlgorithmAggregateFeaturesDataset +import com.twitter.ml.featurestore.catalog.datasets.onboarding.WhoToFollowPostNuxFeaturesDataset +import com.twitter.ml.featurestore.catalog.datasets.rux.UserRecentReactivationTimeDataset +import com.twitter.ml.featurestore.catalog.datasets.timelines.AuthorFeaturesEntityDataset +import com.twitter.ml.featurestore.lib.dataset.DatasetParams +import com.twitter.ml.featurestore.lib.dataset.online.BatchingPolicy +import com.twitter.ml.featurestore.lib.params.FeatureStoreParams +import com.twitter.strato.opcontext.Attribution.ManhattanAppId +import com.twitter.strato.opcontext.ServeWithin + +object FeatureStoreParameters { + + private val FeatureServiceBatchSize = 100 + + val featureStoreParams = FeatureStoreParams( + global = DatasetParams( + serveWithin = Some(ServeWithin(duration = 240.millis, roundTripAllowance = None)), + attributions = Seq( + ManhattanAppId("omega", "wtf_impression_store"), + ManhattanAppId("athena", "wtf_athena"), + ManhattanAppId("starbuck", "wtf_starbuck"), + ManhattanAppId("apollo", "wtf_apollo") + ), + batchingPolicy = Some(BatchingPolicy.Isolated(FeatureServiceBatchSize)) + ), + perDataset = Map( + MetricCenterUserCountingFeaturesDataset.id -> + DatasetParams( + stratoSuffix = Some("onboarding"), + batchingPolicy = Some(BatchingPolicy.Isolated(200)) + ), + UsersourceEntityDataset.id -> + DatasetParams( + stratoSuffix = Some("onboarding") + ), + WhoToFollowPostNuxFeaturesDataset.id -> + DatasetParams( + stratoSuffix = Some("onboarding"), + batchingPolicy = Some(BatchingPolicy.Isolated(200)) + ), + AuthorFeaturesEntityDataset.id -> + DatasetParams( + stratoSuffix = Some("onboarding"), + batchingPolicy = Some(BatchingPolicy.Isolated(10)) + ), + UserRecentReactivationTimeDataset.id -> DatasetParams( + stratoSuffix = + None // removed due to low hit rate. we should use a negative cache in the future + ), + UserWtfAlgorithmAggregateFeaturesDataset.id -> DatasetParams( + stratoSuffix = None + ), + NotificationSummariesEntityDataset.id -> DatasetParams( + stratoSuffix = Some("onboarding"), + serveWithin = Some(ServeWithin(duration = 45.millis, roundTripAllowance = None)), + batchingPolicy = Some(BatchingPolicy.Isolated(10)) + ), + UserMobileSdkDataset.id -> DatasetParams( + stratoSuffix = Some("onboarding") + ), + PostNuxAlgorithmIdAggregateDataset.id -> DatasetParams( + stratoSuffix = Some("onboarding") + ), + PostNuxAlgorithmTypeAggregateDataset.id -> DatasetParams( + stratoSuffix = Some("onboarding") + ), + ), + enableFeatureGenerationStats = true + ) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/sources/FeatureStorePostNuxAlgorithmSource.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/sources/FeatureStorePostNuxAlgorithmSource.scala new file mode 100644 index 0000000000..6821e6a5fc --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/sources/FeatureStorePostNuxAlgorithmSource.scala @@ -0,0 +1,232 @@ +package com.twitter.follow_recommendations.common.feature_hydration.sources + +import com.github.benmanes.caffeine.cache.Caffeine +import com.google.inject.Inject +import com.twitter.conversions.DurationOps._ +import com.twitter.finagle.TimeoutException +import com.twitter.finagle.mtls.authentication.ServiceIdentifier +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.follow_recommendations.common.constants.CandidateAlgorithmTypeConstants +import com.twitter.follow_recommendations.common.feature_hydration.adapters.CandidateAlgorithmAdapter.remapCandidateSource +import com.twitter.follow_recommendations.common.feature_hydration.adapters.PostNuxAlgorithmIdAdapter +import com.twitter.follow_recommendations.common.feature_hydration.adapters.PostNuxAlgorithmTypeAdapter +import com.twitter.follow_recommendations.common.feature_hydration.common.FeatureSource +import com.twitter.follow_recommendations.common.feature_hydration.common.FeatureSourceId +import com.twitter.follow_recommendations.common.feature_hydration.common.HasPreFetchedFeature +import com.twitter.follow_recommendations.common.feature_hydration.sources.Utils.adaptAdditionalFeaturesToDataRecord +import com.twitter.follow_recommendations.common.feature_hydration.sources.Utils.randomizedTTL +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.follow_recommendations.common.models.HasDisplayLocation +import com.twitter.follow_recommendations.common.models.HasSimilarToContext +import com.twitter.hermit.constants.AlgorithmFeedbackTokens.AlgorithmToFeedbackTokenMap +import com.twitter.ml.api.DataRecord +import com.twitter.ml.api.DataRecordMerger +import com.twitter.ml.api.FeatureContext +import com.twitter.ml.api.IRecordOneToOneAdapter +import com.twitter.ml.featurestore.catalog.datasets.customer_journey.PostNuxAlgorithmIdAggregateDataset +import com.twitter.ml.featurestore.catalog.datasets.customer_journey.PostNuxAlgorithmTypeAggregateDataset +import com.twitter.ml.featurestore.catalog.entities.onboarding.{WtfAlgorithm => OnboardingWtfAlgoId} +import com.twitter.ml.featurestore.catalog.entities.onboarding.{ + WtfAlgorithmType => OnboardingWtfAlgoType +} +import com.twitter.ml.featurestore.catalog.features.customer_journey.CombineAllFeaturesPolicy +import com.twitter.ml.featurestore.lib.EntityId +import com.twitter.ml.featurestore.lib.WtfAlgorithmId +import com.twitter.ml.featurestore.lib.WtfAlgorithmType +import com.twitter.ml.featurestore.lib.data.PredictionRecord +import com.twitter.ml.featurestore.lib.data.PredictionRecordAdapter +import com.twitter.ml.featurestore.lib.dataset.DatasetId +import com.twitter.ml.featurestore.lib.dataset.online.Hydrator.HydrationResponse +import com.twitter.ml.featurestore.lib.dataset.online.OnlineAccessDataset +import com.twitter.ml.featurestore.lib.dynamic.ClientConfig +import com.twitter.ml.featurestore.lib.dynamic.DynamicFeatureStoreClient +import com.twitter.ml.featurestore.lib.dynamic.DynamicHydrationConfig +import com.twitter.ml.featurestore.lib.dynamic.FeatureStoreParamsConfig +import com.twitter.ml.featurestore.lib.dynamic.GatedFeatures +import com.twitter.ml.featurestore.lib.entity.EntityWithId +import com.twitter.ml.featurestore.lib.feature.BoundFeature +import com.twitter.ml.featurestore.lib.feature.BoundFeatureSet +import com.twitter.ml.featurestore.lib.online.DatasetValuesCache +import com.twitter.ml.featurestore.lib.online.FeatureStoreRequest +import com.twitter.ml.featurestore.lib.online.OnlineFeatureGenerationStats +import com.twitter.product_mixer.core.model.marshalling.request.HasClientContext +import com.twitter.stitch.Stitch +import com.twitter.timelines.configapi.HasParams +import java.util.concurrent.TimeUnit +import scala.collection.JavaConverters._ + +class FeatureStorePostNuxAlgorithmSource @Inject() ( + serviceIdentifier: ServiceIdentifier, + stats: StatsReceiver) + extends FeatureSource { + import FeatureStorePostNuxAlgorithmSource._ + + val backupSourceStats = stats.scope("feature_store_hydration_post_nux_algorithm") + val adapterStats = backupSourceStats.scope("adapters") + override def id: FeatureSourceId = FeatureSourceId.FeatureStorePostNuxAlgorithmSourceId + override def featureContext: FeatureContext = getFeatureContext + + private val dataRecordMerger = new DataRecordMerger + + val clientConfig: ClientConfig[HasParams] = ClientConfig( + dynamicHydrationConfig = dynamicHydrationConfig, + featureStoreParamsConfig = + FeatureStoreParamsConfig(FeatureStoreParameters.featureStoreParams, Map.empty), + /** + * The smaller one between `timeoutProvider` and `FeatureStoreSourceParams.GlobalFetchTimeout` + * used below takes effect. + */ + timeoutProvider = Function.const(800.millis), + serviceIdentifier = serviceIdentifier + ) + + private val datasetsToCache = Set( + PostNuxAlgorithmIdAggregateDataset, + PostNuxAlgorithmTypeAggregateDataset, + ).asInstanceOf[Set[OnlineAccessDataset[_ <: EntityId, _]]] + + private val datasetValuesCache: DatasetValuesCache = + DatasetValuesCache( + Caffeine + .newBuilder() + .expireAfterWrite(randomizedTTL(12.hours.inSeconds), TimeUnit.SECONDS) + .maximumSize(DefaultCacheMaxKeys) + .build[(_ <: EntityId, DatasetId), Stitch[HydrationResponse[_]]] + .asMap, + datasetsToCache, + DatasetCacheScope + ) + + private val dynamicFeatureStoreClient = DynamicFeatureStoreClient( + clientConfig, + backupSourceStats, + Set(datasetValuesCache) + ) + + private val adapterToDataRecord: IRecordOneToOneAdapter[PredictionRecord] = + PredictionRecordAdapter.oneToOne( + BoundFeatureSet(allFeatures), + OnlineFeatureGenerationStats(backupSourceStats) + ) + + // These two calculate the rate for each feature by dividing it by the number of impressions, then + // apply a log transformation. + private val transformAdapters = Seq(PostNuxAlgorithmIdAdapter, PostNuxAlgorithmTypeAdapter) + override def hydrateFeatures( + target: HasClientContext + with HasPreFetchedFeature + with HasParams + with HasSimilarToContext + with HasDisplayLocation, + candidates: Seq[CandidateUser] + ): Stitch[Map[CandidateUser, DataRecord]] = { + target.getOptionalUserId + .map { _: Long => + val candidateAlgoIdEntities = candidates.map { candidate => + candidate.id -> candidate.getAllAlgorithms + .flatMap { algo => + AlgorithmToFeedbackTokenMap.get(remapCandidateSource(algo)) + }.map(algoId => OnboardingWtfAlgoId.withId(WtfAlgorithmId(algoId))) + }.toMap + + val candidateAlgoTypeEntities = candidateAlgoIdEntities.map { + case (candidateId, algoIdEntities) => + candidateId -> algoIdEntities + .map(_.id.algoId) + .flatMap(algoId => CandidateAlgorithmTypeConstants.getAlgorithmTypes(algoId.toString)) + .distinct + .map(algoType => OnboardingWtfAlgoType.withId(WtfAlgorithmType(algoType))) + } + + val entities = { + candidateAlgoIdEntities.values.flatten ++ candidateAlgoTypeEntities.values.flatten + }.toSeq.distinct + val requests = entities.map(entity => FeatureStoreRequest(Seq(entity))) + + val predictionRecordsFut = dynamicFeatureStoreClient(requests, target) + val candidateFeatureMap = predictionRecordsFut.map { + predictionRecords: Seq[PredictionRecord] => + val entityFeatureMap: Map[EntityWithId[_], DataRecord] = entities + .zip(predictionRecords).map { + case (entity, predictionRecord) => + entity -> adaptAdditionalFeaturesToDataRecord( + adapterToDataRecord.adaptToDataRecord(predictionRecord), + adapterStats, + transformAdapters) + }.toMap + + // In case we have more than one algorithm ID, or type, for a candidate, we merge the + // resulting DataRecords using the two merging policies below. + val algoIdMergeFn = + CombineAllFeaturesPolicy(PostNuxAlgorithmIdAdapter.getFeatures).getMergeFn + val algoTypeMergeFn = + CombineAllFeaturesPolicy(PostNuxAlgorithmTypeAdapter.getFeatures).getMergeFn + + val candidateAlgoIdFeaturesMap = candidateAlgoIdEntities.mapValues { entities => + val features = entities.flatMap(e => Option(entityFeatureMap.getOrElse(e, null))) + algoIdMergeFn(features) + } + + val candidateAlgoTypeFeaturesMap = candidateAlgoTypeEntities.mapValues { entities => + val features = entities.flatMap(e => Option(entityFeatureMap.getOrElse(e, null))) + algoTypeMergeFn(features) + } + + candidates.map { candidate => + val idDrOpt = candidateAlgoIdFeaturesMap.getOrElse(candidate.id, None) + val typeDrOpt = candidateAlgoTypeFeaturesMap.getOrElse(candidate.id, None) + + val featureDr = (idDrOpt, typeDrOpt) match { + case (None, Some(typeDataRecord)) => typeDataRecord + case (Some(idDataRecord), None) => idDataRecord + case (None, None) => new DataRecord() + case (Some(idDataRecord), Some(typeDataRecord)) => + dataRecordMerger.merge(idDataRecord, typeDataRecord) + idDataRecord + } + candidate -> featureDr + }.toMap + } + Stitch + .callFuture(candidateFeatureMap) + .within(target.params(FeatureStoreSourceParams.GlobalFetchTimeout))( + com.twitter.finagle.util.DefaultTimer) + .rescue { + case _: TimeoutException => + Stitch.value(Map.empty[CandidateUser, DataRecord]) + } + }.getOrElse(Stitch.value(Map.empty[CandidateUser, DataRecord])) + } +} + +object FeatureStorePostNuxAlgorithmSource { + private val DatasetCacheScope = "feature_store_local_cache_post_nux_algorithm" + private val DefaultCacheMaxKeys = 1000 // Both of these datasets have <50 keys total. + + val allFeatures: Set[BoundFeature[_ <: EntityId, _]] = + FeatureStoreFeatures.postNuxAlgorithmIdAggregateFeatures ++ + FeatureStoreFeatures.postNuxAlgorithmTypeAggregateFeatures + + val algoIdFinalFeatures = CombineAllFeaturesPolicy( + PostNuxAlgorithmIdAdapter.getFeatures).outputFeaturesPostMerge.toSeq + val algoTypeFinalFeatures = CombineAllFeaturesPolicy( + PostNuxAlgorithmTypeAdapter.getFeatures).outputFeaturesPostMerge.toSeq + + val getFeatureContext: FeatureContext = + new FeatureContext().addFeatures((algoIdFinalFeatures ++ algoTypeFinalFeatures).asJava) + + val dynamicHydrationConfig: DynamicHydrationConfig[HasParams] = + DynamicHydrationConfig( + Set( + GatedFeatures( + boundFeatureSet = + BoundFeatureSet(FeatureStoreFeatures.postNuxAlgorithmIdAggregateFeatures), + gate = HasParams.paramGate(FeatureStoreSourceParams.EnableAlgorithmAggregateFeatures) + ), + GatedFeatures( + boundFeatureSet = + BoundFeatureSet(FeatureStoreFeatures.postNuxAlgorithmTypeAggregateFeatures), + gate = HasParams.paramGate(FeatureStoreSourceParams.EnableAlgorithmAggregateFeatures) + ), + )) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/sources/FeatureStoreSource.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/sources/FeatureStoreSource.scala new file mode 100644 index 0000000000..991bdee864 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/sources/FeatureStoreSource.scala @@ -0,0 +1,368 @@ +package com.twitter.follow_recommendations.common.feature_hydration.sources + +import com.github.benmanes.caffeine.cache.Caffeine +import com.google.inject.Inject +import com.google.inject.Singleton +import com.twitter.conversions.DurationOps._ +import com.twitter.finagle.TimeoutException +import com.twitter.finagle.mtls.authentication.ServiceIdentifier +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.follow_recommendations.common.feature_hydration.adapters.CandidateAlgorithmAdapter.remapCandidateSource +import com.twitter.follow_recommendations.common.feature_hydration.common.FeatureSource +import com.twitter.follow_recommendations.common.feature_hydration.common.FeatureSourceId +import com.twitter.follow_recommendations.common.feature_hydration.common.HasPreFetchedFeature +import com.twitter.follow_recommendations.common.feature_hydration.sources.Utils.adaptAdditionalFeaturesToDataRecord +import com.twitter.follow_recommendations.common.feature_hydration.sources.Utils.randomizedTTL +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.follow_recommendations.common.models.HasDisplayLocation +import com.twitter.follow_recommendations.common.models.HasSimilarToContext +import com.twitter.hermit.constants.AlgorithmFeedbackTokens.AlgorithmToFeedbackTokenMap +import com.twitter.ml.api.DataRecord +import com.twitter.ml.api.FeatureContext +import com.twitter.ml.api.IRecordOneToOneAdapter +import com.twitter.ml.featurestore.catalog.datasets.core.UsersourceEntityDataset +import com.twitter.ml.featurestore.catalog.datasets.magicrecs.NotificationSummariesEntityDataset +import com.twitter.ml.featurestore.catalog.datasets.onboarding.MetricCenterUserCountingFeaturesDataset +import com.twitter.ml.featurestore.catalog.datasets.timelines.AuthorFeaturesEntityDataset +import com.twitter.ml.featurestore.catalog.entities.core.{Author => AuthorEntity} +import com.twitter.ml.featurestore.catalog.entities.core.{AuthorTopic => AuthorTopicEntity} +import com.twitter.ml.featurestore.catalog.entities.core.{CandidateUser => CandidateUserEntity} +import com.twitter.ml.featurestore.catalog.entities.core.{Topic => TopicEntity} +import com.twitter.ml.featurestore.catalog.entities.core.{User => UserEntity} +import com.twitter.ml.featurestore.catalog.entities.core.{UserCandidate => UserCandidateEntity} +import com.twitter.ml.featurestore.catalog.entities.onboarding.UserWtfAlgorithmEntity +import com.twitter.ml.featurestore.lib.data.PredictionRecord +import com.twitter.ml.featurestore.lib.data.PredictionRecordAdapter +import com.twitter.ml.featurestore.lib.dataset.online.Hydrator.HydrationResponse +import com.twitter.ml.featurestore.lib.dataset.online.OnlineAccessDataset +import com.twitter.ml.featurestore.lib.dataset.DatasetId +import com.twitter.ml.featurestore.lib.dynamic._ +import com.twitter.ml.featurestore.lib.feature._ +import com.twitter.ml.featurestore.lib.online.DatasetValuesCache +import com.twitter.ml.featurestore.lib.online.FeatureStoreRequest +import com.twitter.ml.featurestore.lib.online.OnlineFeatureGenerationStats +import com.twitter.ml.featurestore.lib.EdgeEntityId +import com.twitter.ml.featurestore.lib.EntityId +import com.twitter.ml.featurestore.lib.TopicId +import com.twitter.ml.featurestore.lib.UserId +import com.twitter.ml.featurestore.lib.WtfAlgorithmId +import com.twitter.onboarding.relevance.adapters.features.featurestore.CandidateAuthorTopicAggregatesAdapter +import com.twitter.onboarding.relevance.adapters.features.featurestore.CandidateTopicEngagementRealTimeAggregatesAdapter +import com.twitter.onboarding.relevance.adapters.features.featurestore.CandidateTopicEngagementUserStateRealTimeAggregatesAdapter +import com.twitter.onboarding.relevance.adapters.features.featurestore.CandidateTopicNegativeEngagementUserStateRealTimeAggregatesAdapter +import com.twitter.onboarding.relevance.adapters.features.featurestore.FeatureStoreAdapter +import com.twitter.product_mixer.core.model.marshalling.request.HasClientContext +import com.twitter.stitch.Stitch +import com.twitter.timelines.configapi.HasParams + +import java.util.concurrent.TimeUnit + +@Singleton +class FeatureStoreSource @Inject() ( + serviceIdentifier: ServiceIdentifier, + stats: StatsReceiver) + extends FeatureSource { + import FeatureStoreSource._ + + override val id: FeatureSourceId = FeatureSourceId.FeatureStoreSourceId + override val featureContext: FeatureContext = FeatureStoreSource.getFeatureContext + val hydrateFeaturesStats = stats.scope("hydrate_features") + val adapterStats = stats.scope("adapters") + val featureSet: BoundFeatureSet = BoundFeatureSet(FeatureStoreSource.allFeatures) + val clientConfig: ClientConfig[HasParams] = ClientConfig( + dynamicHydrationConfig = FeatureStoreSource.dynamicHydrationConfig, + featureStoreParamsConfig = + FeatureStoreParamsConfig(FeatureStoreParameters.featureStoreParams, Map.empty), + /** + * The smaller one between `timeoutProvider` and `FeatureStoreSourceParams.GlobalFetchTimeout` + * used below takes effect. + */ + timeoutProvider = Function.const(800.millis), + serviceIdentifier = serviceIdentifier + ) + + private val datasetsToCache = Set( + MetricCenterUserCountingFeaturesDataset, + UsersourceEntityDataset, + AuthorFeaturesEntityDataset, + NotificationSummariesEntityDataset + ).asInstanceOf[Set[OnlineAccessDataset[_ <: EntityId, _]]] + + private val datasetValuesCache: DatasetValuesCache = + DatasetValuesCache( + Caffeine + .newBuilder() + .expireAfterWrite(randomizedTTL(12.hours.inSeconds), TimeUnit.SECONDS) + .maximumSize(DefaultCacheMaxKeys) + .build[(_ <: EntityId, DatasetId), Stitch[HydrationResponse[_]]] + .asMap, + datasetsToCache, + DatasetCacheScope + ) + + private val dynamicFeatureStoreClient = DynamicFeatureStoreClient( + clientConfig, + stats, + Set(datasetValuesCache) + ) + + private val adapter: IRecordOneToOneAdapter[PredictionRecord] = + PredictionRecordAdapter.oneToOne( + BoundFeatureSet(allFeatures), + OnlineFeatureGenerationStats(stats) + ) + + override def hydrateFeatures( + target: HasClientContext + with HasPreFetchedFeature + with HasParams + with HasSimilarToContext + with HasDisplayLocation, + candidates: Seq[CandidateUser] + ): Stitch[Map[CandidateUser, DataRecord]] = { + target.getOptionalUserId + .map { targetUserId => + val featureRequests = candidates.map { candidate => + val userId = UserId(targetUserId) + val userEntityId = UserEntity.withId(userId) + val candidateEntityId = CandidateUserEntity.withId(UserId(candidate.id)) + val userCandidateEdgeEntityId = + UserCandidateEntity.withId(EdgeEntityId(userId, UserId(candidate.id))) + val similarToUserId = target.similarToUserIds.map(id => AuthorEntity.withId(UserId(id))) + val topicProof = candidate.reason.flatMap(_.accountProof.flatMap(_.topicProof)) + val topicEntities = if (topicProof.isDefined) { + hydrateFeaturesStats.counter("candidates_with_topic_proof").incr() + val topicId = topicProof.get.topicId + val topicEntityId = TopicEntity.withId(TopicId(topicId)) + val authorTopicEntityId = + AuthorTopicEntity.withId(EdgeEntityId(UserId(candidate.id), TopicId(topicId))) + Seq(topicEntityId, authorTopicEntityId) + } else Nil + + val candidateAlgorithmsWithScores = candidate.getAllAlgorithms + val userWtfAlgEdgeEntities = + candidateAlgorithmsWithScores.flatMap(algo => { + val algoId = AlgorithmToFeedbackTokenMap.get(remapCandidateSource(algo)) + algoId.map(id => + UserWtfAlgorithmEntity.withId(EdgeEntityId(userId, WtfAlgorithmId(id)))) + }) + + val entities = Seq( + userEntityId, + candidateEntityId, + userCandidateEdgeEntityId) ++ similarToUserId ++ topicEntities ++ userWtfAlgEdgeEntities + FeatureStoreRequest(entities) + } + + val predictionRecordsFut = dynamicFeatureStoreClient(featureRequests, target) + val candidateFeatureMap = predictionRecordsFut.map { predictionRecords => + // we can zip predictionRecords with candidates as the order is preserved in the client + candidates + .zip(predictionRecords).map { + case (candidate, predictionRecord) => + candidate -> adaptAdditionalFeaturesToDataRecord( + adapter.adaptToDataRecord(predictionRecord), + adapterStats, + FeatureStoreSource.featureAdapters) + }.toMap + } + Stitch + .callFuture(candidateFeatureMap) + .within(target.params(FeatureStoreSourceParams.GlobalFetchTimeout))( + com.twitter.finagle.util.DefaultTimer) + .rescue { + case _: TimeoutException => + Stitch.value(Map.empty[CandidateUser, DataRecord]) + } + }.getOrElse(Stitch.value(Map.empty[CandidateUser, DataRecord])) + } +} + +// list of features that we will be fetching, even if we are only scribing but not scoring with them +object FeatureStoreSource { + + private val DatasetCacheScope = "feature_store_local_cache" + private val DefaultCacheMaxKeys = 70000 + + import FeatureStoreFeatures._ + + ///////////////////// ALL hydrated features ///////////////////// + val allFeatures: Set[BoundFeature[_ <: EntityId, _]] = + //target user + targetUserFeatures ++ + targetUserUserAuthorUserStateRealTimeAggregatesFeature ++ + targetUserResurrectionFeatures ++ + targetUserWtfImpressionFeatures ++ + targetUserStatusFeatures ++ + targetUserMetricCountFeatures ++ + //candidate user + candidateUserFeatures ++ + candidateUserResurrectionFeatures ++ + candidateUserAuthorRealTimeAggregateFeatures ++ + candidateUserStatusFeatures ++ + candidateUserMetricCountFeatures ++ + candidateUserTimelinesAuthorAggregateFeatures ++ + candidateUserClientFeatures ++ + //similar to user + similarToUserFeatures ++ + similarToUserStatusFeatures ++ + similarToUserMetricCountFeatures ++ + similarToUserTimelinesAuthorAggregateFeatures ++ + //other + userCandidateEdgeFeatures ++ + userCandidateWtfImpressionCandidateFeatures ++ + topicFeatures ++ + userWtfAlgorithmEdgeFeatures ++ + targetUserClientFeatures + + val dynamicHydrationConfig: DynamicHydrationConfig[HasParams] = + DynamicHydrationConfig( + Set( + GatedFeatures( + boundFeatureSet = BoundFeatureSet(topicAggregateFeatures), + gate = HasParams.paramGate(FeatureStoreSourceParams.EnableTopicAggregateFeatures) + ), + GatedFeatures( + boundFeatureSet = BoundFeatureSet(authorTopicFeatures), + gate = + HasParams + .paramGate(FeatureStoreSourceParams.EnableSeparateClientForTimelinesAuthors).unary_! & + HasParams.paramGate(FeatureStoreSourceParams.EnableAuthorTopicAggregateFeatures) + ), + GatedFeatures( + boundFeatureSet = BoundFeatureSet(userTopicFeatures), + gate = HasParams.paramGate(FeatureStoreSourceParams.EnableUserTopicFeatures) + ), + GatedFeatures( + boundFeatureSet = BoundFeatureSet(targetUserFeatures), + gate = HasParams.paramGate(FeatureStoreSourceParams.EnableTargetUserFeatures) + ), + GatedFeatures( + boundFeatureSet = BoundFeatureSet(targetUserUserAuthorUserStateRealTimeAggregatesFeature), + gate = HasParams.paramGate( + FeatureStoreSourceParams.EnableTargetUserUserAuthorUserStateRealTimeAggregatesFeature) + ), + GatedFeatures( + boundFeatureSet = BoundFeatureSet(targetUserResurrectionFeatures), + gate = HasParams.paramGate(FeatureStoreSourceParams.EnableTargetUserResurrectionFeatures) + ), + GatedFeatures( + boundFeatureSet = BoundFeatureSet(targetUserWtfImpressionFeatures), + gate = HasParams.paramGate(FeatureStoreSourceParams.EnableTargetUserWtfImpressionFeatures) + ), + GatedFeatures( + boundFeatureSet = BoundFeatureSet(targetUserStatusFeatures), + gate = + HasParams.paramGate(FeatureStoreSourceParams.EnableSeparateClientForGizmoduck).unary_! & + HasParams.paramGate(FeatureStoreSourceParams.EnableTargetUserFeatures) + ), + GatedFeatures( + boundFeatureSet = BoundFeatureSet(targetUserMetricCountFeatures), + gate = HasParams + .paramGate( + FeatureStoreSourceParams.EnableSeparateClientForMetricCenterUserCounting).unary_! & + HasParams.paramGate(FeatureStoreSourceParams.EnableTargetUserFeatures) + ), + GatedFeatures( + boundFeatureSet = BoundFeatureSet(candidateUserFeatures), + gate = HasParams.paramGate(FeatureStoreSourceParams.EnableCandidateUserFeatures) + ), + GatedFeatures( + boundFeatureSet = BoundFeatureSet(candidateUserAuthorRealTimeAggregateFeatures), + gate = HasParams.paramGate( + FeatureStoreSourceParams.EnableCandidateUserAuthorRealTimeAggregateFeatures) + ), + GatedFeatures( + boundFeatureSet = BoundFeatureSet(candidateUserResurrectionFeatures), + gate = + HasParams.paramGate(FeatureStoreSourceParams.EnableCandidateUserResurrectionFeatures) + ), + GatedFeatures( + boundFeatureSet = BoundFeatureSet(candidateUserStatusFeatures), + gate = + HasParams.paramGate(FeatureStoreSourceParams.EnableSeparateClientForGizmoduck).unary_! & + HasParams.paramGate(FeatureStoreSourceParams.EnableCandidateUserFeatures) + ), + GatedFeatures( + boundFeatureSet = BoundFeatureSet(candidateUserTimelinesAuthorAggregateFeatures), + gate = + HasParams + .paramGate(FeatureStoreSourceParams.EnableSeparateClientForTimelinesAuthors).unary_! & + HasParams.paramGate( + FeatureStoreSourceParams.EnableCandidateUserTimelinesAuthorAggregateFeatures) + ), + GatedFeatures( + boundFeatureSet = BoundFeatureSet(candidateUserMetricCountFeatures), + gate = + HasParams + .paramGate( + FeatureStoreSourceParams.EnableSeparateClientForMetricCenterUserCounting).unary_! & + HasParams.paramGate(FeatureStoreSourceParams.EnableCandidateUserFeatures) + ), + GatedFeatures( + boundFeatureSet = BoundFeatureSet(userCandidateEdgeFeatures), + gate = HasParams.paramGate(FeatureStoreSourceParams.EnableUserCandidateEdgeFeatures) + ), + GatedFeatures( + boundFeatureSet = BoundFeatureSet(userCandidateWtfImpressionCandidateFeatures), + gate = HasParams.paramGate( + FeatureStoreSourceParams.EnableUserCandidateWtfImpressionCandidateFeatures) + ), + GatedFeatures( + boundFeatureSet = BoundFeatureSet(userWtfAlgorithmEdgeFeatures), + gate = HasParams.paramGate(FeatureStoreSourceParams.EnableUserWtfAlgEdgeFeatures) + ), + GatedFeatures( + boundFeatureSet = BoundFeatureSet(similarToUserFeatures), + gate = HasParams.paramGate(FeatureStoreSourceParams.EnableSimilarToUserFeatures) + ), + GatedFeatures( + boundFeatureSet = BoundFeatureSet(similarToUserStatusFeatures), + gate = + HasParams.paramGate(FeatureStoreSourceParams.EnableSeparateClientForGizmoduck).unary_! & + HasParams.paramGate(FeatureStoreSourceParams.EnableSimilarToUserFeatures) + ), + GatedFeatures( + boundFeatureSet = BoundFeatureSet(similarToUserTimelinesAuthorAggregateFeatures), + gate = + HasParams + .paramGate(FeatureStoreSourceParams.EnableSeparateClientForTimelinesAuthors).unary_! & + HasParams.paramGate(FeatureStoreSourceParams.EnableSimilarToUserFeatures) + ), + GatedFeatures( + boundFeatureSet = BoundFeatureSet(similarToUserMetricCountFeatures), + gate = + HasParams + .paramGate( + FeatureStoreSourceParams.EnableSeparateClientForMetricCenterUserCounting).unary_! & + HasParams.paramGate(FeatureStoreSourceParams.EnableSimilarToUserFeatures) + ), + GatedFeatures( + boundFeatureSet = BoundFeatureSet(candidateUserClientFeatures), + gate = HasParams.paramGate(FeatureStoreSourceParams.EnableCandidateClientFeatures) + ), + GatedFeatures( + boundFeatureSet = BoundFeatureSet(targetUserClientFeatures), + gate = HasParams.paramGate(FeatureStoreSourceParams.EnableUserClientFeatures) + ), + ) + ) + // for calibrating features, e.g. add log transformed topic features + val featureAdapters: Seq[FeatureStoreAdapter] = Seq( + CandidateTopicEngagementRealTimeAggregatesAdapter, + CandidateTopicNegativeEngagementUserStateRealTimeAggregatesAdapter, + CandidateTopicEngagementUserStateRealTimeAggregatesAdapter, + CandidateAuthorTopicAggregatesAdapter + ) + val additionalFeatureContext: FeatureContext = FeatureContext.merge( + featureAdapters + .foldRight(new FeatureContext())((adapter, context) => + context + .addFeatures(adapter.getFeatureContext)) + ) + val getFeatureContext: FeatureContext = + BoundFeatureSet(allFeatures).toFeatureContext + .addFeatures(additionalFeatureContext) + // The below are aggregated features that are aggregated for a second time over multiple keys. + .addFeatures(maxSumAvgAggregatedFeatureContext) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/sources/FeatureStoreSourceParams.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/sources/FeatureStoreSourceParams.scala new file mode 100644 index 0000000000..15488ce909 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/sources/FeatureStoreSourceParams.scala @@ -0,0 +1,148 @@ +package com.twitter.follow_recommendations.common.feature_hydration.sources + +import com.twitter.timelines.configapi.DurationConversion +import com.twitter.timelines.configapi.FSBoundedParam +import com.twitter.timelines.configapi.FSParam +import com.twitter.timelines.configapi.HasDurationConversion +import com.twitter.util.Duration +import com.twitter.conversions.DurationOps._ + +object FeatureStoreSourceParams { + case object EnableTopicAggregateFeatures + extends FSParam[Boolean]( + name = FeatureHydrationSourcesFeatureSwitchKeys.EnableTopicAggregateFeatures, + default = true + ) + case object EnableAlgorithmAggregateFeatures + extends FSParam[Boolean]( + name = FeatureHydrationSourcesFeatureSwitchKeys.EnableAlgorithmAggregateFeatures, + default = false + ) + case object EnableAuthorTopicAggregateFeatures + extends FSParam[Boolean]( + name = FeatureHydrationSourcesFeatureSwitchKeys.EnableAuthorTopicAggregateFeatures, + default = true + ) + case object EnableUserTopicFeatures + extends FSParam[Boolean]( + name = FeatureHydrationSourcesFeatureSwitchKeys.EnableUserTopicFeatures, + default = false + ) + case object EnableTargetUserFeatures + extends FSParam[Boolean]( + name = FeatureHydrationSourcesFeatureSwitchKeys.EnableTargetUserFeatures, + default = true + ) + case object EnableTargetUserUserAuthorUserStateRealTimeAggregatesFeature + extends FSParam[Boolean]( + name = + FeatureHydrationSourcesFeatureSwitchKeys.EnableTargetUserUserAuthorUserStateRealTimeAggregatesFeature, + default = true + ) + case object EnableTargetUserResurrectionFeatures + extends FSParam[Boolean]( + name = FeatureHydrationSourcesFeatureSwitchKeys.EnableTargetUserResurrectionFeatures, + default = true + ) + case object EnableTargetUserWtfImpressionFeatures + extends FSParam[Boolean]( + name = FeatureHydrationSourcesFeatureSwitchKeys.EnableTargetUserWtfImpressionFeatures, + default = true + ) + case object EnableCandidateUserFeatures + extends FSParam[Boolean]( + name = FeatureHydrationSourcesFeatureSwitchKeys.EnableCandidateUserFeatures, + default = true + ) + case object EnableCandidateUserAuthorRealTimeAggregateFeatures + extends FSParam[Boolean]( + name = + FeatureHydrationSourcesFeatureSwitchKeys.EnableCandidateUserAuthorRealTimeAggregateFeatures, + default = true + ) + case object EnableCandidateUserResurrectionFeatures + extends FSParam[Boolean]( + name = FeatureHydrationSourcesFeatureSwitchKeys.EnableCandidateUserResurrectionFeatures, + default = true + ) + case object EnableCandidateUserTimelinesAuthorAggregateFeatures + extends FSParam[Boolean]( + name = + FeatureHydrationSourcesFeatureSwitchKeys.EnableCandidateUserTimelinesAuthorAggregateFeatures, + default = true + ) + case object EnableUserCandidateEdgeFeatures + extends FSParam[Boolean]( + name = FeatureHydrationSourcesFeatureSwitchKeys.EnableUserCandidateEdgeFeatures, + default = true + ) + case object EnableUserCandidateWtfImpressionCandidateFeatures + extends FSParam[Boolean]( + name = + FeatureHydrationSourcesFeatureSwitchKeys.EnableUserCandidateWtfImpressionCandidateFeatures, + default = true + ) + case object EnableUserWtfAlgEdgeFeatures + extends FSParam[Boolean]( + name = FeatureHydrationSourcesFeatureSwitchKeys.EnableUserWtfAlgEdgeFeatures, + default = false + ) + case object EnableSimilarToUserFeatures + extends FSParam[Boolean]( + name = FeatureHydrationSourcesFeatureSwitchKeys.EnableSimilarToUserFeatures, + default = true + ) + + case object EnableCandidatePrecomputedNotificationFeatures + extends FSParam[Boolean]( + name = + FeatureHydrationSourcesFeatureSwitchKeys.EnableCandidatePrecomputedNotificationFeatures, + default = false + ) + + case object EnableCandidateClientFeatures + extends FSParam[Boolean]( + name = FeatureHydrationSourcesFeatureSwitchKeys.EnableCandidateClientFeatures, + default = false + ) + + case object EnableUserClientFeatures + extends FSParam[Boolean]( + name = FeatureHydrationSourcesFeatureSwitchKeys.EnableUserClientFeatures, + default = false + ) + + case object EnableSeparateClientForTimelinesAuthors + extends FSParam[Boolean]( + name = FeatureHydrationSourcesFeatureSwitchKeys.UseSeparateClientForTimelinesAuthor, + default = false + ) + + case object EnableSeparateClientForMetricCenterUserCounting + extends FSParam[Boolean]( + name = FeatureHydrationSourcesFeatureSwitchKeys.UseSeparateClientMetricCenterUserCounting, + default = false + ) + + case object EnableSeparateClientForNotifications + extends FSParam[Boolean]( + name = FeatureHydrationSourcesFeatureSwitchKeys.UseSeparateClientForNotifications, + default = false + ) + + case object EnableSeparateClientForGizmoduck + extends FSParam[Boolean]( + name = FeatureHydrationSourcesFeatureSwitchKeys.UseSeparateClientForGizmoduck, + default = false + ) + + case object GlobalFetchTimeout + extends FSBoundedParam[Duration]( + name = FeatureHydrationSourcesFeatureSwitchKeys.FeatureHydrationTimeout, + default = 240.millisecond, + min = 100.millisecond, + max = 400.millisecond) + with HasDurationConversion { + override def durationConversion: DurationConversion = DurationConversion.FromMillis + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/sources/FeatureStoreTimelinesAuthorSource.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/sources/FeatureStoreTimelinesAuthorSource.scala new file mode 100644 index 0000000000..179ae70817 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/sources/FeatureStoreTimelinesAuthorSource.scala @@ -0,0 +1,191 @@ +package com.twitter.follow_recommendations.common.feature_hydration.sources + +import com.github.benmanes.caffeine.cache.Caffeine +import com.google.inject.Inject +import com.twitter.finagle.TimeoutException +import com.twitter.finagle.mtls.authentication.ServiceIdentifier +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.follow_recommendations.common.feature_hydration.common.FeatureSource +import com.twitter.follow_recommendations.common.feature_hydration.common.FeatureSourceId +import com.twitter.follow_recommendations.common.feature_hydration.common.HasPreFetchedFeature +import com.twitter.follow_recommendations.common.feature_hydration.sources.Utils.adaptAdditionalFeaturesToDataRecord +import com.twitter.follow_recommendations.common.feature_hydration.sources.Utils.randomizedTTL +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.follow_recommendations.common.models.HasSimilarToContext +import com.twitter.ml.api.DataRecord +import com.twitter.ml.api.FeatureContext +import com.twitter.ml.api.IRecordOneToOneAdapter +import com.twitter.ml.featurestore.catalog.datasets.timelines.AuthorFeaturesEntityDataset +import com.twitter.ml.featurestore.catalog.entities.core.{Author => AuthorEntity} +import com.twitter.ml.featurestore.catalog.entities.core.{AuthorTopic => AuthorTopicEntity} +import com.twitter.ml.featurestore.catalog.entities.core.{CandidateUser => CandidateUserEntity} +import com.twitter.ml.featurestore.catalog.entities.core.{User => UserEntity} +import com.twitter.ml.featurestore.lib.EdgeEntityId +import com.twitter.ml.featurestore.lib.EntityId +import com.twitter.ml.featurestore.lib.TopicId +import com.twitter.ml.featurestore.lib.UserId +import com.twitter.ml.featurestore.lib.data.PredictionRecord +import com.twitter.ml.featurestore.lib.data.PredictionRecordAdapter +import com.twitter.ml.featurestore.lib.dataset.DatasetId +import com.twitter.ml.featurestore.lib.dataset.online.Hydrator.HydrationResponse +import com.twitter.ml.featurestore.lib.dataset.online.OnlineAccessDataset +import com.twitter.ml.featurestore.lib.dynamic.ClientConfig +import com.twitter.ml.featurestore.lib.dynamic.DynamicFeatureStoreClient +import com.twitter.ml.featurestore.lib.dynamic.DynamicHydrationConfig +import com.twitter.ml.featurestore.lib.dynamic.FeatureStoreParamsConfig +import com.twitter.ml.featurestore.lib.dynamic.GatedFeatures +import com.twitter.ml.featurestore.lib.feature.BoundFeature +import com.twitter.ml.featurestore.lib.feature.BoundFeatureSet +import com.twitter.ml.featurestore.lib.online.DatasetValuesCache +import com.twitter.ml.featurestore.lib.online.FeatureStoreRequest +import com.twitter.ml.featurestore.lib.online.OnlineFeatureGenerationStats +import com.twitter.stitch.Stitch +import com.twitter.timelines.configapi.HasParams +import java.util.concurrent.TimeUnit +import com.twitter.conversions.DurationOps._ +import com.twitter.follow_recommendations.common.models.HasDisplayLocation +import com.twitter.product_mixer.core.model.marshalling.request.HasClientContext + +class FeatureStoreTimelinesAuthorSource @Inject() ( + serviceIdentifier: ServiceIdentifier, + stats: StatsReceiver) + extends FeatureSource { + import FeatureStoreTimelinesAuthorSource._ + + val backupSourceStats = stats.scope("feature_store_hydration_timelines_author") + val adapterStats = backupSourceStats.scope("adapters") + override def id: FeatureSourceId = FeatureSourceId.FeatureStoreTimelinesAuthorSourceId + override def featureContext: FeatureContext = getFeatureContext + + val clientConfig: ClientConfig[HasParams] = ClientConfig( + dynamicHydrationConfig = dynamicHydrationConfig, + featureStoreParamsConfig = + FeatureStoreParamsConfig(FeatureStoreParameters.featureStoreParams, Map.empty), + /** + * The smaller one between `timeoutProvider` and `FeatureStoreSourceParams.GlobalFetchTimeout` + * used below takes effect. + */ + timeoutProvider = Function.const(800.millis), + serviceIdentifier = serviceIdentifier + ) + + private val datasetsToCache = Set( + AuthorFeaturesEntityDataset + ).asInstanceOf[Set[OnlineAccessDataset[_ <: EntityId, _]]] + + private val datasetValuesCache: DatasetValuesCache = + DatasetValuesCache( + Caffeine + .newBuilder() + .expireAfterWrite(randomizedTTL(12.hours.inSeconds), TimeUnit.SECONDS) + .maximumSize(DefaultCacheMaxKeys) + .build[(_ <: EntityId, DatasetId), Stitch[HydrationResponse[_]]] + .asMap, + datasetsToCache, + DatasetCacheScope + ) + + private val dynamicFeatureStoreClient = DynamicFeatureStoreClient( + clientConfig, + backupSourceStats, + Set(datasetValuesCache) + ) + + private val adapter: IRecordOneToOneAdapter[PredictionRecord] = + PredictionRecordAdapter.oneToOne( + BoundFeatureSet(allFeatures), + OnlineFeatureGenerationStats(backupSourceStats) + ) + + override def hydrateFeatures( + target: HasClientContext + with HasPreFetchedFeature + with HasParams + with HasSimilarToContext + with HasDisplayLocation, + candidates: Seq[CandidateUser] + ): Stitch[Map[CandidateUser, DataRecord]] = { + target.getOptionalUserId + .map { targetUserId => + val featureRequests = candidates.map { candidate => + val userEntityId = UserEntity.withId(UserId(targetUserId)) + val candidateEntityId = CandidateUserEntity.withId(UserId(candidate.id)) + val similarToUserId = target.similarToUserIds.map(id => AuthorEntity.withId(UserId(id))) + val topicProof = candidate.reason.flatMap(_.accountProof.flatMap(_.topicProof)) + val authorTopicEntity = if (topicProof.isDefined) { + backupSourceStats.counter("candidates_with_topic_proof").incr() + Set( + AuthorTopicEntity.withId( + EdgeEntityId(UserId(candidate.id), TopicId(topicProof.get.topicId)))) + } else Nil + + val entities = + Seq(userEntityId, candidateEntityId) ++ similarToUserId ++ authorTopicEntity + FeatureStoreRequest(entities) + } + + val predictionRecordsFut = dynamicFeatureStoreClient(featureRequests, target) + val candidateFeatureMap = predictionRecordsFut.map { predictionRecords => + // we can zip predictionRecords with candidates as the order is preserved in the client + candidates + .zip(predictionRecords).map { + case (candidate, predictionRecord) => + candidate -> adaptAdditionalFeaturesToDataRecord( + adapter.adaptToDataRecord(predictionRecord), + adapterStats, + FeatureStoreSource.featureAdapters) + }.toMap + } + Stitch + .callFuture(candidateFeatureMap) + .within(target.params(FeatureStoreSourceParams.GlobalFetchTimeout))( + com.twitter.finagle.util.DefaultTimer) + .rescue { + case _: TimeoutException => + Stitch.value(Map.empty[CandidateUser, DataRecord]) + } + }.getOrElse(Stitch.value(Map.empty[CandidateUser, DataRecord])) + } +} + +object FeatureStoreTimelinesAuthorSource { + private val DatasetCacheScope = "feature_store_local_cache_timelines_author" + private val DefaultCacheMaxKeys = 20000 + + import FeatureStoreFeatures._ + + val allFeatures: Set[BoundFeature[_ <: EntityId, _]] = + similarToUserTimelinesAuthorAggregateFeatures ++ + candidateUserTimelinesAuthorAggregateFeatures ++ + authorTopicFeatures + + val getFeatureContext: FeatureContext = + BoundFeatureSet(allFeatures).toFeatureContext + + val dynamicHydrationConfig: DynamicHydrationConfig[HasParams] = + DynamicHydrationConfig( + Set( + GatedFeatures( + boundFeatureSet = BoundFeatureSet(authorTopicFeatures), + gate = + HasParams + .paramGate(FeatureStoreSourceParams.EnableSeparateClientForTimelinesAuthors) & + HasParams.paramGate(FeatureStoreSourceParams.EnableAuthorTopicAggregateFeatures) + ), + GatedFeatures( + boundFeatureSet = BoundFeatureSet(similarToUserTimelinesAuthorAggregateFeatures), + gate = + HasParams + .paramGate(FeatureStoreSourceParams.EnableSeparateClientForTimelinesAuthors) & + HasParams.paramGate(FeatureStoreSourceParams.EnableSimilarToUserFeatures) + ), + GatedFeatures( + boundFeatureSet = BoundFeatureSet(candidateUserTimelinesAuthorAggregateFeatures), + gate = + HasParams + .paramGate(FeatureStoreSourceParams.EnableSeparateClientForTimelinesAuthors) & + HasParams.paramGate( + FeatureStoreSourceParams.EnableCandidateUserTimelinesAuthorAggregateFeatures) + ), + )) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/sources/FeatureStoreUserMetricCountsSource.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/sources/FeatureStoreUserMetricCountsSource.scala new file mode 100644 index 0000000000..110985c921 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/sources/FeatureStoreUserMetricCountsSource.scala @@ -0,0 +1,187 @@ +package com.twitter.follow_recommendations.common.feature_hydration.sources + +import com.github.benmanes.caffeine.cache.Caffeine +import com.google.inject.Inject +import com.twitter.finagle.TimeoutException +import com.twitter.finagle.mtls.authentication.ServiceIdentifier +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.follow_recommendations.common.feature_hydration.common.FeatureSource +import com.twitter.follow_recommendations.common.feature_hydration.common.FeatureSourceId +import com.twitter.follow_recommendations.common.feature_hydration.common.HasPreFetchedFeature +import com.twitter.follow_recommendations.common.feature_hydration.sources.Utils.adaptAdditionalFeaturesToDataRecord +import com.twitter.follow_recommendations.common.feature_hydration.sources.Utils.randomizedTTL +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.follow_recommendations.common.models.HasSimilarToContext +import com.twitter.ml.api.DataRecord +import com.twitter.ml.api.FeatureContext +import com.twitter.ml.api.IRecordOneToOneAdapter +import com.twitter.ml.featurestore.catalog.datasets.onboarding.MetricCenterUserCountingFeaturesDataset +import com.twitter.ml.featurestore.catalog.entities.core.{Author => AuthorEntity} +import com.twitter.ml.featurestore.catalog.entities.core.{AuthorTopic => AuthorTopicEntity} +import com.twitter.ml.featurestore.catalog.entities.core.{CandidateUser => CandidateUserEntity} +import com.twitter.ml.featurestore.catalog.entities.core.{User => UserEntity} +import com.twitter.ml.featurestore.lib.EdgeEntityId +import com.twitter.ml.featurestore.lib.EntityId +import com.twitter.ml.featurestore.lib.TopicId +import com.twitter.ml.featurestore.lib.UserId +import com.twitter.ml.featurestore.lib.data.PredictionRecord +import com.twitter.ml.featurestore.lib.data.PredictionRecordAdapter +import com.twitter.ml.featurestore.lib.dataset.DatasetId +import com.twitter.ml.featurestore.lib.dataset.online.Hydrator.HydrationResponse +import com.twitter.ml.featurestore.lib.dataset.online.OnlineAccessDataset +import com.twitter.ml.featurestore.lib.dynamic.ClientConfig +import com.twitter.ml.featurestore.lib.dynamic.DynamicFeatureStoreClient +import com.twitter.ml.featurestore.lib.dynamic.DynamicHydrationConfig +import com.twitter.ml.featurestore.lib.dynamic.FeatureStoreParamsConfig +import com.twitter.ml.featurestore.lib.dynamic.GatedFeatures +import com.twitter.ml.featurestore.lib.feature.BoundFeature +import com.twitter.ml.featurestore.lib.feature.BoundFeatureSet +import com.twitter.ml.featurestore.lib.online.DatasetValuesCache +import com.twitter.ml.featurestore.lib.online.FeatureStoreRequest +import com.twitter.ml.featurestore.lib.online.OnlineFeatureGenerationStats +import com.twitter.stitch.Stitch +import com.twitter.timelines.configapi.HasParams +import java.util.concurrent.TimeUnit +import com.twitter.conversions.DurationOps._ +import com.twitter.follow_recommendations.common.models.HasDisplayLocation +import com.twitter.product_mixer.core.model.marshalling.request.HasClientContext + +class FeatureStoreUserMetricCountsSource @Inject() ( + serviceIdentifier: ServiceIdentifier, + stats: StatsReceiver) + extends FeatureSource { + import FeatureStoreUserMetricCountsSource._ + + val backupSourceStats = stats.scope("feature_store_hydration_mc_counting") + val adapterStats = backupSourceStats.scope("adapters") + override def id: FeatureSourceId = FeatureSourceId.FeatureStoreUserMetricCountsSourceId + override def featureContext: FeatureContext = getFeatureContext + + val clientConfig: ClientConfig[HasParams] = ClientConfig( + dynamicHydrationConfig = dynamicHydrationConfig, + featureStoreParamsConfig = + FeatureStoreParamsConfig(FeatureStoreParameters.featureStoreParams, Map.empty), + /** + * The smaller one between `timeoutProvider` and `FeatureStoreSourceParams.GlobalFetchTimeout` + * used below takes effect. + */ + timeoutProvider = Function.const(800.millis), + serviceIdentifier = serviceIdentifier + ) + + private val datasetsToCache = Set( + MetricCenterUserCountingFeaturesDataset + ).asInstanceOf[Set[OnlineAccessDataset[_ <: EntityId, _]]] + + private val datasetValuesCache: DatasetValuesCache = + DatasetValuesCache( + Caffeine + .newBuilder() + .expireAfterWrite(randomizedTTL(12.hours.inSeconds), TimeUnit.SECONDS) + .maximumSize(DefaultCacheMaxKeys) + .build[(_ <: EntityId, DatasetId), Stitch[HydrationResponse[_]]] + .asMap, + datasetsToCache, + DatasetCacheScope + ) + + private val dynamicFeatureStoreClient = DynamicFeatureStoreClient( + clientConfig, + backupSourceStats, + Set(datasetValuesCache) + ) + + private val adapter: IRecordOneToOneAdapter[PredictionRecord] = + PredictionRecordAdapter.oneToOne( + BoundFeatureSet(allFeatures), + OnlineFeatureGenerationStats(backupSourceStats) + ) + + override def hydrateFeatures( + target: HasClientContext + with HasPreFetchedFeature + with HasParams + with HasSimilarToContext + with HasDisplayLocation, + candidates: Seq[CandidateUser] + ): Stitch[Map[CandidateUser, DataRecord]] = { + target.getOptionalUserId + .map { targetUserId => + val featureRequests = candidates.map { candidate => + val userEntityId = UserEntity.withId(UserId(targetUserId)) + val candidateEntityId = CandidateUserEntity.withId(UserId(candidate.id)) + val similarToUserId = target.similarToUserIds.map(id => AuthorEntity.withId(UserId(id))) + val topicProof = candidate.reason.flatMap(_.accountProof.flatMap(_.topicProof)) + val authorTopicEntity = if (topicProof.isDefined) { + backupSourceStats.counter("candidates_with_topic_proof").incr() + Set( + AuthorTopicEntity.withId( + EdgeEntityId(UserId(candidate.id), TopicId(topicProof.get.topicId)))) + } else Nil + + val entities = + Seq(userEntityId, candidateEntityId) ++ similarToUserId ++ authorTopicEntity + FeatureStoreRequest(entities) + } + + val predictionRecordsFut = dynamicFeatureStoreClient(featureRequests, target) + val candidateFeatureMap = predictionRecordsFut.map { predictionRecords => + // we can zip predictionRecords with candidates as the order is preserved in the client + candidates + .zip(predictionRecords).map { + case (candidate, predictionRecord) => + candidate -> adaptAdditionalFeaturesToDataRecord( + adapter.adaptToDataRecord(predictionRecord), + adapterStats, + FeatureStoreSource.featureAdapters) + }.toMap + } + Stitch + .callFuture(candidateFeatureMap) + .within(target.params(FeatureStoreSourceParams.GlobalFetchTimeout))( + com.twitter.finagle.util.DefaultTimer) + .rescue { + case _: TimeoutException => + Stitch.value(Map.empty[CandidateUser, DataRecord]) + } + }.getOrElse(Stitch.value(Map.empty[CandidateUser, DataRecord])) + } +} + +object FeatureStoreUserMetricCountsSource { + private val DatasetCacheScope = "feature_store_local_cache_mc_user_counting" + private val DefaultCacheMaxKeys = 20000 + + val allFeatures: Set[BoundFeature[_ <: EntityId, _]] = + FeatureStoreFeatures.candidateUserMetricCountFeatures ++ + FeatureStoreFeatures.similarToUserMetricCountFeatures ++ + FeatureStoreFeatures.targetUserMetricCountFeatures + + val getFeatureContext: FeatureContext = + BoundFeatureSet(allFeatures).toFeatureContext + + val dynamicHydrationConfig: DynamicHydrationConfig[HasParams] = + DynamicHydrationConfig( + Set( + GatedFeatures( + boundFeatureSet = BoundFeatureSet(FeatureStoreFeatures.targetUserMetricCountFeatures), + gate = HasParams + .paramGate(FeatureStoreSourceParams.EnableSeparateClientForMetricCenterUserCounting) & + HasParams.paramGate(FeatureStoreSourceParams.EnableTargetUserFeatures) + ), + GatedFeatures( + boundFeatureSet = BoundFeatureSet(FeatureStoreFeatures.candidateUserMetricCountFeatures), + gate = + HasParams + .paramGate(FeatureStoreSourceParams.EnableSeparateClientForMetricCenterUserCounting) & + HasParams.paramGate(FeatureStoreSourceParams.EnableCandidateUserFeatures) + ), + GatedFeatures( + boundFeatureSet = BoundFeatureSet(FeatureStoreFeatures.similarToUserMetricCountFeatures), + gate = + HasParams + .paramGate(FeatureStoreSourceParams.EnableSeparateClientForMetricCenterUserCounting) & + HasParams.paramGate(FeatureStoreSourceParams.EnableSimilarToUserFeatures) + ), + )) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/sources/HydrationSourcesModule.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/sources/HydrationSourcesModule.scala new file mode 100644 index 0000000000..59e3ea1868 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/sources/HydrationSourcesModule.scala @@ -0,0 +1,152 @@ +package com.twitter.follow_recommendations.common.feature_hydration.sources + +import com.google.inject.Provides +import com.google.inject.Singleton +import com.twitter.escherbird.util.stitchcache.StitchCache +import com.twitter.finagle.mtls.authentication.ServiceIdentifier +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.inject.TwitterModule +import com.twitter.stitch.Stitch +import com.twitter.storage.client.manhattan.bijections.Bijections.BinaryCompactScalaInjection +import com.twitter.storage.client.manhattan.bijections.Bijections.LongInjection +import com.twitter.storage.client.manhattan.kv.Guarantee +import com.twitter.storage.client.manhattan.kv.ManhattanKVClient +import com.twitter.storage.client.manhattan.kv.ManhattanKVClientMtlsParams +import com.twitter.storage.client.manhattan.kv.ManhattanKVEndpoint +import com.twitter.storage.client.manhattan.kv.ManhattanKVEndpointBuilder +import com.twitter.storage.client.manhattan.kv.impl.Component +import com.twitter.storage.client.manhattan.kv.impl.Component0 +import com.twitter.storage.client.manhattan.kv.impl.KeyDescriptor +import com.twitter.storage.client.manhattan.kv.impl.ValueDescriptor +import com.twitter.strato.generated.client.ml.featureStore.McUserCountingOnUserClientColumn +import com.twitter.strato.generated.client.ml.featureStore.onboarding.TimelinesAuthorFeaturesOnUserClientColumn +import com.twitter.timelines.author_features.v1.thriftscala.AuthorFeatures +import com.twitter.conversions.DurationOps._ +import com.twitter.onboarding.relevance.features.thriftscala.MCUserCountingFeatures +import java.lang.{Long => JLong} +import scala.util.Random + +object HydrationSourcesModule extends TwitterModule { + + val readFromManhattan = flag( + "feature_hydration_enable_reading_from_manhattan", + false, + "Whether to read the data from Manhattan or Strato") + + val manhattanAppId = + flag("frs_readonly.appId", "ml_features_athena", "RO App Id used by the RO FRS service") + val manhattanDestName = flag( + "frs_readonly.destName", + "/s/manhattan/athena.native-thrift", + "manhattan Dest Name used by the RO FRS service") + + @Provides + @Singleton + def providesAthenaManhattanClient( + serviceIdentifier: ServiceIdentifier + ): ManhattanKVEndpoint = { + val client = ManhattanKVClient( + manhattanAppId(), + manhattanDestName(), + ManhattanKVClientMtlsParams(serviceIdentifier) + ) + ManhattanKVEndpointBuilder(client) + .defaultGuarantee(Guarantee.Weak) + .build() + } + + val manhattanAuthorDataset = "timelines_author_features" + private val defaultCacheMaxKeys = 60000 + private val cacheTTL = 12.hours + private val earlyExpiration = 0.2 + + val authorKeyDesc = KeyDescriptor(Component(LongInjection), Component0) + val authorDatasetKey = authorKeyDesc.withDataset(manhattanAuthorDataset) + val authorValDesc = ValueDescriptor(BinaryCompactScalaInjection(AuthorFeatures)) + + @Provides + @Singleton + def timelinesAuthorStitchCache( + manhattanReadOnlyEndpoint: ManhattanKVEndpoint, + timelinesAuthorFeaturesColumn: TimelinesAuthorFeaturesOnUserClientColumn, + stats: StatsReceiver + ): StitchCache[JLong, Option[AuthorFeatures]] = { + + val stitchCacheStats = + stats + .scope("direct_ds_source_feature_hydration_module").scope("timelines_author") + + val stStat = stitchCacheStats.counter("readFromStrato-each") + val mhtStat = stitchCacheStats.counter("readFromManhattan-each") + + val timelinesAuthorUnderlyingCall = if (readFromManhattan()) { + stitchCacheStats.counter("readFromManhattan").incr() + val authorCacheUnderlyingManhattanCall: JLong => Stitch[Option[AuthorFeatures]] = id => { + mhtStat.incr() + val key = authorDatasetKey.withPkey(id) + manhattanReadOnlyEndpoint + .get(key = key, valueDesc = authorValDesc).map(_.map(value => + clearUnsedFieldsForAuthorFeature(value.contents))) + } + authorCacheUnderlyingManhattanCall + } else { + stitchCacheStats.counter("readFromStrato").incr() + val authorCacheUnderlyingStratoCall: JLong => Stitch[Option[AuthorFeatures]] = id => { + stStat.incr() + val timelinesAuthorFeaturesFetcher = timelinesAuthorFeaturesColumn.fetcher + timelinesAuthorFeaturesFetcher + .fetch(id).map(result => result.v.map(clearUnsedFieldsForAuthorFeature)) + } + authorCacheUnderlyingStratoCall + } + + StitchCache[JLong, Option[AuthorFeatures]]( + underlyingCall = timelinesAuthorUnderlyingCall, + maxCacheSize = defaultCacheMaxKeys, + ttl = randomizedTTL(cacheTTL.inSeconds).seconds, + statsReceiver = stitchCacheStats + ) + + } + + // Not adding manhattan since it didn't seem useful for Author Data, we can add in another phab + // if deemed helpful + @Provides + @Singleton + def metricCenterUserCountingStitchCache( + mcUserCountingFeaturesColumn: McUserCountingOnUserClientColumn, + stats: StatsReceiver + ): StitchCache[JLong, Option[MCUserCountingFeatures]] = { + + val stitchCacheStats = + stats + .scope("direct_ds_source_feature_hydration_module").scope("mc_user_counting") + + val stStat = stitchCacheStats.counter("readFromStrato-each") + stitchCacheStats.counter("readFromStrato").incr() + + val mcUserCountingCacheUnderlyingCall: JLong => Stitch[Option[MCUserCountingFeatures]] = id => { + stStat.incr() + val mcUserCountingFeaturesFetcher = mcUserCountingFeaturesColumn.fetcher + mcUserCountingFeaturesFetcher.fetch(id).map(_.v) + } + + StitchCache[JLong, Option[MCUserCountingFeatures]]( + underlyingCall = mcUserCountingCacheUnderlyingCall, + maxCacheSize = defaultCacheMaxKeys, + ttl = randomizedTTL(cacheTTL.inSeconds).seconds, + statsReceiver = stitchCacheStats + ) + + } + + // clear out fields we don't need to save cache space + private def clearUnsedFieldsForAuthorFeature(entry: AuthorFeatures): AuthorFeatures = { + entry.unsetUserTopics.unsetUserHealth.unsetAuthorCountryCodeAggregates.unsetOriginalAuthorCountryCodeAggregates + } + + // To avoid a cache stampede. See https://en.wikipedia.org/wiki/Cache_stampede + private def randomizedTTL(ttl: Long): Long = { + (ttl - ttl * earlyExpiration * Random.nextDouble()).toLong + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/sources/PreFetchedFeatureSource.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/sources/PreFetchedFeatureSource.scala new file mode 100644 index 0000000000..51975c4870 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/sources/PreFetchedFeatureSource.scala @@ -0,0 +1,36 @@ +package com.twitter.follow_recommendations.common.feature_hydration.sources + +import com.google.inject.Inject +import com.google.inject.Provides +import com.google.inject.Singleton +import com.twitter.follow_recommendations.common.feature_hydration.adapters.PreFetchedFeatureAdapter +import com.twitter.follow_recommendations.common.feature_hydration.common.FeatureSource +import com.twitter.follow_recommendations.common.feature_hydration.common.FeatureSourceId +import com.twitter.follow_recommendations.common.feature_hydration.common.HasPreFetchedFeature +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.follow_recommendations.common.models.HasDisplayLocation +import com.twitter.follow_recommendations.common.models.HasSimilarToContext +import com.twitter.ml.api.DataRecord +import com.twitter.ml.api.FeatureContext +import com.twitter.product_mixer.core.model.marshalling.request.HasClientContext +import com.twitter.stitch.Stitch +import com.twitter.timelines.configapi.HasParams + +@Provides +@Singleton +class PreFetchedFeatureSource @Inject() () extends FeatureSource { + override def id: FeatureSourceId = FeatureSourceId.PreFetchedFeatureSourceId + override def featureContext: FeatureContext = PreFetchedFeatureAdapter.getFeatureContext + override def hydrateFeatures( + target: HasClientContext + with HasPreFetchedFeature + with HasParams + with HasSimilarToContext + with HasDisplayLocation, + candidates: Seq[CandidateUser] + ): Stitch[Map[CandidateUser, DataRecord]] = { + Stitch.value(candidates.map { candidate => + candidate -> PreFetchedFeatureAdapter.adaptToDataRecord((target, candidate)) + }.toMap) + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/sources/UserScoringFeatureSource.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/sources/UserScoringFeatureSource.scala new file mode 100644 index 0000000000..155d9e4425 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/sources/UserScoringFeatureSource.scala @@ -0,0 +1,86 @@ +package com.twitter.follow_recommendations.common.feature_hydration.sources + +import com.google.inject.Inject +import com.google.inject.Provides +import com.google.inject.Singleton +import com.twitter.follow_recommendations.common.feature_hydration.common.FeatureSource +import com.twitter.follow_recommendations.common.feature_hydration.common.FeatureSourceId +import com.twitter.follow_recommendations.common.feature_hydration.common.HasPreFetchedFeature +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.follow_recommendations.common.models.HasDisplayLocation +import com.twitter.follow_recommendations.common.models.HasSimilarToContext +import com.twitter.ml.api.DataRecord +import com.twitter.ml.api.DataRecordMerger +import com.twitter.ml.api.FeatureContext +import com.twitter.product_mixer.core.model.marshalling.request.HasClientContext +import com.twitter.stitch.Stitch +import com.twitter.timelines.configapi.HasParams + +/** + * This source wraps around the separate sources that we hydrate features from + * @param featureStoreSource gets features that require a RPC call to feature store + * @param stratoFeatureHydrationSource gets features that require a RPC call to strato columns + * @param clientContextSource gets features that are already present in the request context + * @param candidateAlgorithmSource gets features that are already present from candidate generation + * @param preFetchedFeatureSource gets features that were prehydrated (shared in request lifecycle) + */ +@Provides +@Singleton +class UserScoringFeatureSource @Inject() ( + featureStoreSource: FeatureStoreSource, + featureStoreGizmoduckSource: FeatureStoreGizmoduckSource, + featureStorePostNuxAlgorithmSource: FeatureStorePostNuxAlgorithmSource, + featureStoreTimelinesAuthorSource: FeatureStoreTimelinesAuthorSource, + featureStoreUserMetricCountsSource: FeatureStoreUserMetricCountsSource, + clientContextSource: ClientContextSource, + candidateAlgorithmSource: CandidateAlgorithmSource, + preFetchedFeatureSource: PreFetchedFeatureSource) + extends FeatureSource { + + override val id: FeatureSourceId = FeatureSourceId.UserScoringFeatureSourceId + + override val featureContext: FeatureContext = FeatureContext.merge( + featureStoreSource.featureContext, + featureStoreGizmoduckSource.featureContext, + featureStorePostNuxAlgorithmSource.featureContext, + featureStoreTimelinesAuthorSource.featureContext, + featureStoreUserMetricCountsSource.featureContext, + clientContextSource.featureContext, + candidateAlgorithmSource.featureContext, + preFetchedFeatureSource.featureContext, + ) + + val sources = + Seq( + featureStoreSource, + featureStorePostNuxAlgorithmSource, + featureStoreTimelinesAuthorSource, + featureStoreUserMetricCountsSource, + featureStoreGizmoduckSource, + clientContextSource, + candidateAlgorithmSource, + preFetchedFeatureSource + ) + + val dataRecordMerger = new DataRecordMerger + + def hydrateFeatures( + target: HasClientContext + with HasPreFetchedFeature + with HasParams + with HasSimilarToContext + with HasDisplayLocation, + candidates: Seq[CandidateUser] + ): Stitch[Map[CandidateUser, DataRecord]] = { + Stitch.collect(sources.map(_.hydrateFeatures(target, candidates))).map { featureMaps => + (for { + candidate <- candidates + } yield { + val combinedDataRecord = new DataRecord + featureMaps + .flatMap(_.get(candidate).toSeq).foreach(dataRecordMerger.merge(combinedDataRecord, _)) + candidate -> combinedDataRecord + }).toMap + } + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/sources/Utils.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/sources/Utils.scala new file mode 100644 index 0000000000..99bd71310c --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/sources/Utils.scala @@ -0,0 +1,30 @@ +package com.twitter.follow_recommendations.common.feature_hydration.sources + +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.ml.api.DataRecord +import com.twitter.ml.api.IRecordOneToOneAdapter +import scala.util.Random + +/** + * Helper functions for FeatureStoreSource operations in FRS are available here. + */ +object Utils { + + private val EarlyExpiration = 0.2 + + private[common] def adaptAdditionalFeaturesToDataRecord( + record: DataRecord, + adapterStats: StatsReceiver, + featureAdapters: Seq[IRecordOneToOneAdapter[DataRecord]] + ): DataRecord = { + featureAdapters.foldRight(record) { (adapter, record) => + adapterStats.counter(adapter.getClass.getSimpleName).incr() + adapter.adaptToDataRecord(record) + } + } + + // To avoid a cache stampede. See https://en.wikipedia.org/wiki/Cache_stampede + private[common] def randomizedTTL(ttl: Long): Long = { + (ttl - ttl * EarlyExpiration * Random.nextDouble()).toLong + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/features/BUILD b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/features/BUILD new file mode 100644 index 0000000000..b5ece498a5 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/features/BUILD @@ -0,0 +1,9 @@ +scala_library( + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline", + ], +) diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/features/LocationFeature.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/features/LocationFeature.scala new file mode 100644 index 0000000000..5325c6b560 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/features/LocationFeature.scala @@ -0,0 +1,10 @@ +package com.twitter.follow_recommendations.common.features + +import com.twitter.follow_recommendations.common.models.GeohashAndCountryCode +import com.twitter.product_mixer.core.feature.FeatureWithDefaultOnFailure +import com.twitter.product_mixer.core.pipeline.PipelineQuery + +case object LocationFeature + extends FeatureWithDefaultOnFailure[PipelineQuery, Option[GeohashAndCountryCode]] { + override val defaultValue: Option[GeohashAndCountryCode] = None +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/features/TrackingTokenFeature.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/features/TrackingTokenFeature.scala new file mode 100644 index 0000000000..23571c8df7 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/features/TrackingTokenFeature.scala @@ -0,0 +1,8 @@ +package com.twitter.follow_recommendations.common.features + +import com.twitter.product_mixer.core.feature.FeatureWithDefaultOnFailure +import com.twitter.product_mixer.core.pipeline.PipelineQuery + +case object TrackingTokenFeature extends FeatureWithDefaultOnFailure[PipelineQuery, Option[Int]] { + override val defaultValue: Option[Int] = None +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/features/UserStateFeature.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/features/UserStateFeature.scala new file mode 100644 index 0000000000..73072b295f --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/features/UserStateFeature.scala @@ -0,0 +1,7 @@ +package com.twitter.follow_recommendations.common.features + +import com.twitter.core_workflows.user_model.thriftscala.UserState +import com.twitter.product_mixer.core.feature.Feature +import com.twitter.product_mixer.core.pipeline.PipelineQuery + +case object UserStateFeature extends Feature[PipelineQuery, Option[UserState]] {} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/AddressBookMetadata.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/AddressBookMetadata.scala new file mode 100644 index 0000000000..303417d23a --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/AddressBookMetadata.scala @@ -0,0 +1,29 @@ +package com.twitter.follow_recommendations.common.models + +import com.twitter.hermit.model.Algorithm +import com.twitter.product_mixer.core.model.common.identifier.CandidateSourceIdentifier + +/** + * contains information if a candidate is from a candidate source generated using the following signals. + */ +case class AddressBookMetadata( + inForwardPhoneBook: Boolean, + inReversePhoneBook: Boolean, + inForwardEmailBook: Boolean, + inReverseEmailBook: Boolean) + +object AddressBookMetadata { + + val ForwardPhoneBookCandidateSource = CandidateSourceIdentifier( + Algorithm.ForwardPhoneBook.toString) + + val ReversePhoneBookCandidateSource = CandidateSourceIdentifier( + Algorithm.ReversePhoneBook.toString) + + val ForwardEmailBookCandidateSource = CandidateSourceIdentifier( + Algorithm.ForwardEmailBook.toString) + + val ReverseEmailBookCandidateSource = CandidateSourceIdentifier( + Algorithm.ReverseEmailBookIbis.toString) + +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/AlgorithmType.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/AlgorithmType.scala new file mode 100644 index 0000000000..b60afb8b37 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/AlgorithmType.scala @@ -0,0 +1,20 @@ +package com.twitter.follow_recommendations.common.models + +/** + * Each candidate source algorithm could be based on one, or more, of the 4 general type of + * information we have on a user: + * 1. Social: the user's connections in Twitter's social graph. + * 2. Geo: the user's geographical information. + * 3. Interest: information on the user's chosen interests. + * 4. Activity: information on the user's past activity. + * + * Note that an algorithm can fall under more than one of these categories. + */ +object AlgorithmType extends Enumeration { + type AlgorithmType = Value + + val Social: Value = Value("social") + val Geo: Value = Value("geo") + val Activity: Value = Value("activity") + val Interest: Value = Value("interest") +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/BUILD b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/BUILD new file mode 100644 index 0000000000..c4916b6d03 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/BUILD @@ -0,0 +1,29 @@ +scala_library( + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "configapi/configapi-core/src/main/scala/com/twitter/timelines/configapi", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/common", + "follow-recommendations-service/thrift/src/main/thrift:thrift-scala", + "hermit/hermit-core/src/main/scala/com/twitter/hermit/constants", + "hermit/hermit-core/src/main/scala/com/twitter/hermit/model", + "hermit/hermit-ml/src/main/scala/com/twitter/hermit/ml/models", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/candidate_source", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/common", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/common/identifier", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/request", + "product-mixer/core/src/main/thrift/com/twitter/product_mixer/core:thrift-scala", + "scrooge/scrooge-serializer/src/main/scala", + "src/java/com/twitter/ml/api:api-base", + "src/scala/com/twitter/ml/api/util", + "src/scala/com/twitter/wtf/scalding/jobs/strong_tie_prediction", + "src/thrift/com/twitter/ads/adserver:adserver_rpc-scala", + "src/thrift/com/twitter/timelines/author_features/user_health:thrift-scala", + "user-signal-service/thrift/src/main/thrift:thrift-scala", + "util/util-slf4j-api/src/main/scala/com/twitter/util/logging", + ], + exports = [ + "util/util-slf4j-api/src/main/scala/com/twitter/util/logging", + ], +) diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/CandidateUser.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/CandidateUser.scala new file mode 100644 index 0000000000..178f34b303 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/CandidateUser.scala @@ -0,0 +1,192 @@ +package com.twitter.follow_recommendations.common.models + +import com.twitter.follow_recommendations.logging.{thriftscala => offline} +import com.twitter.follow_recommendations.{thriftscala => t} +import com.twitter.hermit.constants.AlgorithmFeedbackTokens +import com.twitter.ml.api.thriftscala.{DataRecord => TDataRecord} +import com.twitter.ml.api.util.ScalaToJavaDataRecordConversions +import com.twitter.timelines.configapi.HasParams +import com.twitter.timelines.configapi.Params +import com.twitter.product_mixer.core.model.common.UniversalNoun +import com.twitter.product_mixer.core.model.common.identifier.CandidateSourceIdentifier + +trait FollowableEntity extends UniversalNoun[Long] + +trait Recommendation + extends FollowableEntity + with HasReason + with HasAdMetadata + with HasTrackingToken { + val score: Option[Double] + + def toThrift: t.Recommendation + + def toOfflineThrift: offline.OfflineRecommendation +} + +case class CandidateUser( + override val id: Long, + override val score: Option[Double] = None, + override val reason: Option[Reason] = None, + override val userCandidateSourceDetails: Option[UserCandidateSourceDetails] = None, + override val adMetadata: Option[AdMetadata] = None, + override val trackingToken: Option[TrackingToken] = None, + override val dataRecord: Option[RichDataRecord] = None, + override val scores: Option[Scores] = None, + override val infoPerRankingStage: Option[scala.collection.Map[String, RankingInfo]] = None, + override val params: Params = Params.Invalid, + override val engagements: Seq[EngagementType] = Nil, + override val recommendationFlowIdentifier: Option[String] = None) + extends Recommendation + with HasUserCandidateSourceDetails + with HasDataRecord + with HasScores + with HasParams + with HasEngagements + with HasRecommendationFlowIdentifier + with HasInfoPerRankingStage { + + val rankerIdsStr: Option[Seq[String]] = { + val strs = scores.map(_.scores.flatMap(_.rankerId.map(_.toString))) + if (strs.exists(_.nonEmpty)) strs else None + } + + val thriftDataRecord: Option[TDataRecord] = for { + richDataRecord <- dataRecord + dr <- richDataRecord.dataRecord + } yield { + ScalaToJavaDataRecordConversions.javaDataRecord2ScalaDataRecord(dr) + } + + val toOfflineUserThrift: offline.OfflineUserRecommendation = { + val scoringDetails = + if (userCandidateSourceDetails.isEmpty && score.isEmpty && thriftDataRecord.isEmpty) { + None + } else { + Some( + offline.ScoringDetails( + candidateSourceDetails = userCandidateSourceDetails.map(_.toOfflineThrift), + score = score, + dataRecord = thriftDataRecord, + rankerIds = rankerIdsStr, + infoPerRankingStage = infoPerRankingStage.map(_.mapValues(_.toOfflineThrift)) + ) + ) + } + offline + .OfflineUserRecommendation( + id, + reason.map(_.toOfflineThrift), + adMetadata.map(_.adImpression), + trackingToken.map(_.toOfflineThrift), + scoringDetails = scoringDetails + ) + } + + override val toOfflineThrift: offline.OfflineRecommendation = + offline.OfflineRecommendation.User(toOfflineUserThrift) + + val toUserThrift: t.UserRecommendation = { + val scoringDetails = + if (userCandidateSourceDetails.isEmpty && score.isEmpty && thriftDataRecord.isEmpty && scores.isEmpty) { + None + } else { + Some( + t.ScoringDetails( + candidateSourceDetails = userCandidateSourceDetails.map(_.toThrift), + score = score, + dataRecord = thriftDataRecord, + rankerIds = rankerIdsStr, + debugDataRecord = dataRecord.flatMap(_.debugDataRecord), + infoPerRankingStage = infoPerRankingStage.map(_.mapValues(_.toThrift)) + ) + ) + } + t.UserRecommendation( + userId = id, + reason = reason.map(_.toThrift), + adImpression = adMetadata.map(_.adImpression), + trackingInfo = trackingToken.map(TrackingToken.serialize), + scoringDetails = scoringDetails, + recommendationFlowIdentifier = recommendationFlowIdentifier + ) + } + + override val toThrift: t.Recommendation = + t.Recommendation.User(toUserThrift) + + def setFollowProof(followProofOpt: Option[FollowProof]): CandidateUser = { + this.copy( + reason = reason + .map { reason => + reason.copy( + accountProof = reason.accountProof + .map { accountProof => + accountProof.copy(followProof = followProofOpt) + }.orElse(Some(AccountProof(followProof = followProofOpt))) + ) + }.orElse(Some(Reason(Some(AccountProof(followProof = followProofOpt))))) + ) + } + + def addScore(score: Score): CandidateUser = { + val newScores = scores match { + case Some(existingScores) => existingScores.copy(scores = existingScores.scores :+ score) + case None => Scores(Seq(score)) + } + this.copy(scores = Some(newScores)) + } +} + +object CandidateUser { + val DefaultCandidateScore = 1.0 + + // for converting candidate in ScoringUserRequest + def fromUserRecommendation(candidate: t.UserRecommendation): CandidateUser = { + // we only use the primary candidate source for now + val userCandidateSourceDetails = for { + scoringDetails <- candidate.scoringDetails + candidateSourceDetails <- scoringDetails.candidateSourceDetails + } yield UserCandidateSourceDetails( + primaryCandidateSource = candidateSourceDetails.primarySource + .flatMap(AlgorithmFeedbackTokens.TokenToAlgorithmMap.get).map { algo => + CandidateSourceIdentifier(algo.toString) + }, + candidateSourceScores = fromThriftScoreMap(candidateSourceDetails.candidateSourceScores), + candidateSourceRanks = fromThriftRankMap(candidateSourceDetails.candidateSourceRanks), + addressBookMetadata = None + ) + CandidateUser( + id = candidate.userId, + score = candidate.scoringDetails.flatMap(_.score), + reason = candidate.reason.map(Reason.fromThrift), + userCandidateSourceDetails = userCandidateSourceDetails, + trackingToken = candidate.trackingInfo.map(TrackingToken.deserialize), + recommendationFlowIdentifier = candidate.recommendationFlowIdentifier, + infoPerRankingStage = candidate.scoringDetails.flatMap( + _.infoPerRankingStage.map(_.mapValues(RankingInfo.fromThrift))) + ) + } + + def fromThriftScoreMap( + thriftMapOpt: Option[scala.collection.Map[String, Double]] + ): Map[CandidateSourceIdentifier, Option[Double]] = { + (for { + thriftMap <- thriftMapOpt.toSeq + (algoName, score) <- thriftMap.toSeq + } yield { + CandidateSourceIdentifier(algoName) -> Some(score) + }).toMap + } + + def fromThriftRankMap( + thriftMapOpt: Option[scala.collection.Map[String, Int]] + ): Map[CandidateSourceIdentifier, Int] = { + (for { + thriftMap <- thriftMapOpt.toSeq + (algoName, rank) <- thriftMap.toSeq + } yield { + CandidateSourceIdentifier(algoName) -> rank + }).toMap + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/ClientContextConverter.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/ClientContextConverter.scala new file mode 100644 index 0000000000..ac601371c5 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/ClientContextConverter.scala @@ -0,0 +1,53 @@ +package com.twitter.follow_recommendations.common.models + +import com.twitter.follow_recommendations.logging.{thriftscala => offline} +import com.twitter.follow_recommendations.{thriftscala => frs} +import com.twitter.product_mixer.core.model.marshalling.request.ClientContext + +object ClientContextConverter { + def toFRSOfflineClientContextThrift( + productMixerClientContext: ClientContext + ): offline.OfflineClientContext = + offline.OfflineClientContext( + productMixerClientContext.userId, + productMixerClientContext.guestId, + productMixerClientContext.appId, + productMixerClientContext.countryCode, + productMixerClientContext.languageCode, + productMixerClientContext.guestIdAds, + productMixerClientContext.guestIdMarketing + ) + + def fromThrift(clientContext: frs.ClientContext): ClientContext = ClientContext( + userId = clientContext.userId, + guestId = clientContext.guestId, + appId = clientContext.appId, + ipAddress = clientContext.ipAddress, + userAgent = clientContext.userAgent, + countryCode = clientContext.countryCode, + languageCode = clientContext.languageCode, + isTwoffice = clientContext.isTwoffice, + userRoles = clientContext.userRoles.map(_.toSet), + deviceId = clientContext.deviceId, + guestIdAds = clientContext.guestIdAds, + guestIdMarketing = clientContext.guestIdMarketing, + mobileDeviceId = None, + mobileDeviceAdId = None, + limitAdTracking = None + ) + + def toThrift(clientContext: ClientContext): frs.ClientContext = frs.ClientContext( + userId = clientContext.userId, + guestId = clientContext.guestIdAds, + appId = clientContext.appId, + ipAddress = clientContext.ipAddress, + userAgent = clientContext.userAgent, + countryCode = clientContext.countryCode, + languageCode = clientContext.languageCode, + isTwoffice = clientContext.isTwoffice, + userRoles = clientContext.userRoles, + deviceId = clientContext.deviceId, + guestIdAds = clientContext.guestIdAds, + guestIdMarketing = clientContext.guestIdMarketing + ) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/DisplayLocation.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/DisplayLocation.scala new file mode 100644 index 0000000000..b49baf034d --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/DisplayLocation.scala @@ -0,0 +1,420 @@ +package com.twitter.follow_recommendations.common.models + +import com.twitter.adserver.thriftscala.{DisplayLocation => AdDisplayLocation} +import com.twitter.follow_recommendations.logging.thriftscala.{ + OfflineDisplayLocation => TOfflineDisplayLocation +} +import com.twitter.follow_recommendations.thriftscala.{DisplayLocation => TDisplayLocation} + +sealed trait DisplayLocation { + def toThrift: TDisplayLocation + + def toOfflineThrift: TOfflineDisplayLocation + + def toFsName: String + + // corresponding display location in adserver if available + // make sure to be consistent with the definition here + def toAdDisplayLocation: Option[AdDisplayLocation] = None +} + +/** + * Make sure you add the new DL to the following files and redeploy our attribution jobs + * - follow-recommendations-service/thrift/src/main/thrift/display_location.thrift + * - follow-recommendations-service/thrift/src/main/thrift/logging/display_location.thrift + * - follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/DisplayLocation.scala + */ + +object DisplayLocation { + + case object ProfileSidebar extends DisplayLocation { + override val toThrift: TDisplayLocation = TDisplayLocation.ProfileSidebar + override val toOfflineThrift: TOfflineDisplayLocation = TOfflineDisplayLocation.ProfileSidebar + override val toFsName: String = "ProfileSidebar" + + override val toAdDisplayLocation: Option[AdDisplayLocation] = Some( + AdDisplayLocation.ProfileAccountsSidebar + ) + } + + case object HomeTimeline extends DisplayLocation { + override val toThrift: TDisplayLocation = TDisplayLocation.HomeTimeline + override val toOfflineThrift: TOfflineDisplayLocation = TOfflineDisplayLocation.HomeTimeline + override val toFsName: String = "HomeTimeline" + override val toAdDisplayLocation: Option[AdDisplayLocation] = Some( + // it is based on the logic that HTL DL should correspond to Sidebar: + AdDisplayLocation.WtfSidebar + ) + } + + case object ReactiveFollow extends DisplayLocation { + override val toThrift: TDisplayLocation = TDisplayLocation.ReactiveFollow + override val toOfflineThrift: TOfflineDisplayLocation = TOfflineDisplayLocation.ReactiveFollow + override val toFsName: String = "ReactiveFollow" + } + + case object ExploreTab extends DisplayLocation { + override val toThrift: TDisplayLocation = TDisplayLocation.ExploreTab + override val toOfflineThrift: TOfflineDisplayLocation = TOfflineDisplayLocation.ExploreTab + override val toFsName: String = "ExploreTab" + } + + case object MagicRecs extends DisplayLocation { + override val toThrift: TDisplayLocation = TDisplayLocation.MagicRecs + override val toOfflineThrift: TOfflineDisplayLocation = TOfflineDisplayLocation.MagicRecs + override val toFsName: String = "MagicRecs" + } + + case object AbUploadInjection extends DisplayLocation { + override val toThrift: TDisplayLocation = TDisplayLocation.AbUploadInjection + override val toOfflineThrift: TOfflineDisplayLocation = + TOfflineDisplayLocation.AbUploadInjection + override val toFsName: String = "AbUploadInjection" + } + + case object RuxLandingPage extends DisplayLocation { + override val toThrift: TDisplayLocation = TDisplayLocation.RuxLandingPage + override val toOfflineThrift: TOfflineDisplayLocation = TOfflineDisplayLocation.RuxLandingPage + override val toFsName: String = "RuxLandingPage" + } + + case object ProfileBonusFollow extends DisplayLocation { + override val toThrift: TDisplayLocation = TDisplayLocation.ProfileBonusFollow + override val toOfflineThrift: TOfflineDisplayLocation = + TOfflineDisplayLocation.ProfileBonusFollow + override val toFsName: String = "ProfileBonusFollow" + } + + case object ElectionExploreWtf extends DisplayLocation { + override val toThrift: TDisplayLocation = TDisplayLocation.ElectionExploreWtf + override val toOfflineThrift: TOfflineDisplayLocation = + TOfflineDisplayLocation.ElectionExploreWtf + override val toFsName: String = "ElectionExploreWtf" + } + + case object ClusterFollow extends DisplayLocation { + override val toThrift: TDisplayLocation = TDisplayLocation.ClusterFollow + override val toOfflineThrift: TOfflineDisplayLocation = + TOfflineDisplayLocation.ClusterFollow + override val toFsName: String = "ClusterFollow" + override val toAdDisplayLocation: Option[AdDisplayLocation] = Some( + AdDisplayLocation.ClusterFollow + ) + } + + case object HtlBonusFollow extends DisplayLocation { + override val toThrift: TDisplayLocation = TDisplayLocation.HtlBonusFollow + override val toOfflineThrift: TOfflineDisplayLocation = + TOfflineDisplayLocation.HtlBonusFollow + override val toFsName: String = "HtlBonusFollow" + } + + case object TopicLandingPageHeader extends DisplayLocation { + override val toThrift: TDisplayLocation = TDisplayLocation.TopicLandingPageHeader + override val toOfflineThrift: TOfflineDisplayLocation = + TOfflineDisplayLocation.TopicLandingPageHeader + override val toFsName: String = "TopicLandingPageHeader" + } + + case object NewUserSarusBackfill extends DisplayLocation { + override val toThrift: TDisplayLocation = TDisplayLocation.NewUserSarusBackfill + override val toOfflineThrift: TOfflineDisplayLocation = + TOfflineDisplayLocation.NewUserSarusBackfill + override val toFsName: String = "NewUserSarusBackfill" + } + + case object NuxPymk extends DisplayLocation { + override val toThrift: TDisplayLocation = TDisplayLocation.NuxPymk + override val toOfflineThrift: TOfflineDisplayLocation = + TOfflineDisplayLocation.NuxPymk + override val toFsName: String = "NuxPymk" + } + + case object NuxInterests extends DisplayLocation { + override val toThrift: TDisplayLocation = TDisplayLocation.NuxInterests + override val toOfflineThrift: TOfflineDisplayLocation = + TOfflineDisplayLocation.NuxInterests + override val toFsName: String = "NuxInterests" + } + + case object NuxTopicBonusFollow extends DisplayLocation { + override val toThrift: TDisplayLocation = TDisplayLocation.NuxTopicBonusFollow + override val toOfflineThrift: TOfflineDisplayLocation = + TOfflineDisplayLocation.NuxTopicBonusFollow + override val toFsName: String = "NuxTopicBonusFollow" + } + + case object Sidebar extends DisplayLocation { + override val toThrift: TDisplayLocation = TDisplayLocation.Sidebar + override val toOfflineThrift: TOfflineDisplayLocation = TOfflineDisplayLocation.Sidebar + override val toFsName: String = "Sidebar" + + override val toAdDisplayLocation: Option[AdDisplayLocation] = Some( + AdDisplayLocation.WtfSidebar + ) + } + + case object CampaignForm extends DisplayLocation { + override val toThrift: TDisplayLocation = TDisplayLocation.CampaignForm + override val toOfflineThrift: TOfflineDisplayLocation = TOfflineDisplayLocation.CampaignForm + override val toFsName: String = "CampaignForm" + } + + case object ProfileTopFollowers extends DisplayLocation { + override val toThrift: TDisplayLocation = TDisplayLocation.ProfileTopFollowers + override val toOfflineThrift: TOfflineDisplayLocation = + TOfflineDisplayLocation.ProfileTopFollowers + override val toFsName: String = "ProfileTopFollowers" + } + + case object ProfileTopFollowing extends DisplayLocation { + override val toThrift: TDisplayLocation = TDisplayLocation.ProfileTopFollowing + override val toOfflineThrift: TOfflineDisplayLocation = + TOfflineDisplayLocation.ProfileTopFollowing + override val toFsName: String = "ProfileTopFollowing" + } + + case object RuxPymk extends DisplayLocation { + override val toThrift: TDisplayLocation = TDisplayLocation.RuxPymk + override val toOfflineThrift: TOfflineDisplayLocation = + TOfflineDisplayLocation.RuxPymk + override val toFsName: String = "RuxPymk" + } + + case object IndiaCovid19CuratedAccountsWtf extends DisplayLocation { + override val toThrift: TDisplayLocation = TDisplayLocation.IndiaCovid19CuratedAccountsWtf + override val toOfflineThrift: TOfflineDisplayLocation = + TOfflineDisplayLocation.IndiaCovid19CuratedAccountsWtf + override val toFsName: String = "IndiaCovid19CuratedAccountsWtf" + } + + case object PeoplePlusPlus extends DisplayLocation { + override val toThrift: TDisplayLocation = TDisplayLocation.PeoplePlusPlus + override val toOfflineThrift: TOfflineDisplayLocation = + TOfflineDisplayLocation.PeoplePlusPlus + override val toFsName: String = "PeoplePlusPlus" + } + + case object TweetNotificationRecs extends DisplayLocation { + override val toThrift: TDisplayLocation = TDisplayLocation.TweetNotificationRecs + override val toOfflineThrift: TOfflineDisplayLocation = + TOfflineDisplayLocation.TweetNotificationRecs + override val toFsName: String = "TweetNotificationRecs" + } + + case object ProfileDeviceFollow extends DisplayLocation { + override val toThrift: TDisplayLocation = TDisplayLocation.ProfileDeviceFollow + override val toOfflineThrift: TOfflineDisplayLocation = + TOfflineDisplayLocation.ProfileDeviceFollow + override val toFsName: String = "ProfileDeviceFollow" + } + + case object RecosBackfill extends DisplayLocation { + override val toThrift: TDisplayLocation = TDisplayLocation.RecosBackfill + override val toOfflineThrift: TOfflineDisplayLocation = + TOfflineDisplayLocation.RecosBackfill + override val toFsName: String = "RecosBackfill" + } + + case object HtlSpaceHosts extends DisplayLocation { + override val toThrift: TDisplayLocation = TDisplayLocation.HtlSpaceHosts + override val toOfflineThrift: TOfflineDisplayLocation = + TOfflineDisplayLocation.HtlSpaceHosts + override val toFsName: String = "HtlSpaceHosts" + } + + case object PostNuxFollowTask extends DisplayLocation { + override val toThrift: TDisplayLocation = TDisplayLocation.PostNuxFollowTask + override val toOfflineThrift: TOfflineDisplayLocation = + TOfflineDisplayLocation.PostNuxFollowTask + override val toFsName: String = "PostNuxFollowTask" + } + + case object TopicLandingPage extends DisplayLocation { + override val toThrift: TDisplayLocation = TDisplayLocation.TopicLandingPage + override val toOfflineThrift: TOfflineDisplayLocation = + TOfflineDisplayLocation.TopicLandingPage + override val toFsName: String = "TopicLandingPage" + } + + case object UserTypeaheadPrefetch extends DisplayLocation { + override val toThrift: TDisplayLocation = TDisplayLocation.UserTypeaheadPrefetch + override val toOfflineThrift: TOfflineDisplayLocation = + TOfflineDisplayLocation.UserTypeaheadPrefetch + override val toFsName: String = "UserTypeaheadPrefetch" + } + + case object HomeTimelineRelatableAccounts extends DisplayLocation { + override val toThrift: TDisplayLocation = TDisplayLocation.HomeTimelineRelatableAccounts + override val toOfflineThrift: TOfflineDisplayLocation = + TOfflineDisplayLocation.HomeTimelineRelatableAccounts + override val toFsName: String = "HomeTimelineRelatableAccounts" + } + + case object NuxGeoCategory extends DisplayLocation { + override val toThrift: TDisplayLocation = TDisplayLocation.NuxGeoCategory + override val toOfflineThrift: TOfflineDisplayLocation = + TOfflineDisplayLocation.NuxGeoCategory + override val toFsName: String = "NuxGeoCategory" + } + + case object NuxInterestsCategory extends DisplayLocation { + override val toThrift: TDisplayLocation = TDisplayLocation.NuxInterestsCategory + override val toOfflineThrift: TOfflineDisplayLocation = + TOfflineDisplayLocation.NuxInterestsCategory + override val toFsName: String = "NuxInterestsCategory" + } + + case object TopArticles extends DisplayLocation { + override val toThrift: TDisplayLocation = TDisplayLocation.TopArticles + override val toOfflineThrift: TOfflineDisplayLocation = + TOfflineDisplayLocation.TopArticles + override val toFsName: String = "TopArticles" + } + + case object NuxPymkCategory extends DisplayLocation { + override val toThrift: TDisplayLocation = TDisplayLocation.NuxPymkCategory + override val toOfflineThrift: TOfflineDisplayLocation = + TOfflineDisplayLocation.NuxPymkCategory + override val toFsName: String = "NuxPymkCategory" + } + + case object HomeTimelineTweetRecs extends DisplayLocation { + override val toThrift: TDisplayLocation = TDisplayLocation.HomeTimelineTweetRecs + override val toOfflineThrift: TOfflineDisplayLocation = + TOfflineDisplayLocation.HomeTimelineTweetRecs + override val toFsName: String = "HomeTimelineTweetRecs" + } + + case object HtlBulkFriendFollows extends DisplayLocation { + override val toThrift: TDisplayLocation = TDisplayLocation.HtlBulkFriendFollows + override val toOfflineThrift: TOfflineDisplayLocation = + TOfflineDisplayLocation.HtlBulkFriendFollows + override val toFsName: String = "HtlBulkFriendFollows" + } + + case object NuxAutoFollow extends DisplayLocation { + override val toThrift: TDisplayLocation = TDisplayLocation.NuxAutoFollow + override val toOfflineThrift: TOfflineDisplayLocation = + TOfflineDisplayLocation.NuxAutoFollow + override val toFsName: String = "NuxAutoFollow" + } + + case object SearchBonusFollow extends DisplayLocation { + override val toThrift: TDisplayLocation = TDisplayLocation.SearchBonusFollow + override val toOfflineThrift: TOfflineDisplayLocation = + TOfflineDisplayLocation.SearchBonusFollow + override val toFsName: String = "SearchBonusFollow" + } + + case object ContentRecommender extends DisplayLocation { + override val toThrift: TDisplayLocation = TDisplayLocation.ContentRecommender + override val toOfflineThrift: TOfflineDisplayLocation = + TOfflineDisplayLocation.ContentRecommender + override val toFsName: String = "ContentRecommender" + } + + case object HomeTimelineReverseChron extends DisplayLocation { + override val toThrift: TDisplayLocation = TDisplayLocation.HomeTimelineReverseChron + override val toOfflineThrift: TOfflineDisplayLocation = + TOfflineDisplayLocation.HomeTimelineReverseChron + override val toFsName: String = "HomeTimelineReverseChron" + } + + def fromThrift(displayLocation: TDisplayLocation): DisplayLocation = displayLocation match { + case TDisplayLocation.ProfileSidebar => ProfileSidebar + case TDisplayLocation.HomeTimeline => HomeTimeline + case TDisplayLocation.MagicRecs => MagicRecs + case TDisplayLocation.AbUploadInjection => AbUploadInjection + case TDisplayLocation.RuxLandingPage => RuxLandingPage + case TDisplayLocation.ProfileBonusFollow => ProfileBonusFollow + case TDisplayLocation.ElectionExploreWtf => ElectionExploreWtf + case TDisplayLocation.ClusterFollow => ClusterFollow + case TDisplayLocation.HtlBonusFollow => HtlBonusFollow + case TDisplayLocation.ReactiveFollow => ReactiveFollow + case TDisplayLocation.TopicLandingPageHeader => TopicLandingPageHeader + case TDisplayLocation.NewUserSarusBackfill => NewUserSarusBackfill + case TDisplayLocation.NuxPymk => NuxPymk + case TDisplayLocation.NuxInterests => NuxInterests + case TDisplayLocation.NuxTopicBonusFollow => NuxTopicBonusFollow + case TDisplayLocation.ExploreTab => ExploreTab + case TDisplayLocation.Sidebar => Sidebar + case TDisplayLocation.CampaignForm => CampaignForm + case TDisplayLocation.ProfileTopFollowers => ProfileTopFollowers + case TDisplayLocation.ProfileTopFollowing => ProfileTopFollowing + case TDisplayLocation.RuxPymk => RuxPymk + case TDisplayLocation.IndiaCovid19CuratedAccountsWtf => IndiaCovid19CuratedAccountsWtf + case TDisplayLocation.PeoplePlusPlus => PeoplePlusPlus + case TDisplayLocation.TweetNotificationRecs => TweetNotificationRecs + case TDisplayLocation.ProfileDeviceFollow => ProfileDeviceFollow + case TDisplayLocation.RecosBackfill => RecosBackfill + case TDisplayLocation.HtlSpaceHosts => HtlSpaceHosts + case TDisplayLocation.PostNuxFollowTask => PostNuxFollowTask + case TDisplayLocation.TopicLandingPage => TopicLandingPage + case TDisplayLocation.UserTypeaheadPrefetch => UserTypeaheadPrefetch + case TDisplayLocation.HomeTimelineRelatableAccounts => HomeTimelineRelatableAccounts + case TDisplayLocation.NuxGeoCategory => NuxGeoCategory + case TDisplayLocation.NuxInterestsCategory => NuxInterestsCategory + case TDisplayLocation.TopArticles => TopArticles + case TDisplayLocation.NuxPymkCategory => NuxPymkCategory + case TDisplayLocation.HomeTimelineTweetRecs => HomeTimelineTweetRecs + case TDisplayLocation.HtlBulkFriendFollows => HtlBulkFriendFollows + case TDisplayLocation.NuxAutoFollow => NuxAutoFollow + case TDisplayLocation.SearchBonusFollow => SearchBonusFollow + case TDisplayLocation.ContentRecommender => ContentRecommender + case TDisplayLocation.HomeTimelineReverseChron => HomeTimelineReverseChron + case TDisplayLocation.EnumUnknownDisplayLocation(i) => + throw new UnknownDisplayLocationException( + s"Unknown display location thrift enum with value: ${i}") + } + + def fromOfflineThrift(displayLocation: TOfflineDisplayLocation): DisplayLocation = + displayLocation match { + case TOfflineDisplayLocation.ProfileSidebar => ProfileSidebar + case TOfflineDisplayLocation.HomeTimeline => HomeTimeline + case TOfflineDisplayLocation.MagicRecs => MagicRecs + case TOfflineDisplayLocation.AbUploadInjection => AbUploadInjection + case TOfflineDisplayLocation.RuxLandingPage => RuxLandingPage + case TOfflineDisplayLocation.ProfileBonusFollow => ProfileBonusFollow + case TOfflineDisplayLocation.ElectionExploreWtf => ElectionExploreWtf + case TOfflineDisplayLocation.ClusterFollow => ClusterFollow + case TOfflineDisplayLocation.HtlBonusFollow => HtlBonusFollow + case TOfflineDisplayLocation.TopicLandingPageHeader => TopicLandingPageHeader + case TOfflineDisplayLocation.NewUserSarusBackfill => NewUserSarusBackfill + case TOfflineDisplayLocation.NuxPymk => NuxPymk + case TOfflineDisplayLocation.NuxInterests => NuxInterests + case TOfflineDisplayLocation.NuxTopicBonusFollow => NuxTopicBonusFollow + case TOfflineDisplayLocation.ExploreTab => ExploreTab + case TOfflineDisplayLocation.ReactiveFollow => ReactiveFollow + case TOfflineDisplayLocation.Sidebar => Sidebar + case TOfflineDisplayLocation.CampaignForm => CampaignForm + case TOfflineDisplayLocation.ProfileTopFollowers => ProfileTopFollowers + case TOfflineDisplayLocation.ProfileTopFollowing => ProfileTopFollowing + case TOfflineDisplayLocation.RuxPymk => RuxPymk + case TOfflineDisplayLocation.IndiaCovid19CuratedAccountsWtf => IndiaCovid19CuratedAccountsWtf + case TOfflineDisplayLocation.PeoplePlusPlus => PeoplePlusPlus + case TOfflineDisplayLocation.TweetNotificationRecs => TweetNotificationRecs + case TOfflineDisplayLocation.ProfileDeviceFollow => ProfileDeviceFollow + case TOfflineDisplayLocation.RecosBackfill => RecosBackfill + case TOfflineDisplayLocation.HtlSpaceHosts => HtlSpaceHosts + case TOfflineDisplayLocation.PostNuxFollowTask => PostNuxFollowTask + case TOfflineDisplayLocation.TopicLandingPage => TopicLandingPage + case TOfflineDisplayLocation.UserTypeaheadPrefetch => UserTypeaheadPrefetch + case TOfflineDisplayLocation.HomeTimelineRelatableAccounts => HomeTimelineRelatableAccounts + case TOfflineDisplayLocation.NuxGeoCategory => NuxGeoCategory + case TOfflineDisplayLocation.NuxInterestsCategory => NuxInterestsCategory + case TOfflineDisplayLocation.TopArticles => TopArticles + case TOfflineDisplayLocation.NuxPymkCategory => NuxPymkCategory + case TOfflineDisplayLocation.HomeTimelineTweetRecs => HomeTimelineTweetRecs + case TOfflineDisplayLocation.HtlBulkFriendFollows => HtlBulkFriendFollows + case TOfflineDisplayLocation.NuxAutoFollow => NuxAutoFollow + case TOfflineDisplayLocation.SearchBonusFollow => SearchBonusFollow + case TOfflineDisplayLocation.ContentRecommender => ContentRecommender + case TOfflineDisplayLocation.HomeTimelineReverseChron => HomeTimelineReverseChron + case TOfflineDisplayLocation.EnumUnknownOfflineDisplayLocation(i) => + throw new UnknownDisplayLocationException( + s"Unknown offline display location thrift enum with value: ${i}") + } +} + +class UnknownDisplayLocationException(message: String) extends Exception(message) diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/EngagementType.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/EngagementType.scala new file mode 100644 index 0000000000..b12a4404c5 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/EngagementType.scala @@ -0,0 +1,62 @@ +package com.twitter.follow_recommendations.common.models + +import com.twitter.follow_recommendations.thriftscala.{EngagementType => TEngagementType} +import com.twitter.follow_recommendations.logging.thriftscala.{ + EngagementType => OfflineEngagementType +} +sealed trait EngagementType { + def toThrift: TEngagementType + def toOfflineThrift: OfflineEngagementType +} + +object EngagementType { + object Click extends EngagementType { + override val toThrift: TEngagementType = TEngagementType.Click + + override val toOfflineThrift: OfflineEngagementType = OfflineEngagementType.Click + } + object Like extends EngagementType { + override val toThrift: TEngagementType = TEngagementType.Like + + override val toOfflineThrift: OfflineEngagementType = OfflineEngagementType.Like + } + object Mention extends EngagementType { + override val toThrift: TEngagementType = TEngagementType.Mention + + override val toOfflineThrift: OfflineEngagementType = OfflineEngagementType.Mention + } + object Retweet extends EngagementType { + override val toThrift: TEngagementType = TEngagementType.Retweet + + override val toOfflineThrift: OfflineEngagementType = OfflineEngagementType.Retweet + } + object ProfileView extends EngagementType { + override val toThrift: TEngagementType = TEngagementType.ProfileView + + override val toOfflineThrift: OfflineEngagementType = OfflineEngagementType.ProfileView + } + + def fromThrift(engagementType: TEngagementType): EngagementType = engagementType match { + case TEngagementType.Click => Click + case TEngagementType.Like => Like + case TEngagementType.Mention => Mention + case TEngagementType.Retweet => Retweet + case TEngagementType.ProfileView => ProfileView + case TEngagementType.EnumUnknownEngagementType(i) => + throw new UnknownEngagementTypeException( + s"Unknown engagement type thrift enum with value: ${i}") + } + + def fromOfflineThrift(engagementType: OfflineEngagementType): EngagementType = + engagementType match { + case OfflineEngagementType.Click => Click + case OfflineEngagementType.Like => Like + case OfflineEngagementType.Mention => Mention + case OfflineEngagementType.Retweet => Retweet + case OfflineEngagementType.ProfileView => ProfileView + case OfflineEngagementType.EnumUnknownEngagementType(i) => + throw new UnknownEngagementTypeException( + s"Unknown engagement type offline thrift enum with value: ${i}") + } +} +class UnknownEngagementTypeException(message: String) extends Exception(message) diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/FilterReason.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/FilterReason.scala new file mode 100644 index 0000000000..86b4967764 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/FilterReason.scala @@ -0,0 +1,133 @@ +package com.twitter.follow_recommendations.common.models + +sealed trait FilterReason { + def reason: String +} + +object FilterReason { + + case object NoReason extends FilterReason { + override val reason: String = "no_reason" + } + + case class ParamReason(paramName: String) extends FilterReason { + override val reason: String = s"param_$paramName" + } + + case object ExcludedId extends FilterReason { + override val reason: String = "excluded_id_from_request" + } + + case object ProfileSidebarBlacklist extends FilterReason { + override val reason: String = "profile_sidebar_blacklisted_id" + } + + case object CuratedAccountsCompetitorList extends FilterReason { + override val reason: String = "curated_blacklisted_id" + } + + case class InvalidRelationshipTypes(relationshipTypes: String) extends FilterReason { + override val reason: String = s"invalid_relationship_types $relationshipTypes" + } + + case object ProfileId extends FilterReason { + override val reason: String = "candidate_has_same_id_as_profile" + } + + case object DismissedId extends FilterReason { + override val reason: String = s"dismissed_candidate" + } + + case object OptedOutId extends FilterReason { + override val reason: String = s"candidate_opted_out_from_criteria_in_request" + } + + // gizmoduck predicates + case object NoUser extends FilterReason { + override val reason: String = "no_user_result_from_gizmoduck" + } + + case object AddressBookUndiscoverable extends FilterReason { + override val reason: String = "not_discoverable_via_address_book" + } + + case object PhoneBookUndiscoverable extends FilterReason { + override val reason: String = "not_discoverable_via_phone_book" + } + + case object Deactivated extends FilterReason { + override val reason: String = "deactivated" + } + + case object Suspended extends FilterReason { + override val reason: String = "suspended" + } + + case object Restricted extends FilterReason { + override val reason: String = "restricted" + } + + case object NsfwUser extends FilterReason { + override val reason: String = "nsfwUser" + } + + case object NsfwAdmin extends FilterReason { + override val reason: String = "nsfwAdmin" + } + + case object HssSignal extends FilterReason { + override val reason: String = "hssSignal" + } + + case object IsProtected extends FilterReason { + override val reason: String = "isProtected" + } + + case class CountryTakedown(countryCode: String) extends FilterReason { + override val reason: String = s"takedown_in_$countryCode" + } + + case object Blink extends FilterReason { + override val reason: String = "blink" + } + + case object AlreadyFollowed extends FilterReason { + override val reason: String = "already_followed" + } + + case object InvalidRelationship extends FilterReason { + override val reason: String = "invalid_relationship" + } + + case object NotFollowingTargetUser extends FilterReason { + override val reason: String = "not_following_target_user" + } + + case object CandidateSideHoldback extends FilterReason { + override val reason: String = "candidate_side_holdback" + } + + case object Inactive extends FilterReason { + override val reason: String = "inactive" + } + + case object MissingRecommendabilityData extends FilterReason { + override val reason: String = "missing_recommendability_data" + } + + case object HighTweetVelocity extends FilterReason { + override val reason: String = "high_tweet_velocity" + } + + case object AlreadyRecommended extends FilterReason { + override val reason: String = "already_recommended" + } + + case object MinStateNotMet extends FilterReason { + override val reason: String = "min_state_user_not_met" + } + + case object FailOpen extends FilterReason { + override val reason: String = "fail_open" + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/FlowContext.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/FlowContext.scala new file mode 100644 index 0000000000..15e36321e7 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/FlowContext.scala @@ -0,0 +1,20 @@ +package com.twitter.follow_recommendations.common.models + +import com.twitter.follow_recommendations.logging.{thriftscala => offline} +import com.twitter.follow_recommendations.{thriftscala => t} + +case class FlowContext(steps: Seq[RecommendationStep]) { + + def toThrift: t.FlowContext = t.FlowContext(steps = steps.map(_.toThrift)) + + def toOfflineThrift: offline.OfflineFlowContext = + offline.OfflineFlowContext(steps = steps.map(_.toOfflineThrift)) +} + +object FlowContext { + + def fromThrift(flowContext: t.FlowContext): FlowContext = { + FlowContext(steps = flowContext.steps.map(RecommendationStep.fromThrift)) + } + +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/FlowRecommendation.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/FlowRecommendation.scala new file mode 100644 index 0000000000..118ff258d4 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/FlowRecommendation.scala @@ -0,0 +1,23 @@ +package com.twitter.follow_recommendations.common.models + +import com.twitter.follow_recommendations.logging.{thriftscala => offline} +import com.twitter.follow_recommendations.{thriftscala => t} + +case class FlowRecommendation(userId: Long) { + + def toThrift: t.FlowRecommendation = + t.FlowRecommendation(userId = userId) + + def toOfflineThrift: offline.OfflineFlowRecommendation = + offline.OfflineFlowRecommendation(userId = userId) + +} + +object FlowRecommendation { + def fromThrift(flowRecommendation: t.FlowRecommendation): FlowRecommendation = { + FlowRecommendation( + userId = flowRecommendation.userId + ) + } + +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/GeohashAndCountryCode.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/GeohashAndCountryCode.scala new file mode 100644 index 0000000000..782d3fc9e9 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/GeohashAndCountryCode.scala @@ -0,0 +1,3 @@ +package com.twitter.follow_recommendations.common.models + +case class GeohashAndCountryCode(geohash: Option[String], countryCode: Option[String]) diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasAdMetadata.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasAdMetadata.scala new file mode 100644 index 0000000000..57979e376c --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasAdMetadata.scala @@ -0,0 +1,23 @@ +package com.twitter.follow_recommendations.common.models + +import com.twitter.adserver.{thriftscala => t} + +case class AdMetadata( + insertPosition: Int, + // use original ad impression info to avoid losing data in domain model translations + adImpression: t.AdImpression) + +trait HasAdMetadata { + + def adMetadata: Option[AdMetadata] + + def adImpression: Option[t.AdImpression] = { + adMetadata.map(_.adImpression) + } + + def insertPosition: Option[Int] = { + adMetadata.map(_.insertPosition) + } + + def isPromotedAccount: Boolean = adMetadata.isDefined +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasByfSeedUserIds.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasByfSeedUserIds.scala new file mode 100644 index 0000000000..d4cbdcee83 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasByfSeedUserIds.scala @@ -0,0 +1,5 @@ +package com.twitter.follow_recommendations.common.models + +trait HasByfSeedUserIds { + def byfSeedUserIds: Option[Seq[Long]] +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasDataRecord.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasDataRecord.scala new file mode 100644 index 0000000000..4e7047b4e9 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasDataRecord.scala @@ -0,0 +1,86 @@ +package com.twitter.follow_recommendations.common.models + +import com.twitter.follow_recommendations.thriftscala.DebugDataRecord +import com.twitter.ml.api.DataRecord +import com.twitter.ml.api.FeatureContext +import com.twitter.util.Try +import com.twitter.util.logging.Logging +import scala.collection.convert.ImplicitConversions._ + +// contains the standard dataRecord struct, and the debug version if required +case class RichDataRecord( + dataRecord: Option[DataRecord] = None, + debugDataRecord: Option[DebugDataRecord] = None, +) + +trait HasDataRecord extends Logging { + def dataRecord: Option[RichDataRecord] + + def toDebugDataRecord(dr: DataRecord, featureContext: FeatureContext): DebugDataRecord = { + + val binaryFeatures: Option[Set[String]] = if (dr.isSetBinaryFeatures) { + Some(dr.getBinaryFeatures.flatMap { id => + Try(featureContext.getFeature(id).getFeatureName).toOption + }.toSet) + } else None + + val continuousFeatures: Option[Map[String, Double]] = if (dr.isSetContinuousFeatures) { + Some(dr.getContinuousFeatures.flatMap { + case (id, value) => + Try(featureContext.getFeature(id).getFeatureName).toOption.map { id => + id -> value.toDouble + } + }.toMap) + } else None + + val discreteFeatures: Option[Map[String, Long]] = if (dr.isSetDiscreteFeatures) { + Some(dr.getDiscreteFeatures.flatMap { + case (id, value) => + Try(featureContext.getFeature(id).getFeatureName).toOption.map { id => + id -> value.toLong + } + }.toMap) + } else None + + val stringFeatures: Option[Map[String, String]] = if (dr.isSetStringFeatures) { + Some(dr.getStringFeatures.flatMap { + case (id, value) => + Try(featureContext.getFeature(id).getFeatureName).toOption.map { id => + id -> value + } + }.toMap) + } else None + + val sparseBinaryFeatures: Option[Map[String, Set[String]]] = if (dr.isSetSparseBinaryFeatures) { + Some(dr.getSparseBinaryFeatures.flatMap { + case (id, values) => + Try(featureContext.getFeature(id).getFeatureName).toOption.map { id => + id -> values.toSet + } + }.toMap) + } else None + + val sparseContinuousFeatures: Option[Map[String, Map[String, Double]]] = + if (dr.isSetSparseContinuousFeatures) { + Some(dr.getSparseContinuousFeatures.flatMap { + case (id, values) => + Try(featureContext.getFeature(id).getFeatureName).toOption.map { id => + id -> values.map { + case (str, value) => + str -> value.toDouble + }.toMap + } + }.toMap) + } else None + + DebugDataRecord( + binaryFeatures = binaryFeatures, + continuousFeatures = continuousFeatures, + discreteFeatures = discreteFeatures, + stringFeatures = stringFeatures, + sparseBinaryFeatures = sparseBinaryFeatures, + sparseContinuousFeatures = sparseContinuousFeatures, + ) + } + +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasDebugOptions.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasDebugOptions.scala new file mode 100644 index 0000000000..0956ca34e1 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasDebugOptions.scala @@ -0,0 +1,30 @@ +package com.twitter.follow_recommendations.common.models + +import com.twitter.follow_recommendations.thriftscala.DebugParams + +case class DebugOptions( + randomizationSeed: Option[Long] = None, + fetchDebugInfo: Boolean = false, + doNotLog: Boolean = false) + +object DebugOptions { + def fromDebugParamsThrift(debugParams: DebugParams): DebugOptions = { + DebugOptions( + debugParams.randomizationSeed, + debugParams.includeDebugInfoInResults.getOrElse(false), + debugParams.doNotLog.getOrElse(false) + ) + } +} + +trait HasDebugOptions { + def debugOptions: Option[DebugOptions] + + def getRandomizationSeed: Option[Long] = debugOptions.flatMap(_.randomizationSeed) + + def fetchDebugInfo: Option[Boolean] = debugOptions.map(_.fetchDebugInfo) +} + +trait HasFrsDebugOptions { + def frsDebugOptions: Option[DebugOptions] +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasDismissedUserIds.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasDismissedUserIds.scala new file mode 100644 index 0000000000..3f21549923 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasDismissedUserIds.scala @@ -0,0 +1,6 @@ +package com.twitter.follow_recommendations.common.models + +trait HasDismissedUserIds { + // user ids that are recently followed by the target user + def dismissedUserIds: Option[Seq[Long]] +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasDisplayLocation.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasDisplayLocation.scala new file mode 100644 index 0000000000..e74ae83e16 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasDisplayLocation.scala @@ -0,0 +1,5 @@ +package com.twitter.follow_recommendations.common.models + +trait HasDisplayLocation { + def displayLocation: DisplayLocation +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasEngagements.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasEngagements.scala new file mode 100644 index 0000000000..de59e4479b --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasEngagements.scala @@ -0,0 +1,7 @@ +package com.twitter.follow_recommendations.common.models + +trait HasEngagements { + + def engagements: Seq[EngagementType] + +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasExcludedUserIds.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasExcludedUserIds.scala new file mode 100644 index 0000000000..3addcef83e --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasExcludedUserIds.scala @@ -0,0 +1,6 @@ +package com.twitter.follow_recommendations.common.models + +trait HasExcludedUserIds { + // user ids that are going to be excluded from recommendations + def excludedUserIds: Seq[Long] +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasGeohashAndCountryCode.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasGeohashAndCountryCode.scala new file mode 100644 index 0000000000..a4364bbf4c --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasGeohashAndCountryCode.scala @@ -0,0 +1,5 @@ +package com.twitter.follow_recommendations.common.models + +trait HasGeohashAndCountryCode { + def geohashAndCountryCode: Option[GeohashAndCountryCode] +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasInfoPerRankingStage.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasInfoPerRankingStage.scala new file mode 100644 index 0000000000..c4d2774127 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasInfoPerRankingStage.scala @@ -0,0 +1,5 @@ +package com.twitter.follow_recommendations.common.models + +trait HasInfoPerRankingStage { + def infoPerRankingStage: Option[scala.collection.Map[String, RankingInfo]] +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasInterestIds.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasInterestIds.scala new file mode 100644 index 0000000000..69f97b673f --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasInterestIds.scala @@ -0,0 +1,11 @@ +package com.twitter.follow_recommendations.common.models + +trait HasCustomInterests { + def customInterests: Option[Seq[String]] +} + +trait HasUttInterests { + def uttInterestIds: Option[Seq[Long]] +} + +trait HasInterestIds extends HasCustomInterests with HasUttInterests {} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasInvalidRelationshipUserIds.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasInvalidRelationshipUserIds.scala new file mode 100644 index 0000000000..3cf3f66dbe --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasInvalidRelationshipUserIds.scala @@ -0,0 +1,6 @@ +package com.twitter.follow_recommendations.common.models + +trait HasInvalidRelationshipUserIds { + // user ids that have invalid relationship with the target user + def invalidRelationshipUserIds: Option[Set[Long]] +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasIsSoftUser.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasIsSoftUser.scala new file mode 100644 index 0000000000..8cf1532cee --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasIsSoftUser.scala @@ -0,0 +1,5 @@ +package com.twitter.follow_recommendations.common.models + +trait HasIsSoftUser { + def isSoftUser: Boolean +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasMutualFollowedUserIds.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasMutualFollowedUserIds.scala new file mode 100644 index 0000000000..c5e1e16cc2 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasMutualFollowedUserIds.scala @@ -0,0 +1,10 @@ +package com.twitter.follow_recommendations.common.models + +// intersection of recent followers and followed by +trait HasMutualFollowedUserIds extends HasRecentFollowedUserIds with HasRecentFollowedByUserIds { + + lazy val recentMutualFollows: Seq[Long] = + recentFollowedUserIds.getOrElse(Nil).intersect(recentFollowedByUserIds.getOrElse(Nil)) + + lazy val numRecentMutualFollows: Int = recentMutualFollows.size +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasPreviousRecommendationsContext.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasPreviousRecommendationsContext.scala new file mode 100644 index 0000000000..2480faaadb --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasPreviousRecommendationsContext.scala @@ -0,0 +1,12 @@ +package com.twitter.follow_recommendations.common.models + +trait HasPreviousRecommendationsContext { + + def previouslyRecommendedUserIDs: Set[Long] + + def previouslyFollowedUserIds: Set[Long] + + def skippedFollows: Set[Long] = { + previouslyRecommendedUserIDs.diff(previouslyFollowedUserIds) + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasProfileId.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasProfileId.scala new file mode 100644 index 0000000000..10cd7c02f5 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasProfileId.scala @@ -0,0 +1,5 @@ +package com.twitter.follow_recommendations.common.models + +trait HasProfileId { + def profileId: Long +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasQualityFactor.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasQualityFactor.scala new file mode 100644 index 0000000000..96527b2ba5 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasQualityFactor.scala @@ -0,0 +1,5 @@ +package com.twitter.follow_recommendations.common.models + +trait HasQualityFactor { + def qualityFactor: Option[Double] +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasRecentFollowedByUserIds.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasRecentFollowedByUserIds.scala new file mode 100644 index 0000000000..bc15e8bd82 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasRecentFollowedByUserIds.scala @@ -0,0 +1,8 @@ +package com.twitter.follow_recommendations.common.models + +trait HasRecentFollowedByUserIds { + // user ids that have recently followed the target user; target user has been "followed by" them. + def recentFollowedByUserIds: Option[Seq[Long]] + + lazy val numRecentFollowedByUserIds: Int = recentFollowedByUserIds.map(_.size).getOrElse(0) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasRecentFollowedUserIds.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasRecentFollowedUserIds.scala new file mode 100644 index 0000000000..67ada7c668 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasRecentFollowedUserIds.scala @@ -0,0 +1,14 @@ +package com.twitter.follow_recommendations.common.models + +trait HasRecentFollowedUserIds { + // user ids that are recently followed by the target user + def recentFollowedUserIds: Option[Seq[Long]] + + // user ids that are recently followed by the target user in set data-structure + lazy val recentFollowedUserIdsSet: Option[Set[Long]] = recentFollowedUserIds match { + case Some(users) => Some(users.toSet) + case None => Some(Set.empty) + } + + lazy val numRecentFollowedUserIds: Int = recentFollowedUserIds.map(_.size).getOrElse(0) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasRecentFollowedUserIdsWithTime.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasRecentFollowedUserIdsWithTime.scala new file mode 100644 index 0000000000..7e3cba4a70 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasRecentFollowedUserIdsWithTime.scala @@ -0,0 +1,9 @@ +package com.twitter.follow_recommendations.common.models + +trait HasRecentFollowedUserIdsWithTime { + // user ids that are recently followed by the target user + def recentFollowedUserIdsWithTime: Option[Seq[UserIdWithTimestamp]] + + lazy val numRecentFollowedUserIdsWithTime: Int = + recentFollowedUserIdsWithTime.map(_.size).getOrElse(0) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasRecentlyEngagedUserIds.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasRecentlyEngagedUserIds.scala new file mode 100644 index 0000000000..44420ec274 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasRecentlyEngagedUserIds.scala @@ -0,0 +1,5 @@ +package com.twitter.follow_recommendations.common.models + +trait HasRecentlyEngagedUserIds { + val recentlyEngagedUserIds: Option[Seq[RecentlyEngagedUserId]] +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasRecommendationFlowIdentifier.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasRecommendationFlowIdentifier.scala new file mode 100644 index 0000000000..5706c7a295 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasRecommendationFlowIdentifier.scala @@ -0,0 +1,5 @@ +package com.twitter.follow_recommendations.common.models + +trait HasRecommendationFlowIdentifier { + def recommendationFlowIdentifier: Option[String] +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasScores.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasScores.scala new file mode 100644 index 0000000000..e8a6698ee7 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasScores.scala @@ -0,0 +1,5 @@ +package com.twitter.follow_recommendations.common.models + +trait HasScores { + def scores: Option[Scores] +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasSimilarToContext.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasSimilarToContext.scala new file mode 100644 index 0000000000..bbe2ac2586 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasSimilarToContext.scala @@ -0,0 +1,7 @@ +package com.twitter.follow_recommendations.common.models + +trait HasSimilarToContext { + + // user ids that are used to generate similar to recommendations + def similarToUserIds: Seq[Long] +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasTopicId.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasTopicId.scala new file mode 100644 index 0000000000..4bd6e63e77 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasTopicId.scala @@ -0,0 +1,5 @@ +package com.twitter.follow_recommendations.common.models + +trait HasTopicId { + def topicId: Long +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasUserCandidateSourceDetails.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasUserCandidateSourceDetails.scala new file mode 100644 index 0000000000..e0e363449e --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasUserCandidateSourceDetails.scala @@ -0,0 +1,162 @@ +package com.twitter.follow_recommendations.common.models + +import com.twitter.hermit.ml.models.Feature +import com.twitter.hermit.model.Algorithm +import com.twitter.hermit.model.Algorithm.Algorithm +import com.twitter.product_mixer.core.model.common.identifier.CandidateSourceIdentifier + +/** + * Used to keep track of a candidate's source not so much as a feature but for filtering candidate + * from specific sources (eg. GizmoduckPredicate) + */ +trait HasUserCandidateSourceDetails { candidateUser: CandidateUser => + def userCandidateSourceDetails: Option[UserCandidateSourceDetails] + + def getAlgorithm: Algorithm = { + val algorithm = for { + details <- userCandidateSourceDetails + identifier <- details.primaryCandidateSource + algorithm <- Algorithm.withNameOpt(identifier.name) + } yield algorithm + + algorithm.getOrElse(throw new Exception("Algorithm missing on candidate user!")) + } + + def getAllAlgorithms: Seq[Algorithm] = { + getCandidateSources.keys + .flatMap(identifier => Algorithm.withNameOpt(identifier.name)).toSeq + } + + def getAddressBookMetadata: Option[AddressBookMetadata] = { + userCandidateSourceDetails.flatMap(_.addressBookMetadata) + } + + def getCandidateSources: Map[CandidateSourceIdentifier, Option[Double]] = { + userCandidateSourceDetails.map(_.candidateSourceScores).getOrElse(Map.empty) + } + + def getCandidateRanks: Map[CandidateSourceIdentifier, Int] = { + userCandidateSourceDetails.map(_.candidateSourceRanks).getOrElse(Map.empty) + } + + def getCandidateFeatures: Map[CandidateSourceIdentifier, Seq[Feature]] = { + userCandidateSourceDetails.map(_.candidateSourceFeatures).getOrElse(Map.empty) + } + + def getPrimaryCandidateSource: Option[CandidateSourceIdentifier] = { + userCandidateSourceDetails.flatMap(_.primaryCandidateSource) + } + + def withCandidateSource(source: CandidateSourceIdentifier): CandidateUser = { + withCandidateSourceAndScore(source, candidateUser.score) + } + + def withCandidateSourceAndScore( + source: CandidateSourceIdentifier, + score: Option[Double] + ): CandidateUser = { + withCandidateSourceScoreAndFeatures(source, score, Nil) + } + + def withCandidateSourceAndFeatures( + source: CandidateSourceIdentifier, + features: Seq[Feature] + ): CandidateUser = { + withCandidateSourceScoreAndFeatures(source, candidateUser.score, features) + } + + def withCandidateSourceScoreAndFeatures( + source: CandidateSourceIdentifier, + score: Option[Double], + features: Seq[Feature] + ): CandidateUser = { + val candidateSourceDetails = + candidateUser.userCandidateSourceDetails + .map { details => + details.copy( + primaryCandidateSource = Some(source), + candidateSourceScores = details.candidateSourceScores + (source -> score), + candidateSourceFeatures = details.candidateSourceFeatures + (source -> features) + ) + }.getOrElse( + UserCandidateSourceDetails( + Some(source), + Map(source -> score), + Map.empty, + None, + Map(source -> features))) + candidateUser.copy( + userCandidateSourceDetails = Some(candidateSourceDetails) + ) + } + + def addCandidateSourceScoresMap( + scoreMap: Map[CandidateSourceIdentifier, Option[Double]] + ): CandidateUser = { + val candidateSourceDetails = candidateUser.userCandidateSourceDetails + .map { details => + details.copy(candidateSourceScores = details.candidateSourceScores ++ scoreMap) + }.getOrElse(UserCandidateSourceDetails(scoreMap.keys.headOption, scoreMap, Map.empty, None)) + candidateUser.copy( + userCandidateSourceDetails = Some(candidateSourceDetails) + ) + } + + def addCandidateSourceRanksMap( + rankMap: Map[CandidateSourceIdentifier, Int] + ): CandidateUser = { + val candidateSourceDetails = candidateUser.userCandidateSourceDetails + .map { details => + details.copy(candidateSourceRanks = details.candidateSourceRanks ++ rankMap) + }.getOrElse(UserCandidateSourceDetails(rankMap.keys.headOption, Map.empty, rankMap, None)) + candidateUser.copy( + userCandidateSourceDetails = Some(candidateSourceDetails) + ) + } + + def addInfoPerRankingStage( + rankingStage: String, + scores: Option[Scores], + rank: Int + ): CandidateUser = { + val scoresOpt: Option[Scores] = scores.orElse(candidateUser.scores) + val originalInfoPerRankingStage = + candidateUser.infoPerRankingStage.getOrElse(Map[String, RankingInfo]()) + candidateUser.copy( + infoPerRankingStage = + Some(originalInfoPerRankingStage + (rankingStage -> RankingInfo(scoresOpt, Some(rank)))) + ) + } + + def addAddressBookMetadataIfAvailable( + candidateSources: Seq[CandidateSourceIdentifier] + ): CandidateUser = { + + val addressBookMetadata = AddressBookMetadata( + inForwardPhoneBook = + candidateSources.contains(AddressBookMetadata.ForwardPhoneBookCandidateSource), + inReversePhoneBook = + candidateSources.contains(AddressBookMetadata.ReversePhoneBookCandidateSource), + inForwardEmailBook = + candidateSources.contains(AddressBookMetadata.ForwardEmailBookCandidateSource), + inReverseEmailBook = + candidateSources.contains(AddressBookMetadata.ReverseEmailBookCandidateSource) + ) + + val newCandidateSourceDetails = candidateUser.userCandidateSourceDetails + .map { details => + details.copy(addressBookMetadata = Some(addressBookMetadata)) + }.getOrElse( + UserCandidateSourceDetails( + None, + Map.empty, + Map.empty, + Some(addressBookMetadata), + Map.empty)) + + candidateUser.copy( + userCandidateSourceDetails = Some(newCandidateSourceDetails) + ) + } + +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasUserState.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasUserState.scala new file mode 100644 index 0000000000..bf0df46f7f --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasUserState.scala @@ -0,0 +1,7 @@ +package com.twitter.follow_recommendations.common.models + +import com.twitter.core_workflows.user_model.thriftscala.UserState + +trait HasUserState { + def userState: Option[UserState] +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasWtfImpressions.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasWtfImpressions.scala new file mode 100644 index 0000000000..d840ed6fd4 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/HasWtfImpressions.scala @@ -0,0 +1,30 @@ +package com.twitter.follow_recommendations.common.models + +import com.twitter.util.Time + +trait HasWtfImpressions { + + def wtfImpressions: Option[Seq[WtfImpression]] + + lazy val numWtfImpressions: Int = wtfImpressions.map(_.size).getOrElse(0) + + lazy val candidateImpressions: Map[Long, WtfImpression] = wtfImpressions + .map { imprMap => + imprMap.map { i => + i.candidateId -> i + }.toMap + }.getOrElse(Map.empty) + + lazy val latestImpressionTime: Time = { + if (wtfImpressions.exists(_.nonEmpty)) { + wtfImpressions.get.map(_.latestTime).max + } else Time.Top + } + + def getCandidateImpressionCounts(id: Long): Option[Int] = + candidateImpressions.get(id).map(_.counts) + + def getCandidateLatestTime(id: Long): Option[Time] = { + candidateImpressions.get(id).map(_.latestTime) + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/OptimusRequest.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/OptimusRequest.scala new file mode 100644 index 0000000000..df4c228e18 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/OptimusRequest.scala @@ -0,0 +1,15 @@ +package com.twitter.follow_recommendations.common.models + +import com.twitter.product_mixer.core.model.marshalling.request.HasClientContext +import com.twitter.timelines.configapi.HasParams + +/** +Convenience trait to group together all traits needed for optimus ranking + */ +trait OptimusRequest + extends HasParams + with HasClientContext + with HasDisplayLocation + with HasInterestIds + with HasDebugOptions + with HasPreviousRecommendationsContext {} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/Product.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/Product.scala new file mode 100644 index 0000000000..f37cff56e5 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/Product.scala @@ -0,0 +1,15 @@ +package com.twitter.follow_recommendations.common.models + +import com.twitter.product_mixer.core.model.common.identifier.ProductIdentifier +import com.twitter.product_mixer.core.model.marshalling.request.{Product => ProductMixerProduct} + +object Product { + case object MagicRecs extends ProductMixerProduct { + override val identifier: ProductIdentifier = ProductIdentifier("MagicRecs") + override val stringCenterProject: Option[String] = Some("people-discovery") + } + + case object PlaceholderProductMixerProduct extends ProductMixerProduct { + override val identifier: ProductIdentifier = ProductIdentifier("PlaceholderProductMixerProduct") + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/RankingInfo.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/RankingInfo.scala new file mode 100644 index 0000000000..02eb46b5a7 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/RankingInfo.scala @@ -0,0 +1,28 @@ +package com.twitter.follow_recommendations.common.models + +import com.twitter.follow_recommendations.{thriftscala => t} +import com.twitter.follow_recommendations.logging.{thriftscala => offline} + +case class RankingInfo( + scores: Option[Scores], + rank: Option[Int]) { + + def toThrift: t.RankingInfo = { + t.RankingInfo(scores.map(_.toThrift), rank) + } + + def toOfflineThrift: offline.RankingInfo = { + offline.RankingInfo(scores.map(_.toOfflineThrift), rank) + } +} + +object RankingInfo { + + def fromThrift(rankingInfo: t.RankingInfo): RankingInfo = { + RankingInfo( + scores = rankingInfo.scores.map(Scores.fromThrift), + rank = rankingInfo.rank + ) + } + +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/Reason.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/Reason.scala new file mode 100644 index 0000000000..b7c9c6c759 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/Reason.scala @@ -0,0 +1,206 @@ +package com.twitter.follow_recommendations.common.models + +import com.twitter.follow_recommendations.{thriftscala => t} +import com.twitter.follow_recommendations.logging.{thriftscala => offline} + +case class FollowProof(followedBy: Seq[Long], numIds: Int) { + def toThrift: t.FollowProof = { + t.FollowProof(followedBy, numIds) + } + + def toOfflineThrift: offline.FollowProof = offline.FollowProof(followedBy, numIds) +} + +object FollowProof { + + def fromThrift(proof: t.FollowProof): FollowProof = { + FollowProof(proof.userIds, proof.numIds) + } +} + +case class SimilarToProof(similarTo: Seq[Long]) { + def toThrift: t.SimilarToProof = { + t.SimilarToProof(similarTo) + } + + def toOfflineThrift: offline.SimilarToProof = offline.SimilarToProof(similarTo) +} + +object SimilarToProof { + def fromThrift(proof: t.SimilarToProof): SimilarToProof = { + SimilarToProof(proof.userIds) + } +} + +case class PopularInGeoProof(location: String) { + def toThrift: t.PopularInGeoProof = { + t.PopularInGeoProof(location) + } + + def toOfflineThrift: offline.PopularInGeoProof = offline.PopularInGeoProof(location) +} + +object PopularInGeoProof { + + def fromThrift(proof: t.PopularInGeoProof): PopularInGeoProof = { + PopularInGeoProof(proof.location) + } +} + +case class TttInterestProof(interestId: Long, interestDisplayName: String) { + def toThrift: t.TttInterestProof = { + t.TttInterestProof(interestId, interestDisplayName) + } + + def toOfflineThrift: offline.TttInterestProof = + offline.TttInterestProof(interestId, interestDisplayName) +} + +object TttInterestProof { + + def fromThrift(proof: t.TttInterestProof): TttInterestProof = { + TttInterestProof(proof.interestId, proof.interestDisplayName) + } +} + +case class TopicProof(topicId: Long) { + def toThrift: t.TopicProof = { + t.TopicProof(topicId) + } + + def toOfflineThrift: offline.TopicProof = + offline.TopicProof(topicId) +} + +object TopicProof { + def fromThrift(proof: t.TopicProof): TopicProof = { + TopicProof(proof.topicId) + } +} + +case class CustomInterest(query: String) { + def toThrift: t.CustomInterestProof = { + t.CustomInterestProof(query) + } + + def toOfflineThrift: offline.CustomInterestProof = + offline.CustomInterestProof(query) +} + +object CustomInterest { + def fromThrift(proof: t.CustomInterestProof): CustomInterest = { + CustomInterest(proof.query) + } +} + +case class TweetsAuthorProof(tweetIds: Seq[Long]) { + def toThrift: t.TweetsAuthorProof = { + t.TweetsAuthorProof(tweetIds) + } + + def toOfflineThrift: offline.TweetsAuthorProof = + offline.TweetsAuthorProof(tweetIds) +} + +object TweetsAuthorProof { + def fromThrift(proof: t.TweetsAuthorProof): TweetsAuthorProof = { + TweetsAuthorProof(proof.tweetIds) + } +} + +case class DeviceFollowProof(isDeviceFollow: Boolean) { + def toThrift: t.DeviceFollowProof = { + t.DeviceFollowProof(isDeviceFollow) + } + def toOfflineThrift: offline.DeviceFollowProof = + offline.DeviceFollowProof(isDeviceFollow) +} + +object DeviceFollowProof { + def fromThrift(proof: t.DeviceFollowProof): DeviceFollowProof = { + DeviceFollowProof(proof.isDeviceFollow) + } + +} + +case class AccountProof( + followProof: Option[FollowProof] = None, + similarToProof: Option[SimilarToProof] = None, + popularInGeoProof: Option[PopularInGeoProof] = None, + tttInterestProof: Option[TttInterestProof] = None, + topicProof: Option[TopicProof] = None, + customInterestProof: Option[CustomInterest] = None, + tweetsAuthorProof: Option[TweetsAuthorProof] = None, + deviceFollowProof: Option[DeviceFollowProof] = None) { + def toThrift: t.AccountProof = { + t.AccountProof( + followProof.map(_.toThrift), + similarToProof.map(_.toThrift), + popularInGeoProof.map(_.toThrift), + tttInterestProof.map(_.toThrift), + topicProof.map(_.toThrift), + customInterestProof.map(_.toThrift), + tweetsAuthorProof.map(_.toThrift), + deviceFollowProof.map(_.toThrift) + ) + } + + def toOfflineThrift: offline.AccountProof = { + offline.AccountProof( + followProof.map(_.toOfflineThrift), + similarToProof.map(_.toOfflineThrift), + popularInGeoProof.map(_.toOfflineThrift), + tttInterestProof.map(_.toOfflineThrift), + topicProof.map(_.toOfflineThrift), + customInterestProof.map(_.toOfflineThrift), + tweetsAuthorProof.map(_.toOfflineThrift), + deviceFollowProof.map(_.toOfflineThrift) + ) + } +} + +object AccountProof { + def fromThrift(proof: t.AccountProof): AccountProof = { + AccountProof( + proof.followProof.map(FollowProof.fromThrift), + proof.similarToProof.map(SimilarToProof.fromThrift), + proof.popularInGeoProof.map(PopularInGeoProof.fromThrift), + proof.tttInterestProof.map(TttInterestProof.fromThrift), + proof.topicProof.map(TopicProof.fromThrift), + proof.customInterestProof.map(CustomInterest.fromThrift), + proof.tweetsAuthorProof.map(TweetsAuthorProof.fromThrift), + proof.deviceFollowProof.map(DeviceFollowProof.fromThrift) + ) + } +} + +case class Reason(accountProof: Option[AccountProof]) { + def toThrift: t.Reason = { + t.Reason(accountProof.map(_.toThrift)) + } + + def toOfflineThrift: offline.Reason = { + offline.Reason(accountProof.map(_.toOfflineThrift)) + } +} + +object Reason { + + def fromThrift(reason: t.Reason): Reason = { + Reason(reason.accountProof.map(AccountProof.fromThrift)) + } +} + +trait HasReason { + + def reason: Option[Reason] + // helper methods below + + def followedBy: Option[Seq[Long]] = { + for { + reason <- reason + accountProof <- reason.accountProof + followProof <- accountProof.followProof + } yield { followProof.followedBy } + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/RecentlyEngagedUserId.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/RecentlyEngagedUserId.scala new file mode 100644 index 0000000000..b20d5d6fbd --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/RecentlyEngagedUserId.scala @@ -0,0 +1,31 @@ +package com.twitter.follow_recommendations.common.models + +import com.twitter.follow_recommendations.logging.{thriftscala => offline} +import com.twitter.follow_recommendations.{thriftscala => t} + +case class RecentlyEngagedUserId(id: Long, engagementType: EngagementType) { + def toThrift: t.RecentlyEngagedUserId = + t.RecentlyEngagedUserId(id = id, engagementType = engagementType.toThrift) + + def toOfflineThrift: offline.RecentlyEngagedUserId = + offline.RecentlyEngagedUserId(id = id, engagementType = engagementType.toOfflineThrift) +} + +object RecentlyEngagedUserId { + def fromThrift(recentlyEngagedUserId: t.RecentlyEngagedUserId): RecentlyEngagedUserId = { + RecentlyEngagedUserId( + id = recentlyEngagedUserId.id, + engagementType = EngagementType.fromThrift(recentlyEngagedUserId.engagementType) + ) + } + + def fromOfflineThrift( + recentlyEngagedUserId: offline.RecentlyEngagedUserId + ): RecentlyEngagedUserId = { + RecentlyEngagedUserId( + id = recentlyEngagedUserId.id, + engagementType = EngagementType.fromOfflineThrift(recentlyEngagedUserId.engagementType) + ) + } + +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/RecommendationStep.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/RecommendationStep.scala new file mode 100644 index 0000000000..4fd4bc70ed --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/RecommendationStep.scala @@ -0,0 +1,30 @@ +package com.twitter.follow_recommendations.common.models + +import com.twitter.follow_recommendations.{thriftscala => t} +import com.twitter.follow_recommendations.logging.{thriftscala => offline} + +case class RecommendationStep( + recommendations: Seq[FlowRecommendation], + followedUserIds: Set[Long]) { + + def toThrift: t.RecommendationStep = t.RecommendationStep( + recommendations = recommendations.map(_.toThrift), + followedUserIds = followedUserIds + ) + + def toOfflineThrift: offline.OfflineRecommendationStep = + offline.OfflineRecommendationStep( + recommendations = recommendations.map(_.toOfflineThrift), + followedUserIds = followedUserIds) + +} + +object RecommendationStep { + + def fromThrift(recommendationStep: t.RecommendationStep): RecommendationStep = { + RecommendationStep( + recommendations = recommendationStep.recommendations.map(FlowRecommendation.fromThrift), + followedUserIds = recommendationStep.followedUserIds.toSet) + } + +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/STPGraph.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/STPGraph.scala new file mode 100644 index 0000000000..2d577e21d4 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/STPGraph.scala @@ -0,0 +1,22 @@ +package com.twitter.follow_recommendations.common.models + +import com.twitter.hermit.model.Algorithm.Algorithm +import com.twitter.wtf.scalding.jobs.strong_tie_prediction.FirstDegreeEdge +import com.twitter.wtf.scalding.jobs.strong_tie_prediction.FirstDegreeEdgeInfo +import com.twitter.wtf.scalding.jobs.strong_tie_prediction.SecondDegreeEdge + +case class PotentialFirstDegreeEdge( + userId: Long, + connectingId: Long, + algorithm: Algorithm, + score: Double, + edgeInfo: FirstDegreeEdgeInfo) + +case class IntermediateSecondDegreeEdge( + connectingId: Long, + candidateId: Long, + edgeInfo: FirstDegreeEdgeInfo) + +case class STPGraph( + firstDegreeEdgeInfoList: List[FirstDegreeEdge], + secondDegreeEdgeInfoList: List[SecondDegreeEdge]) diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/SafetyLevel.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/SafetyLevel.scala new file mode 100644 index 0000000000..10c21704a7 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/SafetyLevel.scala @@ -0,0 +1,17 @@ +package com.twitter.follow_recommendations.common.models + +import com.twitter.spam.rtf.thriftscala.{SafetyLevel => ThriftSafetyLevel} + +sealed trait SafetyLevel { + def toThrift: ThriftSafetyLevel +} + +object SafetyLevel { + case object Recommendations extends SafetyLevel { + override val toThrift = ThriftSafetyLevel.Recommendations + } + + case object TopicsLandingPageTopicRecommendations extends SafetyLevel { + override val toThrift = ThriftSafetyLevel.TopicsLandingPageTopicRecommendations + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/Score.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/Score.scala new file mode 100644 index 0000000000..1007c58270 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/Score.scala @@ -0,0 +1,144 @@ +package com.twitter.follow_recommendations.common.models + +import com.twitter.follow_recommendations.common.rankers.common.RankerId +import com.twitter.follow_recommendations.common.rankers.common.RankerId.RankerId +import com.twitter.follow_recommendations.logging.{thriftscala => offline} +import com.twitter.follow_recommendations.{thriftscala => t} + +/** + * Type of Score. This is used to differentiate scores. + * + * Define it as a trait so it is possible to add more information for different score types. + */ +sealed trait ScoreType { + def getName: String +} + +/** + * Existing Score Types + */ +object ScoreType { + + /** + * the score is calculated based on heuristics and most likely not normalized + */ + case object HeuristicBasedScore extends ScoreType { + override def getName: String = "HeuristicBasedScore" + } + + /** + * probability of follow after the candidate is recommended to the user + */ + case object PFollowGivenReco extends ScoreType { + override def getName: String = "PFollowGivenReco" + } + + /** + * probability of engage after the user follows the candidate + */ + case object PEngagementGivenFollow extends ScoreType { + override def getName: String = "PEngagementGivenFollow" + } + + /** + * probability of engage per tweet impression + */ + case object PEngagementPerImpression extends ScoreType { + override def getName: String = "PEngagementPerImpression" + } + + /** + * probability of engage per tweet impression + */ + case object PEngagementGivenReco extends ScoreType { + override def getName: String = "PEngagementGivenReco" + } + + def fromScoreTypeString(scoreTypeName: String): ScoreType = scoreTypeName match { + case "HeuristicBasedScore" => HeuristicBasedScore + case "PFollowGivenReco" => PFollowGivenReco + case "PEngagementGivenFollow" => PEngagementGivenFollow + case "PEngagementPerImpression" => PEngagementPerImpression + case "PEngagementGivenReco" => PEngagementGivenReco + } +} + +/** + * Represent the output from a certain ranker or scorer. All the fields are optional + * + * @param value value of the score + * @param rankerId ranker id + * @param scoreType score type + */ +final case class Score( + value: Double, + rankerId: Option[RankerId] = None, + scoreType: Option[ScoreType] = None) { + + def toThrift: t.Score = t.Score( + value = value, + rankerId = rankerId.map(_.toString), + scoreType = scoreType.map(_.getName) + ) + + def toOfflineThrift: offline.Score = + offline.Score( + value = value, + rankerId = rankerId.map(_.toString), + scoreType = scoreType.map(_.getName) + ) +} + +object Score { + + val RandomScore = Score(0.0d, Some(RankerId.RandomRanker)) + + def optimusScore(score: Double, scoreType: ScoreType): Score = { + Score(value = score, scoreType = Some(scoreType)) + } + + def predictionScore(score: Double, rankerId: RankerId): Score = { + Score(value = score, rankerId = Some(rankerId)) + } + + def fromThrift(thriftScore: t.Score): Score = + Score( + value = thriftScore.value, + rankerId = thriftScore.rankerId.flatMap(RankerId.getRankerByName), + scoreType = thriftScore.scoreType.map(ScoreType.fromScoreTypeString) + ) +} + +/** + * a list of scores + */ +final case class Scores( + scores: Seq[Score], + selectedRankerId: Option[RankerId] = None, + isInProducerScoringExperiment: Boolean = false) { + + def toThrift: t.Scores = + t.Scores( + scores = scores.map(_.toThrift), + selectedRankerId = selectedRankerId.map(_.toString), + isInProducerScoringExperiment = isInProducerScoringExperiment + ) + + def toOfflineThrift: offline.Scores = + offline.Scores( + scores = scores.map(_.toOfflineThrift), + selectedRankerId = selectedRankerId.map(_.toString), + isInProducerScoringExperiment = isInProducerScoringExperiment + ) +} + +object Scores { + val Empty: Scores = Scores(Nil) + + def fromThrift(thriftScores: t.Scores): Scores = + Scores( + scores = thriftScores.scores.map(Score.fromThrift), + selectedRankerId = thriftScores.selectedRankerId.flatMap(RankerId.getRankerByName), + isInProducerScoringExperiment = thriftScores.isInProducerScoringExperiment + ) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/Session.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/Session.scala new file mode 100644 index 0000000000..25ff00b488 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/Session.scala @@ -0,0 +1,16 @@ +package com.twitter.follow_recommendations.common.models + +import com.twitter.finagle.tracing.Trace + +object Session { + + /** + * The sessionId in FRS is the finagle trace id which is static within the lifetime of a single + * request. + * + * It is used when generating per-candidate tokens (in TrackingTokenTransform) and is also passed + * in to downstream Optimus ranker requests. + * + */ + def getSessionId: Long = Trace.id.traceId.toLong +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/SignalData.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/SignalData.scala new file mode 100644 index 0000000000..3af5c6e1d8 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/SignalData.scala @@ -0,0 +1,42 @@ +package com.twitter.follow_recommendations.common.models + +import com.twitter.simclusters_v2.thriftscala.InternalId +import com.twitter.usersignalservice.thriftscala.SignalType +import com.twitter.usersignalservice.thriftscala.Signal + +trait SignalData { + val userId: Long + val signalType: SignalType +} + +case class RecentFollowsSignal( + override val userId: Long, + override val signalType: SignalType, + followedUserId: Long, + timestamp: Long) + extends SignalData + +object RecentFollowsSignal { + + def fromUssSignal(targetUserId: Long, signal: Signal): RecentFollowsSignal = { + val InternalId.UserId(followedUserId) = signal.targetInternalId.getOrElse( + throw new IllegalArgumentException("RecentFollow Signal does not have internalId")) + + RecentFollowsSignal( + userId = targetUserId, + followedUserId = followedUserId, + timestamp = signal.timestamp, + signalType = signal.signalType + ) + } + + def getRecentFollowedUserIds( + signalDataMap: Option[Map[SignalType, Seq[SignalData]]] + ): Option[Seq[Long]] = { + signalDataMap.map(_.getOrElse(SignalType.AccountFollow, default = Seq.empty).flatMap { + case RecentFollowsSignal(userId, signalType, followedUserId, timestamp) => + Some(followedUserId) + case _ => None + }) + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/TrackingToken.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/TrackingToken.scala new file mode 100644 index 0000000000..177e08f65b --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/TrackingToken.scala @@ -0,0 +1,62 @@ +package com.twitter.follow_recommendations.common.models + +import com.twitter.finagle.tracing.Trace +import com.twitter.follow_recommendations.logging.{thriftscala => offline} +import com.twitter.follow_recommendations.{thriftscala => t} +import com.twitter.scrooge.BinaryThriftStructSerializer +import com.twitter.suggests.controller_data.thriftscala.ControllerData +import com.twitter.util.Base64StringEncoder + +/** + * used for attribution per target-candidate pair + * @param sessionId trace-id of the finagle request + * @param controllerData 64-bit encoded binary attributes of our recommendation + * @param algorithmId id for identifying a candidate source. maintained for backwards compatibility + */ +case class TrackingToken( + sessionId: Long, + displayLocation: Option[DisplayLocation], + controllerData: Option[ControllerData], + algorithmId: Option[Int]) { + + def toThrift: t.TrackingToken = { + Trace.id.traceId.toLong + t.TrackingToken( + sessionId = sessionId, + displayLocation = displayLocation.map(_.toThrift), + controllerData = controllerData, + algoId = algorithmId + ) + } + + def toOfflineThrift: offline.TrackingToken = { + offline.TrackingToken( + sessionId = sessionId, + displayLocation = displayLocation.map(_.toOfflineThrift), + controllerData = controllerData, + algoId = algorithmId + ) + } +} + +object TrackingToken { + val binaryThriftSerializer = BinaryThriftStructSerializer[t.TrackingToken](t.TrackingToken) + def serialize(trackingToken: TrackingToken): String = { + Base64StringEncoder.encode(binaryThriftSerializer.toBytes(trackingToken.toThrift)) + } + def deserialize(trackingTokenStr: String): TrackingToken = { + fromThrift(binaryThriftSerializer.fromBytes(Base64StringEncoder.decode(trackingTokenStr))) + } + def fromThrift(token: t.TrackingToken): TrackingToken = { + TrackingToken( + sessionId = token.sessionId, + displayLocation = token.displayLocation.map(DisplayLocation.fromThrift), + controllerData = token.controllerData, + algorithmId = token.algoId + ) + } +} + +trait HasTrackingToken { + def trackingToken: Option[TrackingToken] +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/TweetCandidate.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/TweetCandidate.scala new file mode 100644 index 0000000000..1957e28cc1 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/TweetCandidate.scala @@ -0,0 +1,6 @@ +package com.twitter.follow_recommendations.common.models + +case class TweetCandidate( + tweetId: Long, + authorId: Long, + score: Option[Double]) diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/UserCandidateSourceDetails.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/UserCandidateSourceDetails.scala new file mode 100644 index 0000000000..73b7662320 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/UserCandidateSourceDetails.scala @@ -0,0 +1,97 @@ +package com.twitter.follow_recommendations.common.models + +import com.twitter.follow_recommendations.logging.{thriftscala => offline} +import com.twitter.follow_recommendations.{thriftscala => t} +import com.twitter.hermit.constants.AlgorithmFeedbackTokens._ +import com.twitter.hermit.ml.models.Feature +import com.twitter.hermit.model.Algorithm +import com.twitter.product_mixer.core.model.common.identifier.CandidateSourceIdentifier + +/** + * primaryCandidateSource param is showing the candidate source that responsible for generating this + * candidate, as the candidate might have gone through multiple candidate sources to get generated + * (for example if it has generated by a composite source). WeightedCandidateSourceRanker uses this + * field to do the sampling over candidate sources. All the sources used for generating this + * candidate (including the primary source) and their corresponding score exist in the + * candidateSourceScores field. + */ +case class UserCandidateSourceDetails( + primaryCandidateSource: Option[CandidateSourceIdentifier], + candidateSourceScores: Map[CandidateSourceIdentifier, Option[Double]] = Map.empty, + candidateSourceRanks: Map[CandidateSourceIdentifier, Int] = Map.empty, + addressBookMetadata: Option[AddressBookMetadata] = None, + candidateSourceFeatures: Map[CandidateSourceIdentifier, Seq[Feature]] = Map.empty, +) { + + def toThrift: t.CandidateSourceDetails = { + t.CandidateSourceDetails( + candidateSourceScores = Some(candidateSourceScores.map { + case (identifier, score) => + (identifier.name, score.getOrElse(0.0d)) + }), + primarySource = for { + identifier <- primaryCandidateSource + algo <- Algorithm.withNameOpt(identifier.name) + feedbackToken <- AlgorithmToFeedbackTokenMap.get(algo) + } yield feedbackToken + ) + } + + def toOfflineThrift: offline.CandidateSourceDetails = { + offline.CandidateSourceDetails( + candidateSourceScores = Some(candidateSourceScores.map { + case (identifier, score) => + (identifier.name, score.getOrElse(0.0d)) + }), + primarySource = for { + identifier <- primaryCandidateSource + algo <- Algorithm.withNameOpt(identifier.name) + feedbackToken <- AlgorithmToFeedbackTokenMap.get(algo) + } yield feedbackToken + ) + } +} + +object UserCandidateSourceDetails { + val algorithmNameMap: Map[String, Algorithm.Value] = Algorithm.values.map { + algorithmValue: Algorithm.Value => + (algorithmValue.toString, algorithmValue) + }.toMap + + /** + * This method is used to parse the candidate source of the candidates, which is only passed from + * the scoreUserCandidates endpoint. We create custom candidate source identifiers which + * CandidateAlgorithmSource will read from to hydrate the algorithm id feature. + * candidateSourceScores will not be populated from the endpoint, but we add the conversion for + * completeness. Note that the conversion uses the raw string of the Algorithm rather than the + * assigned strings that we give to our own candidate sources in the FRS. + */ + def fromThrift(details: t.CandidateSourceDetails): UserCandidateSourceDetails = { + val primaryCandidateSource: Option[CandidateSourceIdentifier] = for { + primarySourceToken <- details.primarySource + algo <- TokenToAlgorithmMap.get(primarySourceToken) + } yield CandidateSourceIdentifier(algo.toString) + + val candidateSourceScores = for { + scoreMap <- details.candidateSourceScores.toSeq + (name, score) <- scoreMap + algo <- algorithmNameMap.get(name) + } yield { + CandidateSourceIdentifier(algo.toString) -> Some(score) + } + val candidateSourceRanks = for { + rankMap <- details.candidateSourceRanks.toSeq + (name, rank) <- rankMap + algo <- algorithmNameMap.get(name) + } yield { + CandidateSourceIdentifier(algo.toString) -> rank + } + UserCandidateSourceDetails( + primaryCandidateSource = primaryCandidateSource, + candidateSourceScores = candidateSourceScores.toMap, + candidateSourceRanks = candidateSourceRanks.toMap, + addressBookMetadata = None, + candidateSourceFeatures = Map.empty + ) + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/UserIdAndTimestamp.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/UserIdAndTimestamp.scala new file mode 100644 index 0000000000..74f33eb406 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/UserIdAndTimestamp.scala @@ -0,0 +1,3 @@ +package com.twitter.follow_recommendations.common.models + +case class UserIdWithTimestamp(userId: Long, timeInMs: Long) diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/WtfImpression.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/WtfImpression.scala new file mode 100644 index 0000000000..39e0561a23 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/WtfImpression.scala @@ -0,0 +1,12 @@ +package com.twitter.follow_recommendations.common.models + +import com.twitter.util.Time + +/** + * Domain model for representing impressions on wtf recommendations in the past 16 days + */ +case class WtfImpression( + candidateId: Long, + displayLocation: DisplayLocation, + latestTime: Time, + counts: Int) diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/BUILD b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/BUILD new file mode 100644 index 0000000000..ffcbe65a7e --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/BUILD @@ -0,0 +1,21 @@ +scala_library( + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/com/google/inject:guice", + "3rdparty/jvm/com/google/inject/extensions:guice-assistedinject", + "3rdparty/jvm/net/codingwell:scala-guice", + "3rdparty/jvm/org/slf4j:slf4j-api", + "escherbird/src/scala/com/twitter/escherbird/util/stitchcache", + "finatra/inject/inject-core/src/main/scala", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/strato", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/user_state", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/constants", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/features", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/stores", + "util/util-slf4j-api/src/main/scala", + ], +) diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/CandidateParamPredicate.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/CandidateParamPredicate.scala new file mode 100644 index 0000000000..0713f728fe --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/CandidateParamPredicate.scala @@ -0,0 +1,21 @@ +package com.twitter.follow_recommendations.common.predicates + +import com.twitter.follow_recommendations.common.base.Predicate +import com.twitter.follow_recommendations.common.base.PredicateResult +import com.twitter.follow_recommendations.common.models.FilterReason +import com.twitter.stitch.Stitch +import com.twitter.timelines.configapi.HasParams +import com.twitter.timelines.configapi.Param + +class CandidateParamPredicate[A <: HasParams]( + param: Param[Boolean], + reason: FilterReason) + extends Predicate[A] { + override def apply(candidate: A): Stitch[PredicateResult] = { + if (candidate.params(param)) { + Stitch.value(PredicateResult.Valid) + } else { + Stitch.value(PredicateResult.Invalid(Set(reason))) + } + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/CandidateSourceParamPredicate.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/CandidateSourceParamPredicate.scala new file mode 100644 index 0000000000..cf08f5623d --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/CandidateSourceParamPredicate.scala @@ -0,0 +1,31 @@ +package com.twitter.follow_recommendations.common.predicates + +import com.twitter.follow_recommendations.common.base.Predicate +import com.twitter.follow_recommendations.common.base.PredicateResult +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.follow_recommendations.common.models.FilterReason +import com.twitter.product_mixer.core.model.common.identifier.CandidateSourceIdentifier +import com.twitter.stitch.Stitch +import com.twitter.timelines.configapi.Param + +/** + * This predicate allows us to filter candidates given its source. + * To avoid bucket dilution, we only want to evaluate the param (which would implicitly trigger + * bucketing for FSParams) only if the candidate source fn yields true. + * The param provided should be true when we want to keep the candidate and false otherwise. + */ +class CandidateSourceParamPredicate( + val param: Param[Boolean], + val reason: FilterReason, + candidateSources: Set[CandidateSourceIdentifier]) + extends Predicate[CandidateUser] { + override def apply(candidate: CandidateUser): Stitch[PredicateResult] = { + // we want to avoid evaluating the param if the candidate source fn yields false + if (candidate.getCandidateSources.keys.exists(candidateSources.contains) && !candidate.params( + param)) { + Stitch.value(PredicateResult.Invalid(Set(reason))) + } else { + Stitch.value(PredicateResult.Valid) + } + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/CuratedCompetitorListPredicate.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/CuratedCompetitorListPredicate.scala new file mode 100644 index 0000000000..16d76ce44c --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/CuratedCompetitorListPredicate.scala @@ -0,0 +1,66 @@ +package com.twitter.follow_recommendations.common.predicates + +import com.google.inject.name.Named +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.follow_recommendations.common.base.Predicate +import com.twitter.follow_recommendations.common.base.PredicateResult +import com.twitter.follow_recommendations.common.constants.GuiceNamedConstants +import com.twitter.follow_recommendations.common.models.FilterReason.CuratedAccountsCompetitorList +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.stitch.Stitch +import com.twitter.strato.client.Fetcher +import javax.inject.Inject +import javax.inject.Singleton +import com.twitter.conversions.DurationOps._ +import com.twitter.escherbird.util.stitchcache.StitchCache + +@Singleton +case class CuratedCompetitorListPredicate @Inject() ( + statsReceiver: StatsReceiver, + @Named(GuiceNamedConstants.CURATED_COMPETITOR_ACCOUNTS_FETCHER) competitorAccountFetcher: Fetcher[ + String, + Unit, + Seq[Long] + ]) extends Predicate[CandidateUser] { + + private val stats: StatsReceiver = statsReceiver.scope(this.getClass.getName) + private val cacheStats = stats.scope("cache") + + private val cache = StitchCache[String, Set[Long]]( + maxCacheSize = CuratedCompetitorListPredicate.CacheNumberOfEntries, + ttl = CuratedCompetitorListPredicate.CacheTTL, + statsReceiver = cacheStats, + underlyingCall = (competitorListPrefix: String) => query(competitorListPrefix) + ) + + private def query(prefix: String): Stitch[Set[Long]] = + competitorAccountFetcher.fetch(prefix).map(_.v.getOrElse(Nil).toSet) + + /** + * Caveat here is that though the similarToUserIds allows for a Seq[Long], in practice we would + * only return 1 userId. Multiple userId's would result in filtering candidates associated with + * a different similarToUserId. For example: + * - similarToUser1 -> candidate1, candidate2 + * - similarToUser2 -> candidate3 + * and in the competitorList store we have: + * - similarToUser1 -> candidate3 + * we'll be filtering candidate3 on account of similarToUser1, even though it was generated + * with similarToUser2. This might still be desirable at a product level (since we don't want + * to show these accounts anyway), but might not achieve what you intend to code-wise. + */ + override def apply(candidate: CandidateUser): Stitch[PredicateResult] = { + cache.readThrough(CuratedCompetitorListPredicate.DefaultKey).map { competitorListAccounts => + if (competitorListAccounts.contains(candidate.id)) { + PredicateResult.Invalid(Set(CuratedAccountsCompetitorList)) + } else { + PredicateResult.Valid + } + } + } +} + +object CuratedCompetitorListPredicate { + val DefaultKey: String = "default_list" + val CacheTTL = 5.minutes + val CacheNumberOfEntries = 5 +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/ExcludedUserIdPredicate.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/ExcludedUserIdPredicate.scala new file mode 100644 index 0000000000..01a5f96fda --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/ExcludedUserIdPredicate.scala @@ -0,0 +1,24 @@ +package com.twitter.follow_recommendations.common.predicates + +import com.twitter.follow_recommendations.common.base.Predicate +import com.twitter.follow_recommendations.common.base.PredicateResult +import com.twitter.follow_recommendations.common.models.FilterReason.ExcludedId +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.follow_recommendations.common.models.HasExcludedUserIds +import com.twitter.stitch.Stitch + +object ExcludedUserIdPredicate extends Predicate[(HasExcludedUserIds, CandidateUser)] { + + val ValidStitch: Stitch[PredicateResult.Valid.type] = Stitch.value(PredicateResult.Valid) + val ExcludedStitch: Stitch[PredicateResult.Invalid] = + Stitch.value(PredicateResult.Invalid(Set(ExcludedId))) + + override def apply(pair: (HasExcludedUserIds, CandidateUser)): Stitch[PredicateResult] = { + val (excludedUserIds, candidate) = pair + if (excludedUserIds.excludedUserIds.contains(candidate.id)) { + ExcludedStitch + } else { + ValidStitch + } + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/InactivePredicate.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/InactivePredicate.scala new file mode 100644 index 0000000000..c77538a994 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/InactivePredicate.scala @@ -0,0 +1,121 @@ +package com.twitter.follow_recommendations.common.predicates + +import com.google.inject.name.Named +import com.twitter.core_workflows.user_model.thriftscala.UserState +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.follow_recommendations.common.base.Predicate +import com.twitter.follow_recommendations.common.base.PredicateResult +import com.twitter.follow_recommendations.common.constants.GuiceNamedConstants +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.follow_recommendations.common.models.FilterReason +import com.twitter.follow_recommendations.common.predicates.InactivePredicateParams._ +import com.twitter.service.metastore.gen.thriftscala.UserRecommendabilityFeatures +import com.twitter.stitch.Stitch +import com.twitter.strato.client.Fetcher +import com.twitter.timelines.configapi.HasParams +import com.twitter.util.Duration +import com.twitter.util.Time +import javax.inject.Inject +import javax.inject.Singleton +import com.twitter.conversions.DurationOps._ +import com.twitter.escherbird.util.stitchcache.StitchCache +import com.twitter.follow_recommendations.common.models.HasUserState +import com.twitter.follow_recommendations.common.predicates.InactivePredicateParams.DefaultInactivityThreshold +import com.twitter.product_mixer.core.model.marshalling.request.HasClientContext + +import java.lang.{Long => JLong} + +@Singleton +case class InactivePredicate @Inject() ( + statsReceiver: StatsReceiver, + @Named(GuiceNamedConstants.USER_RECOMMENDABILITY_FETCHER) userRecommendabilityFetcher: Fetcher[ + Long, + Unit, + UserRecommendabilityFeatures + ]) extends Predicate[(HasParams with HasClientContext with HasUserState, CandidateUser)] { + + private val stats: StatsReceiver = statsReceiver.scope("InactivePredicate") + private val cacheStats = stats.scope("cache") + + private def queryUserRecommendable(userId: Long): Stitch[Option[UserRecommendabilityFeatures]] = + userRecommendabilityFetcher.fetch(userId).map(_.v) + + private val userRecommendableCache = + StitchCache[JLong, Option[UserRecommendabilityFeatures]]( + maxCacheSize = 100000, + ttl = 12.hours, + statsReceiver = cacheStats.scope("UserRecommendable"), + underlyingCall = (userId: JLong) => queryUserRecommendable(userId) + ) + + override def apply( + targetAndCandidate: (HasParams with HasClientContext with HasUserState, CandidateUser) + ): Stitch[PredicateResult] = { + val (target, candidate) = targetAndCandidate + + userRecommendableCache + .readThrough(candidate.id).map { + case recFeaturesFetchResult => + recFeaturesFetchResult match { + case None => + PredicateResult.Invalid(Set(FilterReason.MissingRecommendabilityData)) + case Some(recFeatures) => + if (disableInactivityPredicate(target, target.userState, recFeatures.userState)) { + PredicateResult.Valid + } else { + val defaultInactivityThreshold = target.params(DefaultInactivityThreshold).days + val hasBeenActiveRecently = recFeatures.lastStatusUpdateMs + .map(Time.now - Time.fromMilliseconds(_)).getOrElse( + Duration.Top) < defaultInactivityThreshold + stats + .scope(defaultInactivityThreshold.toString).counter( + if (hasBeenActiveRecently) + "active" + else + "inactive" + ).incr() + if (hasBeenActiveRecently && (!target + .params(UseEggFilter) || recFeatures.isNotEgg.contains(1))) { + PredicateResult.Valid + } else { + PredicateResult.Invalid(Set(FilterReason.Inactive)) + } + } + } + }.rescue { + case e: Exception => + stats.counter(e.getClass.getSimpleName).incr() + Stitch(PredicateResult.Invalid(Set(FilterReason.FailOpen))) + } + } + + private[this] def disableInactivityPredicate( + target: HasParams, + consumerState: Option[UserState], + candidateState: Option[UserState] + ): Boolean = { + target.params(MightBeDisabled) && + consumerState.exists(InactivePredicate.ValidConsumerStates.contains) && + ( + ( + candidateState.exists(InactivePredicate.ValidCandidateStates.contains) && + !target.params(OnlyDisableForNewUserStateCandidates) + ) || + ( + candidateState.contains(UserState.New) && + target.params(OnlyDisableForNewUserStateCandidates) + ) + ) + } +} + +object InactivePredicate { + val ValidConsumerStates: Set[UserState] = Set( + UserState.HeavyNonTweeter, + UserState.MediumNonTweeter, + UserState.HeavyTweeter, + UserState.MediumTweeter + ) + val ValidCandidateStates: Set[UserState] = + Set(UserState.New, UserState.VeryLight, UserState.Light, UserState.NearZero) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/InactivePredicateParams.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/InactivePredicateParams.scala new file mode 100644 index 0000000000..0bb52caa86 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/InactivePredicateParams.scala @@ -0,0 +1,21 @@ +package com.twitter.follow_recommendations.common.predicates + +import com.twitter.timelines.configapi.FSBoundedParam +import com.twitter.timelines.configapi.FSParam +import com.twitter.timelines.configapi.Param + +object InactivePredicateParams { + case object DefaultInactivityThreshold + extends FSBoundedParam[Int]( + name = "inactive_predicate_default_inactivity_threshold", + default = 60, + min = 1, + max = 500 + ) + case object UseEggFilter extends Param[Boolean](true) + case object MightBeDisabled extends FSParam[Boolean]("inactive_predicate_might_be_disabled", true) + case object OnlyDisableForNewUserStateCandidates + extends FSParam[Boolean]( + "inactive_predicate_only_disable_for_new_user_state_candidates", + false) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/PreviouslyRecommendedUserIdsPredicate.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/PreviouslyRecommendedUserIdsPredicate.scala new file mode 100644 index 0000000000..7879860ac6 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/PreviouslyRecommendedUserIdsPredicate.scala @@ -0,0 +1,34 @@ +package com.twitter.follow_recommendations.common.predicates + +import com.twitter.follow_recommendations.common.base.Predicate +import com.twitter.follow_recommendations.common.base.PredicateResult +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.follow_recommendations.common.models.FilterReason +import com.twitter.follow_recommendations.common.models.HasPreviousRecommendationsContext +import com.twitter.stitch.Stitch +import javax.inject.Singleton + +@Singleton +class PreviouslyRecommendedUserIdsPredicate + extends Predicate[(HasPreviousRecommendationsContext, CandidateUser)] { + override def apply( + pair: (HasPreviousRecommendationsContext, CandidateUser) + ): Stitch[PredicateResult] = { + + val (targetUser, candidate) = pair + + val previouslyRecommendedUserIDs = targetUser.previouslyRecommendedUserIDs + + if (!previouslyRecommendedUserIDs.contains(candidate.id)) { + PreviouslyRecommendedUserIdsPredicate.ValidStitch + } else { + PreviouslyRecommendedUserIdsPredicate.AlreadyRecommendedStitch + } + } +} + +object PreviouslyRecommendedUserIdsPredicate { + val ValidStitch: Stitch[PredicateResult.Valid.type] = Stitch.value(PredicateResult.Valid) + val AlreadyRecommendedStitch: Stitch[PredicateResult.Invalid] = + Stitch.value(PredicateResult.Invalid(Set(FilterReason.AlreadyRecommended))) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/dismiss/BUILD b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/dismiss/BUILD new file mode 100644 index 0000000000..9d1ca9b40b --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/dismiss/BUILD @@ -0,0 +1,17 @@ +scala_library( + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/com/google/inject:guice", + "3rdparty/jvm/com/google/inject/extensions:guice-assistedinject", + "3rdparty/jvm/net/codingwell:scala-guice", + "3rdparty/jvm/org/slf4j:slf4j-api", + "finatra/inject/inject-core/src/main/scala", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/dismiss_store", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/constants", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models", + "util/util-slf4j-api/src/main/scala", + ], +) diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/dismiss/DismissedCandidatePredicate.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/dismiss/DismissedCandidatePredicate.scala new file mode 100644 index 0000000000..550017b957 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/dismiss/DismissedCandidatePredicate.scala @@ -0,0 +1,32 @@ +package com.twitter.follow_recommendations.common.predicates.dismiss + +import com.twitter.follow_recommendations.common.base.Predicate +import com.twitter.follow_recommendations.common.base.PredicateResult +import com.twitter.follow_recommendations.common.models.FilterReason.DismissedId +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.follow_recommendations.common.models.HasDismissedUserIds +import com.twitter.stitch.Stitch +import javax.inject.Singleton + +@Singleton +class DismissedCandidatePredicate extends Predicate[(HasDismissedUserIds, CandidateUser)] { + + override def apply(pair: (HasDismissedUserIds, CandidateUser)): Stitch[PredicateResult] = { + + val (targetUser, candidate) = pair + targetUser.dismissedUserIds + .map { dismissedUserIds => + if (!dismissedUserIds.contains(candidate.id)) { + DismissedCandidatePredicate.ValidStitch + } else { + DismissedCandidatePredicate.DismissedStitch + } + }.getOrElse(DismissedCandidatePredicate.ValidStitch) + } +} + +object DismissedCandidatePredicate { + val ValidStitch: Stitch[PredicateResult.Valid.type] = Stitch.value(PredicateResult.Valid) + val DismissedStitch: Stitch[PredicateResult.Invalid] = + Stitch.value(PredicateResult.Invalid(Set(DismissedId))) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/dismiss/DismissedCandidatePredicateParams.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/dismiss/DismissedCandidatePredicateParams.scala new file mode 100644 index 0000000000..7f1d51a074 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/dismiss/DismissedCandidatePredicateParams.scala @@ -0,0 +1,9 @@ +package com.twitter.follow_recommendations.common.predicates.dismiss + +import com.twitter.conversions.DurationOps._ +import com.twitter.timelines.configapi.Param +import com.twitter.util.Duration + +object DismissedCandidatePredicateParams { + case object LookBackDuration extends Param[Duration](180.days) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/gizmoduck/BUILD b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/gizmoduck/BUILD new file mode 100644 index 0000000000..a154121e6b --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/gizmoduck/BUILD @@ -0,0 +1,23 @@ +scala_library( + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/com/google/inject:guice", + "3rdparty/jvm/com/google/inject/extensions:guice-assistedinject", + "3rdparty/jvm/net/codingwell:scala-guice", + "3rdparty/jvm/org/slf4j:slf4j-api", + "escherbird/src/scala/com/twitter/escherbird/util/stitchcache", + "finatra/inject/inject-core/src/main/scala", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/cache", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/gizmoduck", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/constants", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/common", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/deciders", + "stitch/stitch-gizmoduck", + "util/util-slf4j-api/src/main/scala", + "util/util-thrift", + ], +) diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/gizmoduck/GizmoduckPredicate.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/gizmoduck/GizmoduckPredicate.scala new file mode 100644 index 0000000000..2ca3e2fc51 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/gizmoduck/GizmoduckPredicate.scala @@ -0,0 +1,284 @@ +package com.twitter.follow_recommendations.common.predicates.gizmoduck + +import com.twitter.decider.Decider +import com.twitter.decider.RandomRecipient +import com.twitter.escherbird.util.stitchcache.StitchCache +import com.twitter.finagle.Memcached.Client +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.finagle.util.DefaultTimer +import com.twitter.follow_recommendations.common.base.StatsUtil +import com.twitter.follow_recommendations.common.base.Predicate +import com.twitter.follow_recommendations.common.base.PredicateResult +import com.twitter.follow_recommendations.common.clients.cache.MemcacheClient +import com.twitter.follow_recommendations.common.clients.cache.ThriftBijection +import com.twitter.follow_recommendations.common.models.FilterReason._ +import com.twitter.follow_recommendations.common.models.AddressBookMetadata +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.follow_recommendations.common.models.FilterReason +import com.twitter.follow_recommendations.common.predicates.gizmoduck.GizmoduckPredicate._ +import com.twitter.follow_recommendations.common.predicates.gizmoduck.GizmoduckPredicateParams._ +import com.twitter.follow_recommendations.configapi.deciders.DeciderKey +import com.twitter.gizmoduck.thriftscala.LabelValue.BlinkBad +import com.twitter.gizmoduck.thriftscala.LabelValue.BlinkWorst +import com.twitter.gizmoduck.thriftscala.LabelValue +import com.twitter.gizmoduck.thriftscala.LookupContext +import com.twitter.gizmoduck.thriftscala.QueryFields +import com.twitter.gizmoduck.thriftscala.User +import com.twitter.gizmoduck.thriftscala.UserResult +import com.twitter.product_mixer.core.model.marshalling.request.HasClientContext +import com.twitter.scrooge.CompactThriftSerializer +import com.twitter.spam.rtf.thriftscala.SafetyLevel +import com.twitter.stitch.Stitch +import com.twitter.stitch.gizmoduck.Gizmoduck +import com.twitter.timelines.configapi.HasParams +import com.twitter.util.Duration +import com.twitter.util.logging.Logging +import java.lang.{Long => JLong} +import javax.inject.Inject +import javax.inject.Singleton + +/** + * In this filter, we want to check 4 categories of conditions: + * - if candidate is discoverable given that it's from an address-book/phone-book based source + * - if candidate is unsuitable based on it's safety sub-fields in gizmoduck + * - if candidate is withheld because of country-specific take-down policies + * - if candidate is marked as bad/worst based on blink labels + * We fail close on the query as this is a product-critical filter + */ +@Singleton +case class GizmoduckPredicate @Inject() ( + gizmoduck: Gizmoduck, + client: Client, + statsReceiver: StatsReceiver, + decider: Decider = Decider.False) + extends Predicate[(HasClientContext with HasParams, CandidateUser)] + with Logging { + private val stats: StatsReceiver = statsReceiver.scope(this.getClass.getName) + + // track # of Gizmoduck predicate queries that yielded valid & invalid predicate results + private val validPredicateResultCounter = stats.counter("predicate_valid") + private val invalidPredicateResultCounter = stats.counter("predicate_invalid") + + // track # of cases where no Gizmoduck user was found + private val noGizmoduckUserCounter = stats.counter("no_gizmoduck_user_found") + + private val gizmoduckCache = StitchCache[JLong, UserResult]( + maxCacheSize = MaxCacheSize, + ttl = CacheTTL, + statsReceiver = stats.scope("cache"), + underlyingCall = getByUserId + ) + + // Distributed Twemcache to store UserResult objects keyed on user IDs + val bijection = new ThriftBijection[UserResult] { + override val serializer = CompactThriftSerializer(UserResult) + } + val memcacheClient = MemcacheClient[UserResult]( + client = client, + dest = "/s/cache/frs:twemcaches", + valueBijection = bijection, + ttl = CacheTTL, + statsReceiver = stats.scope("twemcache") + ) + + // main method used to apply GizmoduckPredicate to a candidate user + override def apply( + pair: (HasClientContext with HasParams, CandidateUser) + ): Stitch[PredicateResult] = { + val (request, candidate) = pair + // measure the latency of the getGizmoduckPredicateResult, since this predicate + // check is product-critical and relies on querying a core service (Gizmoduck) + StatsUtil.profileStitch( + getGizmoduckPredicateResult(request, candidate), + stats.scope("getGizmoduckPredicateResult") + ) + } + + private def getGizmoduckPredicateResult( + request: HasClientContext with HasParams, + candidate: CandidateUser + ): Stitch[PredicateResult] = { + val timeout: Duration = request.params(GizmoduckGetTimeout) + + val deciderKey: String = DeciderKey.EnableGizmoduckCaching.toString + val enableDistributedCaching: Boolean = decider.isAvailable(deciderKey, Some(RandomRecipient)) + + // try getting an existing UserResult from cache if possible + val userResultStitch: Stitch[UserResult] = + enableDistributedCaching match { + // read from memcache + case true => memcacheClient.readThrough( + // add a key prefix to address cache key collisions + key = "GizmoduckPredicate" + candidate.id.toString, + underlyingCall = () => getByUserId(candidate.id) + ) + // read from local cache + case false => gizmoduckCache.readThrough(candidate.id) + } + + val predicateResultStitch = userResultStitch.map { + userResult => { + val predicateResult = getPredicateResult(request, candidate, userResult) + if (enableDistributedCaching) { + predicateResult match { + case PredicateResult.Valid => + stats.scope("twemcache").counter("predicate_valid").incr() + case PredicateResult.Invalid(reasons) => + stats.scope("twemcache").counter("predicate_invalid").incr() + } + // log metrics to check if local cache value matches distributed cache value + logPredicateResultEquality( + request, + candidate, + predicateResult + ) + } else { + predicateResult match { + case PredicateResult.Valid => + stats.scope("cache").counter("predicate_valid").incr() + case PredicateResult.Invalid(reasons) => + stats.scope("cache").counter("predicate_invalid").incr() + } + } + predicateResult + } + } + predicateResultStitch + .within(timeout)(DefaultTimer) + .rescue { // fail-open when timeout or exception + case e: Exception => + stats.scope("rescued").counter(e.getClass.getSimpleName).incr() + invalidPredicateResultCounter.incr() + Stitch(PredicateResult.Invalid(Set(FailOpen))) + } + } + + private def logPredicateResultEquality( + request: HasClientContext with HasParams, + candidate: CandidateUser, + predicateResult: PredicateResult + ): Unit = { + val localCachedUserResult = Option(gizmoduckCache.cache.getIfPresent(candidate.id)) + if (localCachedUserResult.isDefined) { + val localPredicateResult = getPredicateResult(request, candidate, localCachedUserResult.get) + localPredicateResult.equals(predicateResult) match { + case true => stats.scope("has_equal_predicate_value").counter("true").incr() + case false => stats.scope("has_equal_predicate_value").counter("false").incr() + } + } else { + stats.scope("has_equal_predicate_value").counter("undefined").incr() + } + } + + // method to get PredicateResult from UserResult + def getPredicateResult( + request: HasClientContext with HasParams, + candidate: CandidateUser, + userResult: UserResult, + ): PredicateResult = { + userResult.user match { + case Some(user) => + val abPbReasons = getAbPbReason(user, candidate.getAddressBookMetadata) + val safetyReasons = getSafetyReasons(user) + val countryTakedownReasons = getCountryTakedownReasons(user, request.getCountryCode) + val blinkReasons = getBlinkReasons(user) + val allReasons = + abPbReasons ++ safetyReasons ++ countryTakedownReasons ++ blinkReasons + if (allReasons.nonEmpty) { + invalidPredicateResultCounter.incr() + PredicateResult.Invalid(allReasons) + } else { + validPredicateResultCounter.incr() + PredicateResult.Valid + } + case None => + noGizmoduckUserCounter.incr() + invalidPredicateResultCounter.incr() + PredicateResult.Invalid(Set(NoUser)) + } + } + + private def getByUserId(userId: JLong): Stitch[UserResult] = { + StatsUtil.profileStitch( + gizmoduck.getById(userId = userId, queryFields = queryFields, context = lookupContext), + stats.scope("getByUserId") + ) + } +} + +object GizmoduckPredicate { + + private[gizmoduck] val lookupContext: LookupContext = + LookupContext(`includeDeactivated` = true, `safetyLevel` = Some(SafetyLevel.Recommendations)) + + private[gizmoduck] val queryFields: Set[QueryFields] = + Set( + QueryFields.Discoverability, // needed for Address Book / Phone Book discoverability checks in getAbPbReason + QueryFields.Safety, // needed for user state safety checks in getSafetyReasons, getCountryTakedownReasons + QueryFields.Labels, // needed for user label checks in getBlinkReasons + QueryFields.Takedowns // needed for checking takedown labels for a user in getCountryTakedownReasons + ) + + private[gizmoduck] val BlinkLabels: Set[LabelValue] = Set(BlinkBad, BlinkWorst) + + private[gizmoduck] def getAbPbReason( + user: User, + abMetadataOpt: Option[AddressBookMetadata] + ): Set[FilterReason] = { + (for { + discoverability <- user.discoverability + abMetadata <- abMetadataOpt + } yield { + val AddressBookMetadata(fwdPb, rvPb, fwdAb, rvAb) = abMetadata + val abReason: Set[FilterReason] = + if ((!discoverability.discoverableByEmail) && (fwdAb || rvAb)) + Set(AddressBookUndiscoverable) + else Set.empty + val pbReason: Set[FilterReason] = + if ((!discoverability.discoverableByMobilePhone) && (fwdPb || rvPb)) + Set(PhoneBookUndiscoverable) + else Set.empty + abReason ++ pbReason + }).getOrElse(Set.empty) + } + + private[gizmoduck] def getSafetyReasons(user: User): Set[FilterReason] = { + user.safety + .map { s => + val deactivatedReason: Set[FilterReason] = + if (s.deactivated) Set(Deactivated) else Set.empty + val suspendedReason: Set[FilterReason] = if (s.suspended) Set(Suspended) else Set.empty + val restrictedReason: Set[FilterReason] = if (s.restricted) Set(Restricted) else Set.empty + val nsfwUserReason: Set[FilterReason] = if (s.nsfwUser) Set(NsfwUser) else Set.empty + val nsfwAdminReason: Set[FilterReason] = if (s.nsfwAdmin) Set(NsfwAdmin) else Set.empty + val isProtectedReason: Set[FilterReason] = if (s.isProtected) Set(IsProtected) else Set.empty + deactivatedReason ++ suspendedReason ++ restrictedReason ++ nsfwUserReason ++ nsfwAdminReason ++ isProtectedReason + }.getOrElse(Set.empty) + } + + private[gizmoduck] def getCountryTakedownReasons( + user: User, + countryCodeOpt: Option[String] + ): Set[FilterReason] = { + (for { + safety <- user.safety.toSeq + if safety.hasTakedown + takedowns <- user.takedowns.toSeq + takedownCountry <- takedowns.countryCodes + requestingCountry <- countryCodeOpt + if takedownCountry.toLowerCase == requestingCountry.toLowerCase + } yield Set(CountryTakedown(takedownCountry.toLowerCase))).flatten.toSet + } + + private[gizmoduck] def getBlinkReasons(user: User): Set[FilterReason] = { + user.labels + .map(_.labels.map(_.labelValue)) + .getOrElse(Nil) + .exists(BlinkLabels.contains) + for { + labels <- user.labels.toSeq + label <- labels.labels + if (BlinkLabels.contains(label.labelValue)) + } yield Set(Blink) + }.flatten.toSet +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/gizmoduck/GizmoduckPredicateCache.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/gizmoduck/GizmoduckPredicateCache.scala new file mode 100644 index 0000000000..36fb2f20fe --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/gizmoduck/GizmoduckPredicateCache.scala @@ -0,0 +1,50 @@ +package com.twitter.follow_recommendations.common.predicates.gizmoduck + +import java.util.concurrent.TimeUnit + +import com.google.common.base.Ticker +import com.google.common.cache.CacheBuilder +import com.google.common.cache.Cache +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.util.Time +import com.twitter.util.Duration + +/** + * In-memory cache used for caching GizmoduckPredicate query calls in + * com.twitter.follow_recommendations.common.predicates.gizmoduck.GizmoduckPredicate. + * + * References the cache implementation in com.twitter.escherbird.util.stitchcache, + * but without the underlying Stitch call. + */ +object GizmoduckPredicateCache { + + private[GizmoduckPredicateCache] class TimeTicker extends Ticker { + override def read(): Long = Time.now.inNanoseconds + } + + def apply[K, V]( + maxCacheSize: Int, + ttl: Duration, + statsReceiver: StatsReceiver + ): Cache[K, V] = { + + val cache: Cache[K, V] = + CacheBuilder + .newBuilder() + .maximumSize(maxCacheSize) + .asInstanceOf[CacheBuilder[K, V]] + .expireAfterWrite(ttl.inSeconds, TimeUnit.SECONDS) + .recordStats() + .ticker(new TimeTicker()) + .build() + + // metrics for tracking cache usage + statsReceiver.provideGauge("cache_size") { cache.size.toFloat } + statsReceiver.provideGauge("cache_hits") { cache.stats.hitCount.toFloat } + statsReceiver.provideGauge("cache_misses") { cache.stats.missCount.toFloat } + statsReceiver.provideGauge("cache_hit_rate") { cache.stats.hitRate.toFloat } + statsReceiver.provideGauge("cache_evictions") { cache.stats.evictionCount.toFloat } + + cache + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/gizmoduck/GizmoduckPredicateFSConfig.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/gizmoduck/GizmoduckPredicateFSConfig.scala new file mode 100644 index 0000000000..447eac835e --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/gizmoduck/GizmoduckPredicateFSConfig.scala @@ -0,0 +1,17 @@ +package com.twitter.follow_recommendations.common.predicates.gizmoduck + +import com.twitter.follow_recommendations.common.predicates.gizmoduck.GizmoduckPredicateParams._ +import com.twitter.follow_recommendations.configapi.common.FeatureSwitchConfig +import com.twitter.timelines.configapi.FSBoundedParam +import com.twitter.timelines.configapi.HasDurationConversion +import com.twitter.util.Duration + +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class GizmoduckPredicateFSConfig @Inject() () extends FeatureSwitchConfig { + override val durationFSParams: Seq[FSBoundedParam[Duration] with HasDurationConversion] = Seq( + GizmoduckGetTimeout + ) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/gizmoduck/GizmoduckPredicateParams.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/gizmoduck/GizmoduckPredicateParams.scala new file mode 100644 index 0000000000..811897e278 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/gizmoduck/GizmoduckPredicateParams.scala @@ -0,0 +1,21 @@ +package com.twitter.follow_recommendations.common.predicates.gizmoduck + +import com.twitter.timelines.configapi.FSBoundedParam +import com.twitter.timelines.configapi.DurationConversion +import com.twitter.timelines.configapi.HasDurationConversion +import com.twitter.util.Duration +import com.twitter.conversions.DurationOps._ + +object GizmoduckPredicateParams { + case object GizmoduckGetTimeout + extends FSBoundedParam[Duration]( + name = "gizmoduck_predicate_timeout_in_millis", + default = 200.millisecond, + min = 1.millisecond, + max = 500.millisecond) + with HasDurationConversion { + override def durationConversion: DurationConversion = DurationConversion.FromMillis + } + val MaxCacheSize: Int = 250000 + val CacheTTL: Duration = Duration.fromHours(6) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/health/BUILD b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/health/BUILD new file mode 100644 index 0000000000..d0e9e10159 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/health/BUILD @@ -0,0 +1,21 @@ +scala_library( + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/com/google/inject:guice", + "3rdparty/jvm/com/google/inject/extensions:guice-assistedinject", + "3rdparty/jvm/net/codingwell:scala-guice", + "3rdparty/jvm/org/slf4j:slf4j-api", + "escherbird/src/scala/com/twitter/escherbird/util/stitchcache", + "finatra/inject/inject-core/src/main/scala", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/cache", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/constants", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/common", + "strato/config/columns/hss/user_signals/api:api-strato-client", + "util/util-slf4j-api/src/main/scala", + "util/util-thrift", + ], +) diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/health/HssPredicate.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/health/HssPredicate.scala new file mode 100644 index 0000000000..7464be7df7 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/health/HssPredicate.scala @@ -0,0 +1,95 @@ +package com.twitter.follow_recommendations.common.predicates.hss + +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.finagle.util.DefaultTimer +import com.twitter.follow_recommendations.common.base.Predicate +import com.twitter.follow_recommendations.common.base.PredicateResult +import com.twitter.follow_recommendations.common.base.StatsUtil +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.follow_recommendations.common.models.FilterReason +import com.twitter.follow_recommendations.common.models.FilterReason.FailOpen +import com.twitter.hss.api.thriftscala.SignalValue +import com.twitter.hss.api.thriftscala.UserHealthSignal.AgathaCseDouble +import com.twitter.hss.api.thriftscala.UserHealthSignal.NsfwAgathaUserScoreDouble +import com.twitter.product_mixer.core.model.marshalling.request.HasClientContext +import com.twitter.stitch.Stitch +import com.twitter.strato.generated.client.hss.user_signals.api.HealthSignalsOnUserClientColumn +import com.twitter.timelines.configapi.HasParams +import com.twitter.util.logging.Logging +import com.twitter.util.Duration + +import javax.inject.Inject +import javax.inject.Singleton + +/** + * Filter out candidates based on Health Signal Store (HSS) health signals + */ +@Singleton +case class HssPredicate @Inject() ( + healthSignalsOnUserClientColumn: HealthSignalsOnUserClientColumn, + statsReceiver: StatsReceiver) + extends Predicate[(HasClientContext with HasParams, CandidateUser)] + with Logging { + + private val stats: StatsReceiver = statsReceiver.scope(this.getClass.getName) + + override def apply( + pair: (HasClientContext with HasParams, CandidateUser) + ): Stitch[PredicateResult] = { + val (request, candidate) = pair + StatsUtil.profileStitch( + getHssPredicateResult(request, candidate), + stats.scope("getHssPredicateResult") + ) + } + + private def getHssPredicateResult( + request: HasClientContext with HasParams, + candidate: CandidateUser + ): Stitch[PredicateResult] = { + + val hssCseScoreThreshold: Double = request.params(HssPredicateParams.HssCseScoreThreshold) + val hssNsfwScoreThreshold: Double = request.params(HssPredicateParams.HssNsfwScoreThreshold) + val timeout: Duration = request.params(HssPredicateParams.HssApiTimeout) + + healthSignalsOnUserClientColumn.fetcher + .fetch(candidate.id, Seq(AgathaCseDouble, NsfwAgathaUserScoreDouble)) + .map { fetchResult => + fetchResult.v match { + case Some(response) => + val agathaCseScoreDouble: Double = userHealthSignalValueToDoubleOpt( + response.signalValues.get(AgathaCseDouble)).getOrElse(0d) + val agathaNsfwScoreDouble: Double = userHealthSignalValueToDoubleOpt( + response.signalValues.get(NsfwAgathaUserScoreDouble)).getOrElse(0d) + + stats.stat("agathaCseScoreDistribution").add(agathaCseScoreDouble.toFloat) + stats.stat("agathaNsfwScoreDistribution").add(agathaNsfwScoreDouble.toFloat) + + /** + * Only filter out the candidate when it has both high Agatha CSE score and NSFW score, as the Agatha CSE + * model is an old one that may not be precise or have high recall. + */ + if (agathaCseScoreDouble >= hssCseScoreThreshold && agathaNsfwScoreDouble >= hssNsfwScoreThreshold) { + PredicateResult.Invalid(Set(FilterReason.HssSignal)) + } else { + PredicateResult.Valid + } + case None => + PredicateResult.Valid + } + } + .within(timeout)(DefaultTimer) + .rescue { + case e: Exception => + stats.scope("rescued").counter(e.getClass.getSimpleName).incr() + Stitch(PredicateResult.Invalid(Set(FailOpen))) + } + } + + private def userHealthSignalValueToDoubleOpt(signalValue: Option[SignalValue]): Option[Double] = { + signalValue match { + case Some(SignalValue.DoubleValue(value)) => Some(value) + case _ => None + } + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/health/HssPredicateFSConfig.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/health/HssPredicateFSConfig.scala new file mode 100644 index 0000000000..8cc1620a9d --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/health/HssPredicateFSConfig.scala @@ -0,0 +1,22 @@ +package com.twitter.follow_recommendations.common.predicates.hss + +import com.twitter.follow_recommendations.common.predicates.hss.HssPredicateParams._ +import com.twitter.follow_recommendations.configapi.common.FeatureSwitchConfig +import com.twitter.timelines.configapi.FSBoundedParam +import com.twitter.timelines.configapi.HasDurationConversion +import com.twitter.util.Duration + +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class HssPredicateFSConfig @Inject() () extends FeatureSwitchConfig { + override val doubleFSParams: Seq[FSBoundedParam[Double]] = Seq( + HssCseScoreThreshold, + HssNsfwScoreThreshold, + ) + + override val durationFSParams: Seq[FSBoundedParam[Duration] with HasDurationConversion] = Seq( + HssApiTimeout + ) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/health/HssPredicateParams.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/health/HssPredicateParams.scala new file mode 100644 index 0000000000..ac6e14bbe1 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/health/HssPredicateParams.scala @@ -0,0 +1,34 @@ +package com.twitter.follow_recommendations.common.predicates.hss + +import com.twitter.conversions.DurationOps._ +import com.twitter.timelines.configapi.DurationConversion +import com.twitter.timelines.configapi.FSBoundedParam +import com.twitter.timelines.configapi.HasDurationConversion +import com.twitter.util.Duration + +object HssPredicateParams { + object HssCseScoreThreshold + extends FSBoundedParam[Double]( + "hss_predicate_cse_score_threshold", + default = 0.992d, + min = 0.0d, + max = 1.0d) + + object HssNsfwScoreThreshold + extends FSBoundedParam[Double]( + "hss_predicate_nsfw_score_threshold", + default = 1.5d, + min = -100.0d, + max = 100.0d) + + object HssApiTimeout + extends FSBoundedParam[Duration]( + name = "hss_predicate_timeout_in_millis", + default = 200.millisecond, + min = 1.millisecond, + max = 500.millisecond) + with HasDurationConversion { + override def durationConversion: DurationConversion = DurationConversion.FromMillis + } + +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/sgs/BUILD b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/sgs/BUILD new file mode 100644 index 0000000000..0df7b245a2 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/sgs/BUILD @@ -0,0 +1,19 @@ +scala_library( + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/com/google/inject:guice", + "3rdparty/jvm/com/google/inject/extensions:guice-assistedinject", + "3rdparty/jvm/net/codingwell:scala-guice", + "3rdparty/jvm/org/slf4j:slf4j-api", + "finatra/inject/inject-core/src/main/scala", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/socialgraph", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/constants", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/common", + "src/thrift/com/twitter/socialgraph:thrift-scala", + "util/util-slf4j-api/src/main/scala", + ], +) diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/sgs/InvalidRelationshipPredicate.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/sgs/InvalidRelationshipPredicate.scala new file mode 100644 index 0000000000..84b8bf7a62 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/sgs/InvalidRelationshipPredicate.scala @@ -0,0 +1,36 @@ +package com.twitter.follow_recommendations.common.predicates.sgs + +import com.twitter.follow_recommendations.common.base.Predicate +import com.twitter.follow_recommendations.common.base.PredicateResult +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.follow_recommendations.common.models.FilterReason +import com.twitter.follow_recommendations.common.models.HasInvalidRelationshipUserIds +import com.twitter.stitch.Stitch +import javax.inject.Singleton + +@Singleton +class InvalidRelationshipPredicate + extends Predicate[(HasInvalidRelationshipUserIds, CandidateUser)] { + + override def apply( + pair: (HasInvalidRelationshipUserIds, CandidateUser) + ): Stitch[PredicateResult] = { + + val (targetUser, candidate) = pair + targetUser.invalidRelationshipUserIds match { + case Some(users) => + if (!users.contains(candidate.id)) { + InvalidRelationshipPredicate.ValidStitch + } else { + Stitch.value(InvalidRelationshipPredicate.InvalidRelationshipStitch) + } + case None => Stitch.value(PredicateResult.Valid) + } + } +} + +object InvalidRelationshipPredicate { + val ValidStitch: Stitch[PredicateResult.Valid.type] = Stitch.value(PredicateResult.Valid) + val InvalidRelationshipStitch: PredicateResult.Invalid = + PredicateResult.Invalid(Set(FilterReason.InvalidRelationship)) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/sgs/RecentFollowingPredicate.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/sgs/RecentFollowingPredicate.scala new file mode 100644 index 0000000000..60f0080b8a --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/sgs/RecentFollowingPredicate.scala @@ -0,0 +1,33 @@ +package com.twitter.follow_recommendations.common.predicates.sgs + +import com.twitter.follow_recommendations.common.base.Predicate +import com.twitter.follow_recommendations.common.base.PredicateResult +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.follow_recommendations.common.models.FilterReason +import com.twitter.follow_recommendations.common.models.HasRecentFollowedUserIds +import com.twitter.stitch.Stitch +import javax.inject.Singleton + +@Singleton +class RecentFollowingPredicate extends Predicate[(HasRecentFollowedUserIds, CandidateUser)] { + + override def apply(pair: (HasRecentFollowedUserIds, CandidateUser)): Stitch[PredicateResult] = { + + val (targetUser, candidate) = pair + targetUser.recentFollowedUserIdsSet match { + case Some(users) => + if (!users.contains(candidate.id)) { + RecentFollowingPredicate.ValidStitch + } else { + RecentFollowingPredicate.AlreadyFollowedStitch + } + case None => RecentFollowingPredicate.ValidStitch + } + } +} + +object RecentFollowingPredicate { + val ValidStitch: Stitch[PredicateResult.Valid.type] = Stitch.value(PredicateResult.Valid) + val AlreadyFollowedStitch: Stitch[PredicateResult.Invalid] = + Stitch.value(PredicateResult.Invalid(Set(FilterReason.AlreadyFollowed))) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/sgs/SgsPredicateFSConfig.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/sgs/SgsPredicateFSConfig.scala new file mode 100644 index 0000000000..f661dbbab8 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/sgs/SgsPredicateFSConfig.scala @@ -0,0 +1,16 @@ +package com.twitter.follow_recommendations.common.predicates.sgs + +import com.twitter.follow_recommendations.configapi.common.FeatureSwitchConfig +import com.twitter.timelines.configapi.FSBoundedParam +import com.twitter.timelines.configapi.HasDurationConversion +import com.twitter.util.Duration + +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class SgsPredicateFSConfig @Inject() () extends FeatureSwitchConfig { + override val durationFSParams: Seq[FSBoundedParam[Duration] with HasDurationConversion] = Seq( + SgsPredicateParams.SgsRelationshipsPredicateTimeout + ) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/sgs/SgsPredicateParams.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/sgs/SgsPredicateParams.scala new file mode 100644 index 0000000000..dd615c47d3 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/sgs/SgsPredicateParams.scala @@ -0,0 +1,19 @@ +package com.twitter.follow_recommendations.common.predicates.sgs + +import com.twitter.timelines.configapi.FSBoundedParam +import com.twitter.timelines.configapi.DurationConversion +import com.twitter.timelines.configapi.HasDurationConversion +import com.twitter.util.Duration +import com.twitter.conversions.DurationOps._ + +object SgsPredicateParams { + case object SgsRelationshipsPredicateTimeout + extends FSBoundedParam[Duration]( + name = "sgs_predicate_relationships_timeout_in_millis", + default = 300.millisecond, + min = 1.millisecond, + max = 1000.millisecond) + with HasDurationConversion { + override def durationConversion: DurationConversion = DurationConversion.FromMillis + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/sgs/SgsRelationshipsByUserIdPredicate.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/sgs/SgsRelationshipsByUserIdPredicate.scala new file mode 100644 index 0000000000..dec936e586 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/sgs/SgsRelationshipsByUserIdPredicate.scala @@ -0,0 +1,113 @@ +package com.twitter.follow_recommendations.common.predicates.sgs + +import com.google.common.annotations.VisibleForTesting +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.follow_recommendations.common.base.Predicate +import com.twitter.follow_recommendations.common.base.PredicateResult +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.follow_recommendations.common.models.FilterReason.InvalidRelationshipTypes +import com.twitter.socialgraph.thriftscala.ExistsRequest +import com.twitter.socialgraph.thriftscala.ExistsResult +import com.twitter.socialgraph.thriftscala.LookupContext +import com.twitter.socialgraph.thriftscala.Relationship +import com.twitter.socialgraph.thriftscala.RelationshipType +import com.twitter.stitch.Stitch +import com.twitter.stitch.socialgraph.SocialGraph +import com.twitter.util.logging.Logging +import javax.inject.Inject +import javax.inject.Singleton + +class SgsRelationshipsByUserIdPredicate( + socialGraph: SocialGraph, + relationshipMappings: Seq[RelationshipMapping], + statsReceiver: StatsReceiver) + extends Predicate[(Option[Long], CandidateUser)] + with Logging { + private val InvalidFromPrimaryCandidateSourceName = "invalid_from_primary_candidate_source" + private val InvalidFromCandidateSourceName = "invalid_from_candidate_source" + private val NoPrimaryCandidateSource = "no_primary_candidate_source" + + private val stats: StatsReceiver = statsReceiver.scope(this.getClass.getName) + + override def apply( + pair: (Option[Long], CandidateUser) + ): Stitch[PredicateResult] = { + val (idOpt, candidate) = pair + val relationships = relationshipMappings.map { relationshipMapping: RelationshipMapping => + Relationship( + relationshipMapping.relationshipType, + relationshipMapping.includeBasedOnRelationship) + } + idOpt + .map { id: Long => + val existsRequest = ExistsRequest( + id, + candidate.id, + relationships = relationships, + context = SgsRelationshipsByUserIdPredicate.UnionLookupContext + ) + socialGraph + .exists(existsRequest).map { existsResult: ExistsResult => + if (existsResult.exists) { + candidate.getPrimaryCandidateSource match { + case Some(candidateSource) => + stats + .scope(InvalidFromPrimaryCandidateSourceName).counter( + candidateSource.name).incr() + case None => + stats + .scope(InvalidFromPrimaryCandidateSourceName).counter( + NoPrimaryCandidateSource).incr() + } + candidate.getCandidateSources.foreach({ + case (candidateSource, _) => + stats + .scope(InvalidFromCandidateSourceName).counter(candidateSource.name).incr() + }) + PredicateResult.Invalid(Set(InvalidRelationshipTypes(relationshipMappings + .map { relationshipMapping: RelationshipMapping => + relationshipMapping.relationshipType + }.mkString(", ")))) + } else { + PredicateResult.Valid + } + } + } + // if no user id is present, return true by default + .getOrElse(Stitch.value(PredicateResult.Valid)) + } +} + +object SgsRelationshipsByUserIdPredicate { + // OR Operation + @VisibleForTesting + private[follow_recommendations] val UnionLookupContext = Some( + LookupContext(performUnion = Some(true))) +} + +@Singleton +class ExcludeNonFollowersSgsPredicate @Inject() ( + socialGraph: SocialGraph, + statsReceiver: StatsReceiver) + extends SgsRelationshipsByUserIdPredicate( + socialGraph, + Seq(RelationshipMapping(RelationshipType.FollowedBy, includeBasedOnRelationship = false)), + statsReceiver) + +@Singleton +class ExcludeNonFollowingSgsPredicate @Inject() ( + socialGraph: SocialGraph, + statsReceiver: StatsReceiver) + extends SgsRelationshipsByUserIdPredicate( + socialGraph, + Seq(RelationshipMapping(RelationshipType.Following, includeBasedOnRelationship = false)), + statsReceiver) + +@Singleton +class ExcludeFollowingSgsPredicate @Inject() ( + socialGraph: SocialGraph, + statsReceiver: StatsReceiver) + extends SgsRelationshipsByUserIdPredicate( + socialGraph, + Seq(RelationshipMapping(RelationshipType.Following, includeBasedOnRelationship = true)), + statsReceiver) diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/sgs/SgsRelationshipsPredicate.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/sgs/SgsRelationshipsPredicate.scala new file mode 100644 index 0000000000..fdb88ad582 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/sgs/SgsRelationshipsPredicate.scala @@ -0,0 +1,146 @@ +package com.twitter.follow_recommendations.common.predicates.sgs + +import com.google.common.annotations.VisibleForTesting +import com.twitter.finagle.stats.NullStatsReceiver +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.follow_recommendations.common.base.Predicate +import com.twitter.follow_recommendations.common.base.PredicateResult +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.follow_recommendations.common.models.HasProfileId +import com.twitter.follow_recommendations.common.models.FilterReason.FailOpen +import com.twitter.follow_recommendations.common.models.FilterReason.InvalidRelationshipTypes +import com.twitter.product_mixer.core.model.marshalling.request.HasClientContext +import com.twitter.socialgraph.thriftscala.ExistsRequest +import com.twitter.socialgraph.thriftscala.ExistsResult +import com.twitter.socialgraph.thriftscala.LookupContext +import com.twitter.socialgraph.thriftscala.Relationship +import com.twitter.socialgraph.thriftscala.RelationshipType +import com.twitter.stitch.Stitch +import com.twitter.stitch.socialgraph.SocialGraph +import com.twitter.timelines.configapi.HasParams +import com.twitter.util.TimeoutException +import com.twitter.util.logging.Logging + +import javax.inject.Inject +import javax.inject.Singleton + +case class RelationshipMapping( + relationshipType: RelationshipType, + includeBasedOnRelationship: Boolean) + +class SgsRelationshipsPredicate( + socialGraph: SocialGraph, + relationshipMappings: Seq[RelationshipMapping], + statsReceiver: StatsReceiver = NullStatsReceiver) + extends Predicate[(HasClientContext with HasParams, CandidateUser)] + with Logging { + + private val stats: StatsReceiver = statsReceiver.scope(this.getClass.getSimpleName) + + override def apply( + pair: (HasClientContext with HasParams, CandidateUser) + ): Stitch[PredicateResult] = { + val (target, candidate) = pair + val timeout = target.params(SgsPredicateParams.SgsRelationshipsPredicateTimeout) + SgsRelationshipsPredicate + .extractUserId(target) + .map { id => + val relationships = relationshipMappings.map { relationshipMapping: RelationshipMapping => + Relationship( + relationshipMapping.relationshipType, + relationshipMapping.includeBasedOnRelationship) + } + val existsRequest = ExistsRequest( + id, + candidate.id, + relationships = relationships, + context = SgsRelationshipsPredicate.UnionLookupContext + ) + socialGraph + .exists(existsRequest).map { existsResult: ExistsResult => + if (existsResult.exists) { + PredicateResult.Invalid(Set(InvalidRelationshipTypes(relationshipMappings + .map { relationshipMapping: RelationshipMapping => + relationshipMapping.relationshipType + }.mkString(", ")))) + } else { + PredicateResult.Valid + } + } + .within(timeout)(com.twitter.finagle.util.DefaultTimer) + } + // if no user id is present, return true by default + .getOrElse(Stitch.value(PredicateResult.Valid)) + .rescue { + case e: TimeoutException => + stats.counter("timeout").incr() + Stitch(PredicateResult.Invalid(Set(FailOpen))) + case e: Exception => + stats.counter(e.getClass.getSimpleName).incr() + Stitch(PredicateResult.Invalid(Set(FailOpen))) + } + + } +} + +object SgsRelationshipsPredicate { + // OR Operation + @VisibleForTesting + private[follow_recommendations] val UnionLookupContext = Some( + LookupContext(performUnion = Some(true))) + + private def extractUserId(target: HasClientContext with HasParams): Option[Long] = target match { + case profRequest: HasProfileId => Some(profRequest.profileId) + case userRequest: HasClientContext with HasParams => userRequest.getOptionalUserId + case _ => None + } +} + +@Singleton +class InvalidTargetCandidateRelationshipTypesPredicate @Inject() ( + socialGraph: SocialGraph) + extends SgsRelationshipsPredicate( + socialGraph, + InvalidRelationshipTypesPredicate.InvalidRelationshipTypes) {} + +@Singleton +class NoteworthyAccountsSgsPredicate @Inject() ( + socialGraph: SocialGraph) + extends SgsRelationshipsPredicate( + socialGraph, + InvalidRelationshipTypesPredicate.NoteworthyAccountsInvalidRelationshipTypes) + +object InvalidRelationshipTypesPredicate { + + val InvalidRelationshipTypesExcludeFollowing: Seq[RelationshipMapping] = Seq( + RelationshipMapping(RelationshipType.HideRecommendations, true), + RelationshipMapping(RelationshipType.Blocking, true), + RelationshipMapping(RelationshipType.BlockedBy, true), + RelationshipMapping(RelationshipType.Muting, true), + RelationshipMapping(RelationshipType.MutedBy, true), + RelationshipMapping(RelationshipType.ReportedAsSpam, true), + RelationshipMapping(RelationshipType.ReportedAsSpamBy, true), + RelationshipMapping(RelationshipType.ReportedAsAbuse, true), + RelationshipMapping(RelationshipType.ReportedAsAbuseBy, true) + ) + + val InvalidRelationshipTypes: Seq[RelationshipMapping] = Seq( + RelationshipMapping(RelationshipType.FollowRequestOutgoing, true), + RelationshipMapping(RelationshipType.Following, true), + RelationshipMapping( + RelationshipType.UsedToFollow, + true + ) // this data is accessible for 90 days. + ) ++ InvalidRelationshipTypesExcludeFollowing + + val NoteworthyAccountsInvalidRelationshipTypes: Seq[RelationshipMapping] = Seq( + RelationshipMapping(RelationshipType.Blocking, true), + RelationshipMapping(RelationshipType.BlockedBy, true), + RelationshipMapping(RelationshipType.Muting, true), + RelationshipMapping(RelationshipType.MutedBy, true), + RelationshipMapping(RelationshipType.ReportedAsSpam, true), + RelationshipMapping(RelationshipType.ReportedAsSpamBy, true), + RelationshipMapping(RelationshipType.ReportedAsAbuse, true), + RelationshipMapping(RelationshipType.ReportedAsAbuseBy, true) + ) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/user_activity/BUILD b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/user_activity/BUILD new file mode 100644 index 0000000000..fe3df2d8b4 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/user_activity/BUILD @@ -0,0 +1,20 @@ +scala_library( + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/com/google/inject:guice", + "3rdparty/jvm/com/google/inject/extensions:guice-assistedinject", + "3rdparty/jvm/net/codingwell:scala-guice", + "3rdparty/jvm/org/slf4j:slf4j-api", + "finatra/inject/inject-core/src/main/scala", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/cache", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/strato", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/constants", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/deciders", + "strato/config/columns/onboarding:onboarding-strato-client", + "util/util-slf4j-api/src/main/scala", + ], +) diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/user_activity/UserActivityPredicate.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/user_activity/UserActivityPredicate.scala new file mode 100644 index 0000000000..e0fd6b42ce --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/user_activity/UserActivityPredicate.scala @@ -0,0 +1,161 @@ +package com.twitter.follow_recommendations.common.predicates.user_activity + +import com.twitter.core_workflows.user_model.thriftscala.UserState +import com.twitter.decider.Decider +import com.twitter.decider.RandomRecipient +import com.twitter.finagle.Memcached.Client +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.follow_recommendations.common.base.Predicate +import com.twitter.follow_recommendations.common.base.PredicateResult +import com.twitter.follow_recommendations.common.base.StatsUtil +import com.twitter.follow_recommendations.common.clients.cache.MemcacheClient +import com.twitter.follow_recommendations.common.clients.cache.ThriftEnumOptionBijection +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.follow_recommendations.common.models.FilterReason +import com.twitter.follow_recommendations.configapi.deciders.DeciderKey +import com.twitter.product_mixer.core.model.marshalling.request.HasClientContext +import com.twitter.stitch.Stitch +import com.twitter.strato.generated.client.onboarding.UserRecommendabilityWithLongKeysOnUserClientColumn +import com.twitter.timelines.configapi.HasParams +import javax.inject.Inject +import javax.inject.Singleton + +abstract case class UserStateActivityPredicate( + userRecommendabilityClient: UserRecommendabilityWithLongKeysOnUserClientColumn, + validCandidateStates: Set[UserState], + client: Client, + statsReceiver: StatsReceiver, + decider: Decider = Decider.False) + extends Predicate[(HasParams with HasClientContext, CandidateUser)] { + + private val stats: StatsReceiver = statsReceiver.scope(this.getClass.getSimpleName) + + // client to memcache cluster + val bijection = new ThriftEnumOptionBijection[UserState](UserState.apply) + val memcacheClient = MemcacheClient[Option[UserState]]( + client = client, + dest = "/s/cache/follow_recos_service:twemcaches", + valueBijection = bijection, + ttl = UserActivityPredicateParams.CacheTTL, + statsReceiver = stats.scope("twemcache") + ) + + override def apply( + targetAndCandidate: (HasParams with HasClientContext, CandidateUser) + ): Stitch[PredicateResult] = { + val userRecommendabilityFetcher = userRecommendabilityClient.fetcher + val (_, candidate) = targetAndCandidate + + val deciderKey: String = DeciderKey.EnableExperimentalCaching.toString + val enableDistributedCaching: Boolean = decider.isAvailable(deciderKey, Some(RandomRecipient)) + val userStateStitch: Stitch[Option[UserState]] = + enableDistributedCaching match { + case true => { + memcacheClient.readThrough( + // add a key prefix to address cache key collisions + key = "UserActivityPredicate" + candidate.id.toString, + underlyingCall = () => queryUserRecommendable(candidate.id) + ) + } + case false => queryUserRecommendable(candidate.id) + } + val resultStitch: Stitch[PredicateResult] = + userStateStitch.map { userStateOpt => + userStateOpt match { + case Some(userState) => { + if (validCandidateStates.contains(userState)) { + PredicateResult.Valid + } else { + PredicateResult.Invalid(Set(FilterReason.MinStateNotMet)) + } + } + case None => { + PredicateResult.Invalid(Set(FilterReason.MissingRecommendabilityData)) + } + } + } + + StatsUtil.profileStitch(resultStitch, stats.scope("apply")) + .rescue { + case e: Exception => + stats.scope("rescued").counter(e.getClass.getSimpleName).incr() + Stitch(PredicateResult.Invalid(Set(FilterReason.FailOpen))) + } + } + + def queryUserRecommendable( + userId: Long + ): Stitch[Option[UserState]] = { + val userRecommendabilityFetcher = userRecommendabilityClient.fetcher + userRecommendabilityFetcher.fetch(userId).map { userCandidate => + userCandidate.v.flatMap(_.userState) + } + } +} + +@Singleton +class MinStateUserActivityPredicate @Inject() ( + userRecommendabilityClient: UserRecommendabilityWithLongKeysOnUserClientColumn, + client: Client, + statsReceiver: StatsReceiver) + extends UserStateActivityPredicate( + userRecommendabilityClient, + Set( + UserState.Light, + UserState.HeavyNonTweeter, + UserState.MediumNonTweeter, + UserState.HeavyTweeter, + UserState.MediumTweeter + ), + client, + statsReceiver + ) + +@Singleton +class AllTweeterUserActivityPredicate @Inject() ( + userRecommendabilityClient: UserRecommendabilityWithLongKeysOnUserClientColumn, + client: Client, + statsReceiver: StatsReceiver) + extends UserStateActivityPredicate( + userRecommendabilityClient, + Set( + UserState.HeavyTweeter, + UserState.MediumTweeter + ), + client, + statsReceiver + ) + +@Singleton +class HeavyTweeterUserActivityPredicate @Inject() ( + userRecommendabilityClient: UserRecommendabilityWithLongKeysOnUserClientColumn, + client: Client, + statsReceiver: StatsReceiver) + extends UserStateActivityPredicate( + userRecommendabilityClient, + Set( + UserState.HeavyTweeter + ), + client, + statsReceiver + ) + +@Singleton +class NonNearZeroUserActivityPredicate @Inject() ( + userRecommendabilityClient: UserRecommendabilityWithLongKeysOnUserClientColumn, + client: Client, + statsReceiver: StatsReceiver) + extends UserStateActivityPredicate( + userRecommendabilityClient, + Set( + UserState.New, + UserState.VeryLight, + UserState.Light, + UserState.MediumNonTweeter, + UserState.MediumTweeter, + UserState.HeavyNonTweeter, + UserState.HeavyTweeter + ), + client, + statsReceiver + ) diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/user_activity/UserActivityPredicateParams.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/user_activity/UserActivityPredicateParams.scala new file mode 100644 index 0000000000..57e8d958b1 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/user_activity/UserActivityPredicateParams.scala @@ -0,0 +1,10 @@ +package com.twitter.follow_recommendations.common.predicates.user_activity + +import com.twitter.timelines.configapi.FSParam +import com.twitter.util.Duration + +object UserActivityPredicateParams { + case object HeavyTweeterEnabled + extends FSParam[Boolean]("user_activity_predicate_heavy_tweeter_enabled", false) + val CacheTTL: Duration = Duration.fromHours(6) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/common/AdhocScoreModificationType.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/common/AdhocScoreModificationType.scala new file mode 100644 index 0000000000..23ccd6f4fa --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/common/AdhocScoreModificationType.scala @@ -0,0 +1,20 @@ +package com.twitter.follow_recommendations.common.rankers.common + +/** + * To manage the extent of adhoc score modifications, we set a hard limit that from each of the + * types below *ONLY ONE* adhoc scorer can be applied to candidates' scores. More details about the + * usage is available in [[AdhocRanker]] + */ + +object AdhocScoreModificationType extends Enumeration { + type AdhocScoreModificationType = Value + + // This type of scorer increases the score of a subset of candidates through various policies. + val BoostingScorer: AdhocScoreModificationType = Value("boosting") + + // This type of scorer shuffles candidates randomly according to some distribution. + val WeightedRandomSamplingScorer: AdhocScoreModificationType = Value("weighted_random_sampling") + + // This is added solely for testing purposes and should not be used in production. + val InvalidAdhocScorer: AdhocScoreModificationType = Value("invalid_adhoc_scorer") +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/common/BUILD b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/common/BUILD new file mode 100644 index 0000000000..77d496dcf0 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/common/BUILD @@ -0,0 +1,10 @@ +scala_library( + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/org/slf4j:slf4j-api", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/common", + "util/util-slf4j-api/src/main/scala", + ], +) diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/common/DedupCandidates.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/common/DedupCandidates.scala new file mode 100644 index 0000000000..bbbde2b586 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/common/DedupCandidates.scala @@ -0,0 +1,11 @@ +package com.twitter.follow_recommendations.common.rankers.common + +import com.twitter.product_mixer.core.model.common.UniversalNoun +import scala.collection.mutable + +object DedupCandidates { + def apply[C <: UniversalNoun[Long]](input: Seq[C]): Seq[C] = { + val seen = mutable.HashSet[Long]() + input.filter { candidate => seen.add(candidate.id) } + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/common/RankerId.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/common/RankerId.scala new file mode 100644 index 0000000000..f6fdb905ab --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/common/RankerId.scala @@ -0,0 +1,27 @@ +package com.twitter.follow_recommendations.common.rankers.common + +object RankerId extends Enumeration { + type RankerId = Value + + val RandomRanker: RankerId = Value("random") + // The production PostNUX ML warm-start auto-retraining model ranker + val PostNuxProdRanker: RankerId = Value("postnux_prod") + val None: RankerId = Value("none") + + // Sampling from the Placket-Luce distribution. Applied after ranker step. Its ranker id is mainly used for logging. + val PlacketLuceSamplingTransformer: RankerId = Value("placket_luce_sampling_transformer") + + def getRankerByName(name: String): Option[RankerId] = + RankerId.values.toSeq.find(_.equals(Value(name))) + +} + +/** + * ML model based heavy ranker ids. + */ +object ModelBasedHeavyRankerId { + import RankerId._ + val HeavyRankerIds: Set[String] = Set( + PostNuxProdRanker.toString, + ) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/fatigue_ranker/BUILD b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/fatigue_ranker/BUILD new file mode 100644 index 0000000000..2fce8d77a8 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/fatigue_ranker/BUILD @@ -0,0 +1,13 @@ +scala_library( + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/impression_store", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/common", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/utils", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/common", + ], +) diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/fatigue_ranker/ImpressionBasedFatigueRanker.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/fatigue_ranker/ImpressionBasedFatigueRanker.scala new file mode 100644 index 0000000000..18ac0436b7 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/fatigue_ranker/ImpressionBasedFatigueRanker.scala @@ -0,0 +1,141 @@ +package com.twitter.follow_recommendations.common.rankers.fatigue_ranker + +import com.twitter.finagle.stats.Counter +import com.twitter.finagle.stats.Stat +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.follow_recommendations.common.base.Ranker +import com.twitter.follow_recommendations.common.base.StatsUtil +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.follow_recommendations.common.models.HasDisplayLocation +import com.twitter.follow_recommendations.common.models.HasWtfImpressions +import com.twitter.follow_recommendations.common.models.WtfImpression +import com.twitter.follow_recommendations.common.rankers.common.RankerId.RankerId +import com.twitter.follow_recommendations.common.rankers.utils.Utils +import com.twitter.product_mixer.core.model.marshalling.request.HasClientContext +import com.twitter.servo.util.MemoizingStatsReceiver +import com.twitter.stitch.Stitch +import com.twitter.timelines.configapi.HasParams +import com.twitter.util.Time + +/** + * Ranks candidates based on the given weights for each algorithm while preserving the ranks inside each algorithm. + * Reorders the ranked list based on recent impressions from recentImpressionRepo + * + * Note that the penalty is added to the rank of each candidate. To make producer-side experiments + * with multiple rankers possible, we modify the scores for each candidate and ranker as: + * NewScore(C, R) = -(Rank(C, R) + Impression(C, U) x FatigueFactor), + * where C is a candidate, R a ranker and U the target user. + * Note also that fatigue penalty is independent of any of the rankers. + */ +class ImpressionBasedFatigueRanker[ + Target <: HasClientContext with HasDisplayLocation with HasParams with HasWtfImpressions +]( + fatigueFactor: Int, + statsReceiver: StatsReceiver) + extends Ranker[Target, CandidateUser] { + + val name: String = this.getClass.getSimpleName + val stats = statsReceiver.scope("impression_based_fatigue_ranker") + val droppedStats: MemoizingStatsReceiver = new MemoizingStatsReceiver(stats.scope("hard_drops")) + val impressionStats: StatsReceiver = stats.scope("wtf_impressions") + val noImpressionCounter: Counter = impressionStats.counter("no_impressions") + val oldestImpressionStat: Stat = impressionStats.stat("oldest_sec") + + override def rank(target: Target, candidates: Seq[CandidateUser]): Stitch[Seq[CandidateUser]] = { + StatsUtil.profileStitch( + Stitch.value(rankCandidates(target, candidates)), + stats.scope("rank") + ) + } + + private def trackTimeSinceOldestImpression(impressions: Seq[WtfImpression]): Unit = { + val timeSinceOldest = Time.now - impressions.map(_.latestTime).min + oldestImpressionStat.add(timeSinceOldest.inSeconds) + } + + private def rankCandidates( + target: Target, + candidates: Seq[CandidateUser] + ): Seq[CandidateUser] = { + target.wtfImpressions + .map { wtfImpressions => + if (wtfImpressions.isEmpty) { + noImpressionCounter.incr() + candidates + } else { + val rankerIds = + candidates.flatMap(_.scores.map(_.scores.flatMap(_.rankerId))).flatten.sorted.distinct + + /** + * In below we create a Map from each CandidateUser's ID to a Map from each Ranker that + * the user has a score for, and candidate's corresponding rank when candidates are sorted + * by that Ranker (Only candidates who have this Ranker are considered for ranking). + */ + val candidateRanks: Map[Long, Map[RankerId, Int]] = rankerIds + .flatMap { rankerId => + // Candidates with no scores from this Ranker is first removed to calculate ranks. + val relatedCandidates = + candidates.filter(_.scores.exists(_.scores.exists(_.rankerId.contains(rankerId)))) + relatedCandidates + .sortBy(-_.scores + .flatMap(_.scores.find(_.rankerId.contains(rankerId)).map(_.value)).getOrElse( + 0.0)).zipWithIndex.map { + case (candidate, rank) => (candidate.id, rankerId, rank) + } + }.groupBy(_._1).map { + case (candidate, ranksForAllRankers) => + ( + candidate, + ranksForAllRankers.map { case (_, rankerId, rank) => (rankerId, rank) }.toMap) + } + + val idFatigueCountMap = + wtfImpressions.groupBy(_.candidateId).mapValues(_.map(_.counts).sum) + trackTimeSinceOldestImpression(wtfImpressions) + val rankedCandidates: Seq[CandidateUser] = candidates + .map { candidate => + val candidateImpressions = idFatigueCountMap.getOrElse(candidate.id, 0) + val fatiguedScores = candidate.scores.map { ss => + ss.copy(scores = ss.scores.map { s => + s.rankerId match { + // We set the new score as -rank after fatigue penalty is applied. + case Some(rankerId) => + // If the candidate's ID is not in the candidate->ranks map, or there is no + // rank for this specific ranker and this candidate, we use maximum possible + // rank instead. Note that this indicates that there is a problem. + s.copy(value = -(candidateRanks + .getOrElse(candidate.id, Map()).getOrElse(rankerId, candidates.length) + + candidateImpressions * fatigueFactor)) + // In case a score exists without a RankerId, we pass on the score as is. + case None => s + } + }) + } + candidate.copy(scores = fatiguedScores) + }.zipWithIndex.map { + // We re-rank candidates with their input ordering (which is done by the request-level + // ranker) and fatigue penalty. + case (candidate, inputRank) => + val candidateImpressions = idFatigueCountMap.getOrElse(candidate.id, 0) + (candidate, inputRank + candidateImpressions * fatigueFactor) + }.sortBy(_._2).map(_._1) + // Only populate ranking info when WTF impression info present + val scribeRankingInfo: Boolean = + target.params(ImpressionBasedFatigueRankerParams.ScribeRankingInfoInFatigueRanker) + if (scribeRankingInfo) Utils.addRankingInfo(rankedCandidates, name) else rankedCandidates + } + }.getOrElse(candidates) // no reranking/filtering when wtf impressions not present + } +} + +object ImpressionBasedFatigueRanker { + val DefaultFatigueFactor = 5 + + def build[ + Target <: HasClientContext with HasDisplayLocation with HasParams with HasWtfImpressions + ]( + baseStatsReceiver: StatsReceiver, + fatigueFactor: Int = DefaultFatigueFactor + ): ImpressionBasedFatigueRanker[Target] = + new ImpressionBasedFatigueRanker(fatigueFactor, baseStatsReceiver) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/fatigue_ranker/ImpressionBasedFatigueRankerFSConfig.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/fatigue_ranker/ImpressionBasedFatigueRankerFSConfig.scala new file mode 100644 index 0000000000..34fbbeb466 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/fatigue_ranker/ImpressionBasedFatigueRankerFSConfig.scala @@ -0,0 +1,12 @@ +package com.twitter.follow_recommendations.common.rankers.fatigue_ranker + +import com.twitter.follow_recommendations.configapi.common.FeatureSwitchConfig +import com.twitter.timelines.configapi.FSParam +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class ImpressionBasedFatigueRankerFSConfig @Inject() extends FeatureSwitchConfig { + override val booleanFSParams: Seq[FSParam[Boolean]] = + Seq(ImpressionBasedFatigueRankerParams.ScribeRankingInfoInFatigueRanker) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/fatigue_ranker/ImpressionBasedFatigueRankerParams.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/fatigue_ranker/ImpressionBasedFatigueRankerParams.scala new file mode 100644 index 0000000000..075d78bf64 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/fatigue_ranker/ImpressionBasedFatigueRankerParams.scala @@ -0,0 +1,14 @@ +package com.twitter.follow_recommendations.common.rankers.fatigue_ranker + +import com.twitter.timelines.configapi.FSParam +import com.twitter.timelines.configapi.Param + +object ImpressionBasedFatigueRankerParams { + // Whether to enable hard dropping of impressed candidates + object DropImpressedCandidateEnabled extends Param[Boolean](false) + // At what # of impressions to hard drop candidates. + object DropCandidateImpressionThreshold extends Param[Int](default = 10) + // Whether to scribe candidate ranking/scoring info per ranking stage + object ScribeRankingInfoInFatigueRanker + extends FSParam[Boolean]("fatigue_ranker_scribe_ranking_info", true) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/first_n_ranker/BUILD b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/first_n_ranker/BUILD new file mode 100644 index 0000000000..3de5523b1c --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/first_n_ranker/BUILD @@ -0,0 +1,20 @@ +scala_library( + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/com/google/inject:guice", + "3rdparty/jvm/com/google/inject/extensions:guice-assistedinject", + "3rdparty/jvm/net/codingwell:scala-guice", + "3rdparty/jvm/org/slf4j:slf4j-api", + "finatra/inject/inject-core/src/main/scala", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/constants", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/common", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/utils", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/utils", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/common", + "util/util-slf4j-api/src/main/scala", + ], +) diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/first_n_ranker/FirstNRanker.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/first_n_ranker/FirstNRanker.scala new file mode 100644 index 0000000000..8fbeafa25b --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/first_n_ranker/FirstNRanker.scala @@ -0,0 +1,115 @@ +package com.twitter.follow_recommendations.common.rankers.first_n_ranker + +import com.google.inject.Inject +import com.google.inject.Singleton +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.follow_recommendations.common.base.Ranker +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.follow_recommendations.common.models.HasQualityFactor +import com.twitter.follow_recommendations.common.rankers.utils.Utils +import com.twitter.product_mixer.core.model.common.identifier.CandidateSourceIdentifier +import com.twitter.product_mixer.core.model.marshalling.request.HasClientContext +import com.twitter.stitch.Stitch +import com.twitter.timelines.configapi.HasParams + +/** + * This class is meant to filter candidates between stages of our ranker by taking the first N + * candidates, merging any candidate source information for candidates with multiple entries. + * To allow us to chain this truncation operation any number of times sequentially within the main + * ranking builder, we abstract the truncation as a separate Ranker + */ +@Singleton +class FirstNRanker[Target <: HasClientContext with HasParams with HasQualityFactor] @Inject() ( + stats: StatsReceiver) + extends Ranker[Target, CandidateUser] { + + val name: String = this.getClass.getSimpleName + private val baseStats = stats.scope("first_n_ranker") + val scaledDownByQualityFactorCounter = + baseStats.counter("scaled_down_by_quality_factor") + private val mergeStat = baseStats.scope("merged_candidates") + private val mergeStat2 = mergeStat.counter("2") + private val mergeStat3 = mergeStat.counter("3") + private val mergeStat4 = mergeStat.counter("4+") + private val candidateSizeStats = baseStats.scope("candidate_size") + + private case class CandidateSourceScore( + candidateId: Long, + sourceId: CandidateSourceIdentifier, + score: Option[Double]) + + /** + * Adds the rank of each candidate based on the primary candidate source's score. + * In the event where the provided ordering of candidates do not align with the score, + * we will respect the score, since the ordering might have been mixed up due to other previous + * steps like the shuffleFn in the `WeightedCandidateSourceRanker`. + * @param candidates ordered list of candidates + * @return same ordered list of candidates, but with the rank information appended + */ + def addRank(candidates: Seq[CandidateUser]): Seq[CandidateUser] = { + val candidateSourceRanks = for { + (sourceIdOpt, sourceCandidates) <- candidates.groupBy(_.getPrimaryCandidateSource) + (candidate, rank) <- sourceCandidates.sortBy(-_.score.getOrElse(0.0)).zipWithIndex + } yield { + (candidate, sourceIdOpt) -> rank + } + candidates.map { c => + c.getPrimaryCandidateSource + .map { sourceId => + val sourceRank = candidateSourceRanks((c, c.getPrimaryCandidateSource)) + c.addCandidateSourceRanksMap(Map(sourceId -> sourceRank)) + }.getOrElse(c) + } + } + + override def rank(target: Target, candidates: Seq[CandidateUser]): Stitch[Seq[CandidateUser]] = { + + val scaleDownFactor = Math.max( + target.qualityFactor.getOrElse(1.0d), + target.params(FirstNRankerParams.MinNumCandidatesScoredScaleDownFactor) + ) + + if (scaleDownFactor < 1.0d) + scaledDownByQualityFactorCounter.incr() + + val n = (target.params(FirstNRankerParams.CandidatesToRank) * scaleDownFactor).toInt + val scribeRankingInfo: Boolean = + target.params(FirstNRankerParams.ScribeRankingInfoInFirstNRanker) + candidateSizeStats.counter(s"n$n").incr() + val candidatesWithRank = addRank(candidates) + if (target.params(FirstNRankerParams.GroupDuplicateCandidates)) { + val groupedCandidates: Map[Long, Seq[CandidateUser]] = candidatesWithRank.groupBy(_.id) + val topN = candidates + .map { c => + merge(groupedCandidates(c.id)) + }.distinct.take(n) + Stitch.value(if (scribeRankingInfo) Utils.addRankingInfo(topN, name) else topN) + } else { + Stitch.value( + if (scribeRankingInfo) Utils.addRankingInfo(candidatesWithRank, name).take(n) + else candidatesWithRank.take(n)) + } // for efficiency, if don't need to deduplicate + } + + /** + * we use the primary candidate source of the first entry, and aggregate all of the other entries' + * candidate source scores into the first entry's candidateSourceScores + * @param candidates list of candidates with the same id + * @return a single merged candidate + */ + private[first_n_ranker] def merge(candidates: Seq[CandidateUser]): CandidateUser = { + if (candidates.size == 1) { + candidates.head + } else { + candidates.size match { + case 2 => mergeStat2.incr() + case 3 => mergeStat3.incr() + case i if i >= 4 => mergeStat4.incr() + case _ => + } + val allSources = candidates.flatMap(_.getCandidateSources).toMap + val allRanks = candidates.flatMap(_.getCandidateRanks).toMap + candidates.head.addCandidateSourceScoresMap(allSources).addCandidateSourceRanksMap(allRanks) + } + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/first_n_ranker/FirstNRankerFSConfig.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/first_n_ranker/FirstNRankerFSConfig.scala new file mode 100644 index 0000000000..484738dc18 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/first_n_ranker/FirstNRankerFSConfig.scala @@ -0,0 +1,21 @@ +package com.twitter.follow_recommendations.common.rankers.first_n_ranker + +import javax.inject.Inject +import javax.inject.Singleton +import com.twitter.follow_recommendations.configapi.common.FeatureSwitchConfig +import com.twitter.timelines.configapi.FSBoundedParam +import com.twitter.timelines.configapi.FSParam + +@Singleton +class FirstNRankerFSConfig @Inject() extends FeatureSwitchConfig { + override val booleanFSParams: Seq[FSParam[Boolean]] = + Seq(FirstNRankerParams.ScribeRankingInfoInFirstNRanker) + + override val intFSParams: Seq[FSBoundedParam[Int]] = Seq( + FirstNRankerParams.CandidatesToRank + ) + + override val doubleFSParams: Seq[FSBoundedParam[Double]] = Seq( + FirstNRankerParams.MinNumCandidatesScoredScaleDownFactor + ) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/first_n_ranker/FirstNRankerFeatureSwitchKeys.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/first_n_ranker/FirstNRankerFeatureSwitchKeys.scala new file mode 100644 index 0000000000..682b60fed9 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/first_n_ranker/FirstNRankerFeatureSwitchKeys.scala @@ -0,0 +1,8 @@ +package com.twitter.follow_recommendations.common.rankers.first_n_ranker + +object FirstNRankerFeatureSwitchKeys { + val CandidatePoolSize = "first_n_ranker_candidate_pool_size" + val ScribeRankingInfo = "first_n_ranker_scribe_ranking_info" + val MinNumCandidatesScoredScaleDownFactor = + "first_n_ranker_min_scale_down_factor" +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/first_n_ranker/FirstNRankerParams.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/first_n_ranker/FirstNRankerParams.scala new file mode 100644 index 0000000000..ac65a6ddeb --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/first_n_ranker/FirstNRankerParams.scala @@ -0,0 +1,26 @@ +package com.twitter.follow_recommendations.common.rankers.first_n_ranker + +import com.twitter.timelines.configapi.FSBoundedParam +import com.twitter.timelines.configapi.FSParam +import com.twitter.timelines.configapi.Param + +object FirstNRankerParams { + case object CandidatesToRank + extends FSBoundedParam[Int]( + FirstNRankerFeatureSwitchKeys.CandidatePoolSize, + default = 100, + min = 50, + max = 600) + + case object GroupDuplicateCandidates extends Param[Boolean](true) + case object ScribeRankingInfoInFirstNRanker + extends FSParam[Boolean](FirstNRankerFeatureSwitchKeys.ScribeRankingInfo, true) + + // the minimum of candidates to score in each request. + object MinNumCandidatesScoredScaleDownFactor + extends FSBoundedParam[Double]( + name = FirstNRankerFeatureSwitchKeys.MinNumCandidatesScoredScaleDownFactor, + default = 0.3, + min = 0.1, + max = 1.0) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/interleave_ranker/BUILD b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/interleave_ranker/BUILD new file mode 100644 index 0000000000..e71d8a7abf --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/interleave_ranker/BUILD @@ -0,0 +1,21 @@ +scala_library( + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/com/google/inject:guice", + "3rdparty/jvm/com/google/inject/extensions:guice-assistedinject", + "3rdparty/jvm/net/codingwell:scala-guice", + "3rdparty/jvm/org/slf4j:slf4j-api", + "finatra/inject/inject-core/src/main/scala", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/constants", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/common", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/ml_ranker/ranking", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/utils", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/utils", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/common", + "util/util-slf4j-api/src/main/scala", + ], +) diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/interleave_ranker/InterleaveRanker.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/interleave_ranker/InterleaveRanker.scala new file mode 100644 index 0000000000..973275f51c --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/interleave_ranker/InterleaveRanker.scala @@ -0,0 +1,204 @@ +package com.twitter.follow_recommendations.common.rankers.interleave_ranker + +import com.google.common.annotations.VisibleForTesting +import com.google.inject.Inject +import com.google.inject.Singleton +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.follow_recommendations.common.base.Ranker +import com.twitter.follow_recommendations.common.base.StatsUtil +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.follow_recommendations.common.rankers.common.RankerId +import com.twitter.follow_recommendations.common.rankers.utils.Utils +import com.twitter.stitch.Stitch +import com.twitter.timelines.configapi.HasParams + +@Singleton +class InterleaveRanker[Target <: HasParams] @Inject() ( + statsReceiver: StatsReceiver) + extends Ranker[Target, CandidateUser] { + + val name: String = this.getClass.getSimpleName + private val stats = statsReceiver.scope("interleave_ranker") + private val inputStats = stats.scope("input") + private val interleavingStats = stats.scope("interleave") + + override def rank( + target: Target, + candidates: Seq[CandidateUser] + ): Stitch[Seq[CandidateUser]] = { + StatsUtil.profileStitch( + Stitch.value(rankCandidates(target, candidates)), + stats.scope("rank") + ) + } + + private def rankCandidates( + target: Target, + candidates: Seq[CandidateUser] + ): Seq[CandidateUser] = { + + /** + * By this stage, all valid candidates should have: + * 1. Their Scores field populated. + * 2. Their selectedRankerId set. + * 3. Have a score associated to their selectedRankerId. + * If there is any candidate that doesn't meet the conditions above, there is a problem in one + * of the previous rankers. Since no new scoring is done in this ranker, we simply remove them. + */ + val validCandidates = + candidates.filter { c => + c.scores.isDefined && + c.scores.exists(_.selectedRankerId.isDefined) && + getCandidateScoreByRankerId(c, c.scores.flatMap(_.selectedRankerId)).isDefined + } + + // To monitor the percentage of valid candidates, as defined above, we track the following: + inputStats.counter("candidates_with_no_scores").incr(candidates.count(_.scores.isEmpty)) + inputStats + .counter("candidates_with_no_selected_ranker").incr(candidates.count { c => + c.scores.isEmpty || c.scores.exists(_.selectedRankerId.isEmpty) + }) + inputStats + .counter("candidates_with_no_score_for_selected_ranker").incr(candidates.count { c => + c.scores.isEmpty || + c.scores.exists(_.selectedRankerId.isEmpty) || + getCandidateScoreByRankerId(c, c.scores.flatMap(_.selectedRankerId)).isEmpty + }) + inputStats.counter("total_num_candidates").incr(candidates.length) + inputStats.counter("total_valid_candidates").incr(validCandidates.length) + + // We only count rankerIds from those candidates who are valid to exclude those candidates with + // a valid selectedRankerId that don't have an associated score for it. + val rankerIds = validCandidates.flatMap(_.scores.flatMap(_.selectedRankerId)).sorted.distinct + rankerIds.foreach { rankerId => + inputStats + .counter(s"valid_scores_for_${rankerId.toString}").incr( + candidates.count(getCandidateScoreByRankerId(_, Some(rankerId)).isDefined)) + inputStats.counter(s"total_candidates_for_${rankerId.toString}").incr(candidates.length) + } + inputStats.counter(s"num_ranker_ids=${rankerIds.length}").incr() + val scribeRankingInfo: Boolean = + target.params(InterleaveRankerParams.ScribeRankingInfoInInterleaveRanker) + if (rankerIds.length <= 1) + // In the case of "Number of RankerIds = 0", we pass on the candidates even though there is + // a problem in a previous ranker that provided the scores. + if (scribeRankingInfo) Utils.addRankingInfo(candidates, name) else candidates + else + if (scribeRankingInfo) + Utils.addRankingInfo(interleaveCandidates(validCandidates, rankerIds), name) + else interleaveCandidates(validCandidates, rankerIds) + } + + @VisibleForTesting + private[interleave_ranker] def interleaveCandidates( + candidates: Seq[CandidateUser], + rankerIds: Seq[RankerId.RankerId] + ): Seq[CandidateUser] = { + val candidatesWithRank = rankerIds + .flatMap { ranker => + candidates + // We first sort all candidates using this ranker. + .sortBy(-getCandidateScoreByRankerId(_, Some(ranker)).getOrElse(Double.MinValue)) + .zipWithIndex.filter( + // but only hold those candidates whose selected ranker is this ranker. + // These ranks will be forced in the final ordering. + _._1.scores.flatMap(_.selectedRankerId).contains(ranker)) + } + + // Only candidates who have isInProducerScoringExperiment set to true will have their position enforced. We + // separate candidates into two groups: (1) Production and (2) Experiment. + val (expCandidates, prodCandidates) = + candidatesWithRank.partition(_._1.scores.exists(_.isInProducerScoringExperiment)) + + // We resolve (potential) conflicts between the enforced ranks of experimental models. + val expCandidatesFinalPos = resolveConflicts(expCandidates) + + // Retrieve non-occupied positions and assign them to candidates who use production ranker. + val occupiedPos = expCandidatesFinalPos.map(_._2).toSet + val prodCandidatesFinalPos = + prodCandidates + .map(_._1).zip( + candidates.indices.filterNot(occupiedPos.contains).sorted.take(prodCandidates.length)) + + // Merge the two groups and sort them by their corresponding positions. + val finalCandidates = (prodCandidatesFinalPos ++ expCandidatesFinalPos).sortBy(_._2).map(_._1) + + // We count the presence of each ranker in the top-3 final positions. + finalCandidates.zip(0 until 3).foreach { + case (c, r) => + // We only do so for candidates that are in a producer-side experiment. + if (c.scores.exists(_.isInProducerScoringExperiment)) + c.scores.flatMap(_.selectedRankerId).map(_.toString).foreach { rankerName => + interleavingStats + .counter(s"num_final_position_${r}_$rankerName") + .incr() + } + } + + finalCandidates + } + + @VisibleForTesting + private[interleave_ranker] def resolveConflicts( + candidatesWithRank: Seq[(CandidateUser, Int)] + ): Seq[(CandidateUser, Int)] = { + // The following two metrics will allow us to calculate the rate of conflicts occurring. + // Example: If overall there are 10 producers in different bucketing experiments, and 3 of them + // are assigned to the same position. The rate would be 3/10, 30%. + val numCandidatesWithConflicts = interleavingStats.counter("candidates_with_conflict") + val numCandidatesNoConflicts = interleavingStats.counter("candidates_without_conflict") + val candidatesGroupedByRank = candidatesWithRank.groupBy(_._2).toSeq.sortBy(_._1).map { + case (rank, candidatesWithRank) => (rank, candidatesWithRank.map(_._1)) + } + + candidatesGroupedByRank.foldLeft(Seq[(CandidateUser, Int)]()) { (upToHere, nextGroup) => + val (rank, candidates) = nextGroup + if (candidates.length > 1) + numCandidatesWithConflicts.incr(candidates.length) + else + numCandidatesNoConflicts.incr() + + // We use the position after the last-assigned candidate as a starting point, or 0 otherwise. + // If candidates' position is after this "starting point", we enforce that position instead. + val minAvailableIndex = scala.math.max(upToHere.lastOption.map(_._2).getOrElse(-1) + 1, rank) + val enforcedPos = + (minAvailableIndex until minAvailableIndex + candidates.length).toList + val shuffledEnforcedPos = + if (candidates.length > 1) scala.util.Random.shuffle(enforcedPos) else enforcedPos + if (shuffledEnforcedPos.length > 1) { + candidates.zip(shuffledEnforcedPos).sortBy(_._2).map(_._1).zipWithIndex.foreach { + case (c, r) => + c.scores.flatMap(_.selectedRankerId).map(_.toString).foreach { rankerName => + // For each ranker, we count the total number of times it has been in a conflict. + interleavingStats + .counter(s"num_${shuffledEnforcedPos.length}-way_conflicts_$rankerName") + .incr() + // We also count the positions each of the rankers have fallen randomly into. In any + // experiment this should converge to uniform distribution given enough occurrences. + // Note that the position here is relative to the other candidates in the conflict and + // not the overall position of each candidate. + interleavingStats + .counter( + s"num_position_${r}_after_${shuffledEnforcedPos.length}-way_conflict_$rankerName") + .incr() + } + } + } + upToHere ++ candidates.zip(shuffledEnforcedPos).sortBy(_._2) + } + } + + @VisibleForTesting + private[interleave_ranker] def getCandidateScoreByRankerId( + candidate: CandidateUser, + rankerIdOpt: Option[RankerId.RankerId] + ): Option[Double] = { + rankerIdOpt match { + case None => None + case Some(rankerId) => + candidate.scores.flatMap { + _.scores.find(_.rankerId.contains(rankerId)).map(_.value) + } + } + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/interleave_ranker/InterleaveRankerFSConfig.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/interleave_ranker/InterleaveRankerFSConfig.scala new file mode 100644 index 0000000000..5a8c6de2a3 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/interleave_ranker/InterleaveRankerFSConfig.scala @@ -0,0 +1,12 @@ +package com.twitter.follow_recommendations.common.rankers.interleave_ranker + +import javax.inject.Inject +import javax.inject.Singleton +import com.twitter.follow_recommendations.configapi.common.FeatureSwitchConfig +import com.twitter.timelines.configapi.FSParam + +@Singleton +class InterleaveRankerFSConfig @Inject() extends FeatureSwitchConfig { + override val booleanFSParams: Seq[FSParam[Boolean]] = + Seq(InterleaveRankerParams.ScribeRankingInfoInInterleaveRanker) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/interleave_ranker/InterleaveRankerParams.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/interleave_ranker/InterleaveRankerParams.scala new file mode 100644 index 0000000000..84e6ea314a --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/interleave_ranker/InterleaveRankerParams.scala @@ -0,0 +1,8 @@ +package com.twitter.follow_recommendations.common.rankers.interleave_ranker + +import com.twitter.timelines.configapi.FSParam + +object InterleaveRankerParams { + case object ScribeRankingInfoInInterleaveRanker + extends FSParam[Boolean]("interleave_ranker_scribe_ranking_info", true) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/ml_ranker/ranking/BUILD b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/ml_ranker/ranking/BUILD new file mode 100644 index 0000000000..0c277e0059 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/ml_ranker/ranking/BUILD @@ -0,0 +1,37 @@ +scala_library( + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/com/google/inject:guice", + "3rdparty/jvm/com/google/inject/extensions:guice-assistedinject", + "3rdparty/jvm/net/codingwell:scala-guice", + "3rdparty/jvm/org/slf4j:slf4j-api", + "finatra/inject/inject-core/src/main/scala", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/deepbirdv2", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/constants", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/sources", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/common", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/ml_ranker/scoring", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/utils", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/utils", + "src/java/com/twitter/ml/api:api-base", + "util/util-slf4j-api/src/main/scala", + ], +) + +# This is to import only the params from MlRanker, for instance to get request-level heavy ranker. +scala_library( + name = "ml_ranker_params", + sources = [ + "MlRankerParams.scala", + ], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/common", + "timelines/src/main/scala/com/twitter/timelines/config/configapi", + ], +) diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/ml_ranker/ranking/HydrateFeaturesTransform.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/ml_ranker/ranking/HydrateFeaturesTransform.scala new file mode 100644 index 0000000000..499a14b67b --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/ml_ranker/ranking/HydrateFeaturesTransform.scala @@ -0,0 +1,57 @@ +package com.twitter.follow_recommendations.common.rankers.ml_ranker.ranking + +import com.google.inject.Inject +import com.google.inject.Singleton +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.follow_recommendations.common.base.GatedTransform +import com.twitter.follow_recommendations.common.base.StatsUtil.profileStitchMapResults +import com.twitter.follow_recommendations.common.feature_hydration.common.HasPreFetchedFeature +import com.twitter.follow_recommendations.common.feature_hydration.sources.UserScoringFeatureSource +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.follow_recommendations.common.models.HasDebugOptions +import com.twitter.follow_recommendations.common.models.HasDisplayLocation +import com.twitter.follow_recommendations.common.models.HasSimilarToContext +import com.twitter.follow_recommendations.common.models.RichDataRecord +import com.twitter.ml.api.DataRecord +import com.twitter.product_mixer.core.model.marshalling.request.HasClientContext +import com.twitter.stitch.Stitch +import com.twitter.timelines.configapi.HasParams +import com.twitter.util.logging.Logging + +/** + * Hydrate features given target and candidates lists. + * This is a required step before MlRanker. + * If a feature is not hydrated before MlRanker is triggered, a runtime exception will be thrown + */ +@Singleton +class HydrateFeaturesTransform[ + Target <: HasClientContext with HasParams with HasDebugOptions with HasPreFetchedFeature with HasSimilarToContext with HasDisplayLocation] @Inject() ( + userScoringFeatureSource: UserScoringFeatureSource, + stats: StatsReceiver) + extends GatedTransform[Target, CandidateUser] + with Logging { + + private val hydrateFeaturesStats = stats.scope("hydrate_features") + + def transform(target: Target, candidates: Seq[CandidateUser]): Stitch[Seq[CandidateUser]] = { + // get features + val featureMapStitch: Stitch[Map[CandidateUser, DataRecord]] = + profileStitchMapResults( + userScoringFeatureSource.hydrateFeatures(target, candidates), + hydrateFeaturesStats) + + featureMapStitch.map { featureMap => + candidates + .map { candidate => + val dataRecord = featureMap(candidate) + // add debugRecord only when the request parameter is set + val debugDataRecord = if (target.debugOptions.exists(_.fetchDebugInfo)) { + Some(candidate.toDebugDataRecord(dataRecord, userScoringFeatureSource.featureContext)) + } else None + candidate.copy( + dataRecord = Some(RichDataRecord(Some(dataRecord), debugDataRecord)) + ) + } + } + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/ml_ranker/ranking/MlRanker.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/ml_ranker/ranking/MlRanker.scala new file mode 100644 index 0000000000..6344c348ee --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/ml_ranker/ranking/MlRanker.scala @@ -0,0 +1,219 @@ +package com.twitter.follow_recommendations.common.rankers.ml_ranker.ranking + +import com.google.common.annotations.VisibleForTesting +import com.google.inject.Inject +import com.google.inject.Singleton +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.follow_recommendations.common.base.Ranker +import com.twitter.follow_recommendations.common.base.StatsUtil +import com.twitter.follow_recommendations.common.base.StatsUtil.profileSeqResults +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.follow_recommendations.common.models.HasDisplayLocation +import com.twitter.follow_recommendations.common.models.HasDebugOptions +import com.twitter.follow_recommendations.common.models.Scores +import com.twitter.follow_recommendations.common.rankers.common.RankerId +import com.twitter.follow_recommendations.common.rankers.common.RankerId.RankerId +import com.twitter.follow_recommendations.common.rankers.utils.Utils +import com.twitter.follow_recommendations.common.rankers.ml_ranker.scoring.AdhocScorer +import com.twitter.follow_recommendations.common.rankers.ml_ranker.scoring.Scorer +import com.twitter.follow_recommendations.common.rankers.ml_ranker.scoring.ScorerFactory +import com.twitter.follow_recommendations.common.utils.CollectionUtil +import com.twitter.ml.api.DataRecord +import com.twitter.product_mixer.core.model.marshalling.request.HasClientContext +import com.twitter.stitch.Stitch +import com.twitter.timelines.configapi.HasParams +import com.twitter.timelines.configapi.Params +import com.twitter.util.logging.Logging + +/** + * This class has a rank function that will perform 4 steps: + * - choose which scorer to use for each candidate + * - score candidates given their respective features + * - add scoring information to the candidate + * - sort candidates by their respective scores + * The feature source and scorer will depend on the request's params + */ +@Singleton +class MlRanker[ + Target <: HasClientContext with HasParams with HasDisplayLocation with HasDebugOptions] @Inject() ( + scorerFactory: ScorerFactory, + statsReceiver: StatsReceiver) + extends Ranker[Target, CandidateUser] + with Logging { + + private val stats: StatsReceiver = statsReceiver.scope("ml_ranker") + + private val inputStat = stats.scope("1_input") + private val selectScorerStat = stats.scope("2_select_scorer") + private val scoreStat = stats.scope("3_score") + + override def rank( + target: Target, + candidates: Seq[CandidateUser] + ): Stitch[Seq[CandidateUser]] = { + profileSeqResults(candidates, inputStat) + val requestRankerId = target.params(MlRankerParams.RequestScorerIdParam) + val rankerIds = chooseRankerByCandidate(candidates, requestRankerId) + + val scoreStitch = score(candidates, rankerIds, requestRankerId).map { scoredCandidates => + { + // sort the candidates by score + val sortedCandidates = sort(target, scoredCandidates) + // add scribe field to candidates (if applicable) and return candidates + scribeCandidates(target, sortedCandidates) + } + } + StatsUtil.profileStitch(scoreStitch, stats.scope("rank")) + } + + /** + * @param target: The WTF request for a given consumer. + * @param candidates A list of candidates considered for recommendation. + * @return A map from each candidate to a tuple that includes: + * (1) The selected scorer that should be used to rank this candidate + * (2) a flag determining whether the candidate is in a producer-side experiment. + */ + private[ranking] def chooseRankerByCandidate( + candidates: Seq[CandidateUser], + requestRankerId: RankerId + ): Map[CandidateUser, RankerId] = { + candidates.map { candidate => + val selectedCandidateRankerId = + if (candidate.params == Params.Invalid || candidate.params == Params.Empty) { + selectScorerStat.counter("candidate_params_empty").incr() + requestRankerId + } else { + val candidateRankerId = candidate.params(MlRankerParams.CandidateScorerIdParam) + if (candidateRankerId == RankerId.None) { + // This candidate is a not part of any producer-side experiment. + selectScorerStat.counter("default_to_request_ranker").incr() + requestRankerId + } else { + // This candidate is in a treatment bucket of a producer-side experiment. + selectScorerStat.counter("use_candidate_ranker").incr() + candidateRankerId + } + } + selectScorerStat.scope("selected").counter(selectedCandidateRankerId.toString).incr() + candidate -> selectedCandidateRankerId + }.toMap + } + + @VisibleForTesting + private[ranking] def score( + candidates: Seq[CandidateUser], + rankerIds: Map[CandidateUser, RankerId], + requestRankerId: RankerId + ): Stitch[Seq[CandidateUser]] = { + val features = candidates.map(_.dataRecord.flatMap(_.dataRecord)) + + require(features.forall(_.nonEmpty), "features are not hydrated for all the candidates") + + val scorers = scorerFactory.getScorers(rankerIds.values.toSeq.sorted.distinct) + + // Scorers are split into ML-based and Adhoc (defined as a scorer that does not need to call an + // ML prediction service and scores candidates using locally-available data). + val (adhocScorers, mlScorers) = scorers.partition { + case _: AdhocScorer => true + case _ => false + } + + // score candidates + val scoresStitch = score(features.map(_.get), mlScorers) + val candidatesWithMlScoresStitch = scoresStitch.map { scoresSeq => + candidates + .zip(scoresSeq).map { // copy datarecord and score into candidate object + case (candidate, scores) => + val selectedRankerId = rankerIds(candidate) + val useRequestRanker = + candidate.params == Params.Invalid || + candidate.params == Params.Empty || + candidate.params(MlRankerParams.CandidateScorerIdParam) == RankerId.None + candidate.copy( + score = scores.scores.find(_.rankerId.contains(requestRankerId)).map(_.value), + scores = if (scores.scores.nonEmpty) { + Some( + scores.copy( + scores = scores.scores, + selectedRankerId = Some(selectedRankerId), + isInProducerScoringExperiment = !useRequestRanker + )) + } else None + ) + } + } + + candidatesWithMlScoresStitch.map { candidates => + // The basis for adhoc scores are the "request-level" ML ranker. We add the base score here + // while adhoc scorers are applied in [[AdhocRanker]]. + addMlBaseScoresForAdhocScorers(candidates, requestRankerId, adhocScorers) + } + } + + @VisibleForTesting + private[ranking] def addMlBaseScoresForAdhocScorers( + candidates: Seq[CandidateUser], + requestRankerId: RankerId, + adhocScorers: Seq[Scorer] + ): Seq[CandidateUser] = { + candidates.map { candidate => + candidate.scores match { + case Some(oldScores) => + // 1. We fetch the ML score that is the basis of adhoc scores: + val baseMlScoreOpt = Utils.getCandidateScoreByRankerId(candidate, requestRankerId) + + // 2. For each adhoc scorer, we copy the ML score object, changing only the ID and type. + val newScores = adhocScorers flatMap { adhocScorer => + baseMlScoreOpt.map( + _.copy(rankerId = Some(adhocScorer.id), scoreType = adhocScorer.scoreType)) + } + + // 3. We add the new adhoc score entries to the candidate. + candidate.copy(scores = Some(oldScores.copy(scores = oldScores.scores ++ newScores))) + case _ => + // Since there is no base ML score, there should be no adhoc score modification as well. + candidate + } + } + } + + private[this] def score( + dataRecords: Seq[DataRecord], + scorers: Seq[Scorer] + ): Stitch[Seq[Scores]] = { + val scoredResponse = scorers.map { scorer => + StatsUtil.profileStitch(scorer.score(dataRecords), scoreStat.scope(scorer.id.toString)) + } + // If we could score a candidate with too many rankers, it is likely to blow up the whole system. + // and fail back to default production model + StatsUtil.profileStitch(Stitch.collect(scoredResponse), scoreStat).map { scoresByScorerId => + CollectionUtil.transposeLazy(scoresByScorerId).map { scoresPerCandidate => + Scores(scoresPerCandidate) + } + } + } + + // sort candidates using score in descending order + private[this] def sort( + target: Target, + candidates: Seq[CandidateUser] + ): Seq[CandidateUser] = { + candidates.sortBy(c => -c.score.getOrElse(MlRanker.DefaultScore)) + } + + private[this] def scribeCandidates( + target: Target, + candidates: Seq[CandidateUser] + ): Seq[CandidateUser] = { + val scribeRankingInfo: Boolean = target.params(MlRankerParams.ScribeRankingInfoInMlRanker) + scribeRankingInfo match { + case true => Utils.addRankingInfo(candidates, "MlRanker") + case false => candidates + } + } +} + +object MlRanker { + // this is to ensure candidates with absent scores are ranked the last + val DefaultScore: Double = Double.MinValue +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/ml_ranker/ranking/MlRankerFSConfig.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/ml_ranker/ranking/MlRankerFSConfig.scala new file mode 100644 index 0000000000..c69a32fc51 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/ml_ranker/ranking/MlRankerFSConfig.scala @@ -0,0 +1,12 @@ +package com.twitter.follow_recommendations.common.rankers.ml_ranker.ranking + +import javax.inject.Inject +import javax.inject.Singleton +import com.twitter.follow_recommendations.configapi.common.FeatureSwitchConfig +import com.twitter.timelines.configapi.FSParam + +@Singleton +class MlRankerFSConfig @Inject() extends FeatureSwitchConfig { + override val booleanFSParams: Seq[FSParam[Boolean]] = + Seq(MlRankerParams.ScribeRankingInfoInMlRanker) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/ml_ranker/ranking/MlRankerParams.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/ml_ranker/ranking/MlRankerParams.scala new file mode 100644 index 0000000000..8463963a69 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/ml_ranker/ranking/MlRankerParams.scala @@ -0,0 +1,30 @@ +package com.twitter.follow_recommendations.common.rankers.ml_ranker.ranking + +import com.twitter.follow_recommendations.common.rankers.common.RankerId +import com.twitter.timelines.configapi.FSEnumParam +import com.twitter.timelines.configapi.FSParam + +/** + * When adding Producer side experiments, make sure to register the FS Key in [[ProducerFeatureFilter]] + * in [[FeatureSwitchesModule]], otherwise, the FS will not work. + */ +object MlRankerParams { + // which ranker to use by default for the given request + case object RequestScorerIdParam + extends FSEnumParam[RankerId.type]( + name = "post_nux_ml_flow_ml_ranker_id", + default = RankerId.PostNuxProdRanker, + enum = RankerId + ) + + // which ranker to use for the given candidate + case object CandidateScorerIdParam + extends FSEnumParam[RankerId.type]( + name = "post_nux_ml_flow_candidate_user_scorer_id", + default = RankerId.None, + enum = RankerId + ) + + case object ScribeRankingInfoInMlRanker + extends FSParam[Boolean]("post_nux_ml_flow_scribe_ranking_info_in_ml_ranker", true) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/ml_ranker/scoring/AdhocScorer.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/ml_ranker/scoring/AdhocScorer.scala new file mode 100644 index 0000000000..39921bb717 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/ml_ranker/scoring/AdhocScorer.scala @@ -0,0 +1,28 @@ +package com.twitter.follow_recommendations.common.rankers.ml_ranker.scoring + +import com.twitter.follow_recommendations.common.rankers.common.AdhocScoreModificationType.AdhocScoreModificationType +import com.twitter.follow_recommendations.common.models.Score +import com.twitter.ml.api.DataRecord +import com.twitter.stitch.Stitch + +trait AdhocScorer extends Scorer { + + /** + * NOTE: For instances of [[AdhocScorer]] this function SHOULD NOT be used. + * Please use: + * [[score(target: HasClientContext with HasParams, candidates: Seq[CandidateUser])]] + * instead. + */ + @Deprecated + override def score(records: Seq[DataRecord]): Stitch[Seq[Score]] = + throw new UnsupportedOperationException( + "For instances of AdhocScorer this operation is not defined. Please use " + + "`def score(target: HasClientContext with HasParams, candidates: Seq[CandidateUser])` " + + "instead.") + + /** + * This helps us manage the extend of adhoc modification on candidates' score. There is a hard + * limit of applying ONLY ONE scorer of each type to a score. + */ + val scoreModificationType: AdhocScoreModificationType +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/ml_ranker/scoring/BUILD b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/ml_ranker/scoring/BUILD new file mode 100644 index 0000000000..bbcd3c7088 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/ml_ranker/scoring/BUILD @@ -0,0 +1,23 @@ +scala_library( + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/com/google/inject:guice", + "3rdparty/jvm/com/google/inject/extensions:guice-assistedinject", + "3rdparty/jvm/net/codingwell:scala-guice", + "3rdparty/jvm/org/slf4j:slf4j-api", + "finatra/inject/inject-core/src/main/scala", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/deepbirdv2", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/constants", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/common", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/ml_ranker/ranking:ml_ranker_params", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/utils", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/common", + "src/java/com/twitter/ml/api:api-base", + "src/scala/com/twitter/pluck/source/core_workflows/user_model:condensed_user_state-scala", + "util/util-slf4j-api/src/main/scala", + ], +) diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/ml_ranker/scoring/DeepbirdScorer.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/ml_ranker/scoring/DeepbirdScorer.scala new file mode 100644 index 0000000000..d27bc6e37f --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/ml_ranker/scoring/DeepbirdScorer.scala @@ -0,0 +1,151 @@ +package com.twitter.follow_recommendations.common.rankers.ml_ranker.scoring + +import com.twitter.cortex.deepbird.thriftjava.DeepbirdPredictionService +import com.twitter.cortex.deepbird.thriftjava.ModelSelector +import com.twitter.finagle.stats.Stat +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.follow_recommendations.common.models.HasDebugOptions +import com.twitter.follow_recommendations.common.models.HasDisplayLocation +import com.twitter.follow_recommendations.common.models.Score +import com.twitter.ml.api.DataRecord +import com.twitter.ml.api.Feature +import com.twitter.ml.api.RichDataRecord +import com.twitter.ml.prediction_service.{BatchPredictionRequest => JBatchPredictionRequest} +import com.twitter.product_mixer.core.model.marshalling.request.HasClientContext +import com.twitter.stitch.Stitch +import com.twitter.timelines.configapi.HasParams +import com.twitter.util.Future +import com.twitter.util.TimeoutException +import scala.collection.JavaConversions._ +import scala.collection.JavaConverters._ + +/** + * Generic trait that implements the scoring given a deepbirdClient + * To test out a new model, create a scorer extending this trait, override the modelName and inject the scorer + */ +trait DeepbirdScorer extends Scorer { + def modelName: String + def predictionFeature: Feature.Continuous + // Set a default batchSize of 100 when making model prediction calls to the Deepbird V2 prediction server + def batchSize: Int = 100 + def deepbirdClient: DeepbirdPredictionService.ServiceToClient + def baseStats: StatsReceiver + + def modelSelector: ModelSelector = new ModelSelector().setId(modelName) + def stats: StatsReceiver = baseStats.scope(this.getClass.getSimpleName).scope(modelName) + + private def requestCount = stats.counter("requests") + private def emptyRequestCount = stats.counter("empty_requests") + private def successCount = stats.counter("success") + private def failureCount = stats.counter("failures") + private def inputRecordsStat = stats.stat("input_records") + private def outputRecordsStat = stats.stat("output_records") + + // Counters for tracking batch-prediction statistics when making DBv2 prediction calls + // + // numBatchRequests tracks the number of batch prediction requests made to DBv2 prediction servers + private def numBatchRequests = stats.counter("batches") + // numEmptyBatchRequests tracks the number of batch prediction requests made to DBv2 prediction servers + // that had an empty input DataRecord + private def numEmptyBatchRequests = stats.counter("empty_batches") + // numTimedOutBatchRequests tracks the number of batch prediction requests made to DBv2 prediction servers + // that had timed-out + private def numTimedOutBatchRequests = stats.counter("timeout_batches") + + private def batchPredictionLatency = stats.stat("batch_prediction_latency") + private def predictionLatency = stats.stat("prediction_latency") + + private def numEmptyModelPredictions = stats.counter("empty_model_predictions") + private def numNonEmptyModelPredictions = stats.counter("non_empty_model_predictions") + + private val DefaultPredictionScore = 0.0 + + /** + * NOTE: For instances of [[DeepbirdScorer]] this function SHOULD NOT be used. + * Please use [[score(records: Seq[DataRecord])]] instead. + */ + @Deprecated + def score( + target: HasClientContext with HasParams with HasDisplayLocation with HasDebugOptions, + candidates: Seq[CandidateUser] + ): Seq[Option[Score]] = + throw new UnsupportedOperationException( + "For instances of DeepbirdScorer this operation is not defined. Please use " + + "`def score(records: Seq[DataRecord]): Stitch[Seq[Score]]` " + + "instead.") + + override def score(records: Seq[DataRecord]): Stitch[Seq[Score]] = { + requestCount.incr() + if (records.isEmpty) { + emptyRequestCount.incr() + Stitch.Nil + } else { + inputRecordsStat.add(records.size) + Stitch.callFuture( + batchPredict(records, batchSize) + .map { recordList => + val scores = recordList.map { record => + Score( + value = record.getOrElse(DefaultPredictionScore), + rankerId = Some(id), + scoreType = scoreType) + } + outputRecordsStat.add(scores.size) + scores + }.onSuccess(_ => successCount.incr()) + .onFailure(_ => failureCount.incr())) + } + } + + def batchPredict( + dataRecords: Seq[DataRecord], + batchSize: Int + ): Future[Seq[Option[Double]]] = { + Stat + .timeFuture(predictionLatency) { + val batchedDataRecords = dataRecords.grouped(batchSize).toSeq + numBatchRequests.incr(batchedDataRecords.size) + Future + .collect(batchedDataRecords.map(batch => predict(batch))) + .map(res => res.reduce(_ ++ _)) + } + } + + def predict(dataRecords: Seq[DataRecord]): Future[Seq[Option[Double]]] = { + Stat + .timeFuture(batchPredictionLatency) { + if (dataRecords.isEmpty) { + numEmptyBatchRequests.incr() + Future.Nil + } else { + deepbirdClient + .batchPredictFromModel(new JBatchPredictionRequest(dataRecords.asJava), modelSelector) + .map { response => + response.predictions.toSeq.map { prediction => + val predictionFeatureOption = Option( + new RichDataRecord(prediction).getFeatureValue(predictionFeature) + ) + predictionFeatureOption match { + case Some(predictionValue) => + numNonEmptyModelPredictions.incr() + Option(predictionValue.toDouble) + case None => + numEmptyModelPredictions.incr() + Option(DefaultPredictionScore) + } + } + } + .rescue { + case e: TimeoutException => // DBv2 prediction calls that timed out + numTimedOutBatchRequests.incr() + stats.counter(e.getClass.getSimpleName).incr() + Future.value(dataRecords.map(_ => Option(DefaultPredictionScore))) + case e: Exception => // other generic DBv2 prediction call failures + stats.counter(e.getClass.getSimpleName).incr() + Future.value(dataRecords.map(_ => Option(DefaultPredictionScore))) + } + } + } + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/ml_ranker/scoring/PostnuxDeepbirdProdScorer.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/ml_ranker/scoring/PostnuxDeepbirdProdScorer.scala new file mode 100644 index 0000000000..861d02fa74 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/ml_ranker/scoring/PostnuxDeepbirdProdScorer.scala @@ -0,0 +1,34 @@ +package com.twitter.follow_recommendations.common.rankers.ml_ranker.scoring + +import com.twitter.cortex.deepbird.thriftjava.DeepbirdPredictionService +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.follow_recommendations.common.constants.GuiceNamedConstants +import com.twitter.follow_recommendations.common.rankers.common.RankerId +import com.twitter.ml.api.Feature +import javax.inject.Inject +import javax.inject.Named +import javax.inject.Singleton + +// This is a standard DeepbirdV2 ML Ranker scoring config that should be extended by all ML scorers +// +// Only modify this trait when adding new fields to DeepbirdV2 scorers which +trait DeepbirdProdScorer extends DeepbirdScorer { + override val batchSize = 20 +} + +// Feature.Continuous("prediction") is specific to ClemNet architecture, we can change it to be more informative in the next iteration +trait PostNuxV1DeepbirdProdScorer extends DeepbirdProdScorer { + override val predictionFeature: Feature.Continuous = + new Feature.Continuous("prediction") +} + +// The current, primary PostNUX DeepbirdV2 scorer used in production +@Singleton +class PostnuxDeepbirdProdScorer @Inject() ( + @Named(GuiceNamedConstants.WTF_PROD_DEEPBIRDV2_CLIENT) + override val deepbirdClient: DeepbirdPredictionService.ServiceToClient, + override val baseStats: StatsReceiver) + extends PostNuxV1DeepbirdProdScorer { + override val id = RankerId.PostNuxProdRanker + override val modelName = "PostNUX14531GafClemNetWarmStart" +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/ml_ranker/scoring/RandomScorer.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/ml_ranker/scoring/RandomScorer.scala new file mode 100644 index 0000000000..92265cc6b9 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/ml_ranker/scoring/RandomScorer.scala @@ -0,0 +1,42 @@ +package com.twitter.follow_recommendations.common.rankers.ml_ranker.scoring + +import com.twitter.cortex.deepbird.thriftjava.DeepbirdPredictionService +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.follow_recommendations.common.constants.GuiceNamedConstants +import com.twitter.follow_recommendations.common.rankers.common.RankerId +import com.twitter.ml.api.DataRecord +import com.twitter.ml.api.Feature +import com.twitter.util.Future +import javax.inject.Inject +import javax.inject.Named +import javax.inject.Singleton + +/** + * This scorer assigns random values between 0 and 1 to each candidate as scores. + */ + +@Singleton +class RandomScorer @Inject() ( + @Named(GuiceNamedConstants.WTF_PROD_DEEPBIRDV2_CLIENT) + override val deepbirdClient: DeepbirdPredictionService.ServiceToClient, + override val baseStats: StatsReceiver) + extends DeepbirdScorer { + override val id = RankerId.RandomRanker + private val rnd = new scala.util.Random(System.currentTimeMillis()) + + override def predict(dataRecords: Seq[DataRecord]): Future[Seq[Option[Double]]] = { + if (dataRecords.isEmpty) { + Future.Nil + } else { + // All candidates are assigned a random value between 0 and 1 as score. + Future.value(dataRecords.map(_ => Option(rnd.nextDouble()))) + } + } + + override val modelName = "PostNuxRandomRanker" + + // This is not needed since we are overriding the `predict` function, but we have to override + // `predictionFeature` anyway. + override val predictionFeature: Feature.Continuous = + new Feature.Continuous("prediction.pfollow_pengagement") +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/ml_ranker/scoring/Scorer.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/ml_ranker/scoring/Scorer.scala new file mode 100644 index 0000000000..2ca611535b --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/ml_ranker/scoring/Scorer.scala @@ -0,0 +1,34 @@ +package com.twitter.follow_recommendations.common.rankers.ml_ranker.scoring + +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.follow_recommendations.common.models.HasDisplayLocation +import com.twitter.follow_recommendations.common.models.HasDebugOptions +import com.twitter.follow_recommendations.common.models.Score +import com.twitter.follow_recommendations.common.models.ScoreType +import com.twitter.follow_recommendations.common.rankers.common.RankerId +import com.twitter.ml.api.DataRecord +import com.twitter.product_mixer.core.model.marshalling.request.HasClientContext +import com.twitter.stitch.Stitch +import com.twitter.timelines.configapi.HasParams + +trait Scorer { + + // unique id of the scorer + def id: RankerId.Value + + // type of the output scores + def scoreType: Option[ScoreType] = None + + // Scoring when an ML model is used. + def score(records: Seq[DataRecord]): Stitch[Seq[Score]] + + /** + * Scoring when a non-ML method is applied. E.g: Boosting, randomized reordering, etc. + * This method assumes that candidates' scores are already retrieved from heavy-ranker models and + * are available for use. + */ + def score( + target: HasClientContext with HasParams with HasDisplayLocation with HasDebugOptions, + candidates: Seq[CandidateUser] + ): Seq[Option[Score]] +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/ml_ranker/scoring/ScorerFactory.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/ml_ranker/scoring/ScorerFactory.scala new file mode 100644 index 0000000000..a9ea0a21bd --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/ml_ranker/scoring/ScorerFactory.scala @@ -0,0 +1,38 @@ +package com.twitter.follow_recommendations.common.rankers.ml_ranker.scoring + +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.follow_recommendations.common.rankers.common.RankerId +import com.twitter.follow_recommendations.common.rankers.common.RankerId.RankerId +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class ScorerFactory @Inject() ( + postnuxProdScorer: PostnuxDeepbirdProdScorer, + randomScorer: RandomScorer, + stats: StatsReceiver) { + + private val scorerFactoryStats = stats.scope("scorer_factory") + private val scorerStat = scorerFactoryStats.scope("scorer") + + def getScorers( + rankerIds: Seq[RankerId] + ): Seq[Scorer] = { + rankerIds.map { scorerId => + val scorer: Scorer = getScorerById(scorerId) + // count # of times a ranker has been requested + scorerStat.counter(scorer.id.toString).incr() + scorer + } + } + + def getScorerById(scorerId: RankerId): Scorer = scorerId match { + case RankerId.PostNuxProdRanker => + postnuxProdScorer + case RankerId.RandomRanker => + randomScorer + case _ => + scorerStat.counter("invalid_scorer_type").incr() + throw new IllegalArgumentException("unknown_scorer_type") + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/utils/BUILD b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/utils/BUILD new file mode 100644 index 0000000000..82e9fc7f46 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/utils/BUILD @@ -0,0 +1,8 @@ +scala_library( + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models", + ], +) diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/utils/Utils.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/utils/Utils.scala new file mode 100644 index 0000000000..29f00b6984 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/utils/Utils.scala @@ -0,0 +1,28 @@ +package com.twitter.follow_recommendations.common.rankers.utils + +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.follow_recommendations.common.models.Score +import com.twitter.follow_recommendations.common.rankers.common.RankerId.RankerId + +object Utils { + + /** + * Add the ranking and scoring info for a list of candidates on a given ranking stage. + * @param candidates A list of CandidateUser + * @param rankingStage Should use `Ranker.name` as the ranking stage. + * @return The list of CandidateUser with ranking/scoring info added. + */ + def addRankingInfo(candidates: Seq[CandidateUser], rankingStage: String): Seq[CandidateUser] = { + candidates.zipWithIndex.map { + case (candidate, rank) => + // 1-based ranking for better readability + candidate.addInfoPerRankingStage(rankingStage, candidate.scores, rank + 1) + } + } + + def getCandidateScoreByRankerId(candidate: CandidateUser, rankerId: RankerId): Option[Score] = + candidate.scores.flatMap { ss => ss.scores.find(_.rankerId.contains(rankerId)) } + + def getAllRankerIds(candidates: Seq[CandidateUser]): Seq[RankerId] = + candidates.flatMap(_.scores.map(_.scores.flatMap(_.rankerId))).flatten.distinct +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/weighted_candidate_source_ranker/BUILD b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/weighted_candidate_source_ranker/BUILD new file mode 100644 index 0000000000..3de5523b1c --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/weighted_candidate_source_ranker/BUILD @@ -0,0 +1,20 @@ +scala_library( + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/com/google/inject:guice", + "3rdparty/jvm/com/google/inject/extensions:guice-assistedinject", + "3rdparty/jvm/net/codingwell:scala-guice", + "3rdparty/jvm/org/slf4j:slf4j-api", + "finatra/inject/inject-core/src/main/scala", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/constants", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/common", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/utils", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/utils", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/common", + "util/util-slf4j-api/src/main/scala", + ], +) diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/weighted_candidate_source_ranker/CandidateShuffle.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/weighted_candidate_source_ranker/CandidateShuffle.scala new file mode 100644 index 0000000000..be281a5829 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/weighted_candidate_source_ranker/CandidateShuffle.scala @@ -0,0 +1,36 @@ +package com.twitter.follow_recommendations.common.rankers.weighted_candidate_source_ranker + +import com.twitter.follow_recommendations.common.utils.RandomUtil +import scala.util.Random + +sealed trait CandidateShuffler[T] { + def shuffle(seed: Option[Long])(input: Seq[T]): Seq[T] +} + +class NoShuffle[T]() extends CandidateShuffler[T] { + def shuffle(seed: Option[Long])(input: Seq[T]): Seq[T] = input +} + +class RandomShuffler[T]() extends CandidateShuffler[T] { + def shuffle(seed: Option[Long])(input: Seq[T]): Seq[T] = { + seed.map(new Random(_)).getOrElse(Random).shuffle(input) + } +} + +trait RankWeightedRandomShuffler[T] extends CandidateShuffler[T] { + + def rankToWeight(rank: Int): Double + def shuffle(seed: Option[Long])(input: Seq[T]): Seq[T] = { + val candWeights = input.zipWithIndex.map { + case (candidate, rank) => (candidate, rankToWeight(rank)) + } + RandomUtil.weightedRandomShuffle(candWeights, seed.map(new Random(_))).unzip._1 + } +} + +class ExponentialShuffler[T]() extends RankWeightedRandomShuffler[T] { + def rankToWeight(rank: Int): Double = { + 1 / math + .pow(rank.toDouble, 2.0) // this function was proved to be effective in previous DDGs + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/weighted_candidate_source_ranker/WeightMethod.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/weighted_candidate_source_ranker/WeightMethod.scala new file mode 100644 index 0000000000..54e2ad5495 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/weighted_candidate_source_ranker/WeightMethod.scala @@ -0,0 +1,6 @@ +package com.twitter.follow_recommendations.common.rankers.weighted_candidate_source_ranker + +object WeightMethod extends Enumeration { + type WeightMethod = Value + val WeightedRandomSampling, WeightedRoundRobin = Value +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/weighted_candidate_source_ranker/WeightedCandidateSourceBaseRanker.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/weighted_candidate_source_ranker/WeightedCandidateSourceBaseRanker.scala new file mode 100644 index 0000000000..e348560db9 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/weighted_candidate_source_ranker/WeightedCandidateSourceBaseRanker.scala @@ -0,0 +1,118 @@ +package com.twitter.follow_recommendations.common.rankers.weighted_candidate_source_ranker + +import com.twitter.follow_recommendations.common.utils.RandomUtil +import com.twitter.follow_recommendations.common.utils.MergeUtil +import com.twitter.follow_recommendations.common.utils.Weighted +import com.twitter.follow_recommendations.common.rankers.weighted_candidate_source_ranker.WeightMethod._ +import scala.util.Random + +/** + * This ranker selects the next candidate source to select a candidate from. It supports + * two kinds of algorithm, WeightedRandomSampling or WeightedRoundRobin. WeightedRandomSampling + * pick the next candidate source randomly, WeightedRoundRobin picked the next candidate source + * sequentially based on the weight of the candidate source. It is default to WeightedRandomSampling + * if no weight method is provided. + * + * Example usage of this class: + * + * When use WeightedRandomSampling: + * Input candidate sources and their weights are: {{CS1: 3}, {CS2: 2}, {CS3: 5}} + * Ranked candidates sequence is not determined because of random sampling. + * One possible output candidate sequence is: (CS1_candidate1, CS2_candidate1, CS2_candidate2, + * CS3_candidate1, CS3_candidates2, CS3_candidate3, CS1_candidate2, CS1_candidate3, + * CS3_candidate4, CS3_candidate5, CS1_candidate4, CS1_candidate5, CS2_candidate6, CS2_candidate3,...) + * + * When use WeightedRoundRobin: + * Input candidate sources and their weights are: {{CS1: 3}, {CS2: 2}, {CS3: 5}} + * Output candidate sequence is: (CS1_candidate1, CS1_candidate2, CS1_candidate3, + * CS2_candidate1, CS2_candidates2, CS3_candidate1, CS3_candidate2, CS3_candidate3, + * CS3_candidate4, CS3_candidate5, CS1_candidate4, CS1_candidate5, CS1_candidate6, CS2_candidate3,...) + * + * Note: CS1_candidate1 means the first candidate in CS1 candidate source. + * + * @tparam A candidate source type + * @tparam Rec Recommendation type + * @param candidateSourceWeights relative weights for different candidate sources + */ +class WeightedCandidateSourceBaseRanker[A, Rec]( + candidateSourceWeights: Map[A, Double], + weightMethod: WeightMethod = WeightedRandomSampling, + randomSeed: Option[Long]) { + + /** + * Creates a iterator over algorithms and calls next to return a Stream of candidates + * + * + * @param candidateSources the set of candidate sources that are being sampled + * @param candidateSourceWeights map of candidate source to weight + * @param candidates the map of candidate source to the iterator of its results + * @param weightMethod a enum to indict which weight method to use. Two values are supported + * currently. When WeightedRandomSampling is set, the next candidate is picked from a candidate + * source that is randomly chosen. When WeightedRoundRobin is set, the next candidate is picked + * from current candidate source until the number of candidates reaches to the assigned weight of + * the candidate source. The next call of this function will return a candidate from the next + * candidate source which is after previous candidate source based on the order input + * candidate source sequence. + + * @return stream of candidates + */ + def stream( + candidateSources: Set[A], + candidateSourceWeights: Map[A, Double], + candidates: Map[A, Iterator[Rec]], + weightMethod: WeightMethod = WeightedRandomSampling, + random: Option[Random] = None + ): Stream[Rec] = { + val weightedCandidateSource: Weighted[A] = new Weighted[A] { + override def apply(a: A): Double = candidateSourceWeights.getOrElse(a, 0) + } + + /** + * Generates a stream of candidates. + * + * @param candidateSourceIter an iterator over candidate sources returned by the sampling procedure + * @return stream of candidates + */ + def next(candidateSourceIter: Iterator[A]): Stream[Rec] = { + val source = candidateSourceIter.next() + val it = candidates(source) + if (it.hasNext) { + val currCand = it.next() + currCand #:: next(candidateSourceIter) + } else { + assert(candidateSources.contains(source), "Selected source is not in candidate sources") + // Remove the depleted candidate source and re-sample + stream(candidateSources - source, candidateSourceWeights, candidates, weightMethod, random) + } + } + if (candidateSources.isEmpty) + Stream.empty + else { + val candidateSourceSeq = candidateSources.toSeq + val candidateSourceIter = + if (weightMethod == WeightMethod.WeightedRoundRobin) { + MergeUtil.weightedRoundRobin(candidateSourceSeq)(weightedCandidateSource).iterator + } else { + //default to weighted random sampling if no other weight method is provided + RandomUtil + .weightedRandomSamplingWithReplacement( + candidateSourceSeq, + random + )(weightedCandidateSource).iterator + } + next(candidateSourceIter) + } + } + + def apply(input: Map[A, TraversableOnce[Rec]]): Stream[Rec] = { + stream( + input.keySet, + candidateSourceWeights, + input.map { + case (k, v) => k -> v.toIterator + }, // cannot do mapValues here, as that only returns a view + weightMethod, + randomSeed.map(new Random(_)) + ) + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/weighted_candidate_source_ranker/WeightedCandidateSourceRanker.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/weighted_candidate_source_ranker/WeightedCandidateSourceRanker.scala new file mode 100644 index 0000000000..c6f55adbc3 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/weighted_candidate_source_ranker/WeightedCandidateSourceRanker.scala @@ -0,0 +1,100 @@ +package com.twitter.follow_recommendations.common.rankers.weighted_candidate_source_ranker +import com.twitter.follow_recommendations.common.base.Ranker +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.follow_recommendations.common.rankers.common.DedupCandidates +import com.twitter.follow_recommendations.common.rankers.utils.Utils +import com.twitter.product_mixer.core.model.common.identifier.CandidateSourceIdentifier +import com.twitter.stitch.Stitch +import com.twitter.timelines.configapi.HasParams + +/** + * Candidate Ranker that mixes and ranks multiple candidate lists from different candidate sources with the + * following steps: + * 1) generate a ranked candidate list of each candidate source by sorting and shuffling the candidate list + * of the algorithm. + * 2) merge the ranked lists generated in 1) into a single list using weighted randomly sampling. + * 3) If dedup is required, dedup the output from 2) by candidate id. + * + * @param basedRanker base ranker + * @param shuffleFn the shuffle function that will be used to shuffle each algorithm's sorted candidate list. + * @param dedup whether to remove duplicated candidates from the final output. + */ +class WeightedCandidateSourceRanker[Target <: HasParams]( + basedRanker: WeightedCandidateSourceBaseRanker[ + CandidateSourceIdentifier, + CandidateUser + ], + shuffleFn: Seq[CandidateUser] => Seq[CandidateUser], + dedup: Boolean) + extends Ranker[Target, CandidateUser] { + + val name: String = this.getClass.getSimpleName + + override def rank(target: Target, candidates: Seq[CandidateUser]): Stitch[Seq[CandidateUser]] = { + val scribeRankingInfo: Boolean = + target.params(WeightedCandidateSourceRankerParams.ScribeRankingInfoInWeightedRanker) + val rankedCands = rankCandidates(group(candidates)) + Stitch.value(if (scribeRankingInfo) Utils.addRankingInfo(rankedCands, name) else rankedCands) + } + + private def group( + candidates: Seq[CandidateUser] + ): Map[CandidateSourceIdentifier, Seq[CandidateUser]] = { + val flattened = for { + candidate <- candidates + identifier <- candidate.getPrimaryCandidateSource + } yield (identifier, candidate) + flattened.groupBy(_._1).mapValues(_.map(_._2)) + } + + private def rankCandidates( + input: Map[CandidateSourceIdentifier, Seq[CandidateUser]] + ): Seq[CandidateUser] = { + // Sort and shuffle candidates per candidate source. + // Note 1: Using map instead mapValue here since mapValue somehow caused infinite loop when used as part of Stream. + val sortAndShuffledCandidates = input.map { + case (source, candidates) => + // Note 2: toList is required here since candidates is a view, and it will result in infinit loop when used as part of Stream. + // Note 3: there is no real sorting logic here, it assumes the input is already sorted by candidate sources + val sortedCandidates = candidates.toList + source -> shuffleFn(sortedCandidates).iterator + } + val rankedCandidates = basedRanker(sortAndShuffledCandidates) + + if (dedup) DedupCandidates(rankedCandidates) else rankedCandidates + } +} + +object WeightedCandidateSourceRanker { + + def build[Target <: HasParams]( + candidateSourceWeight: Map[CandidateSourceIdentifier, Double], + shuffleFn: Seq[CandidateUser] => Seq[CandidateUser] = identity, + dedup: Boolean = false, + randomSeed: Option[Long] = None + ): WeightedCandidateSourceRanker[Target] = { + new WeightedCandidateSourceRanker( + new WeightedCandidateSourceBaseRanker( + candidateSourceWeight, + WeightMethod.WeightedRandomSampling, + randomSeed = randomSeed), + shuffleFn, + dedup + ) + } +} + +object WeightedCandidateSourceRankerWithoutRandomSampling { + def build[Target <: HasParams]( + candidateSourceWeight: Map[CandidateSourceIdentifier, Double] + ): WeightedCandidateSourceRanker[Target] = { + new WeightedCandidateSourceRanker( + new WeightedCandidateSourceBaseRanker( + candidateSourceWeight, + WeightMethod.WeightedRoundRobin, + randomSeed = None), + identity, + false, + ) + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/weighted_candidate_source_ranker/WeightedCandidateSourceRankerFSConfig.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/weighted_candidate_source_ranker/WeightedCandidateSourceRankerFSConfig.scala new file mode 100644 index 0000000000..58f0fde3eb --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/weighted_candidate_source_ranker/WeightedCandidateSourceRankerFSConfig.scala @@ -0,0 +1,13 @@ +package com.twitter.follow_recommendations.common.rankers.weighted_candidate_source_ranker + +import com.twitter.follow_recommendations.configapi.common.FeatureSwitchConfig +import com.twitter.timelines.configapi.FSParam + +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class WeightedCandidateSourceRankerFSConfig @Inject() extends FeatureSwitchConfig { + override val booleanFSParams: Seq[FSParam[Boolean]] = + Seq(WeightedCandidateSourceRankerParams.ScribeRankingInfoInWeightedRanker) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/weighted_candidate_source_ranker/WeightedCandidateSourceRankerParams.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/weighted_candidate_source_ranker/WeightedCandidateSourceRankerParams.scala new file mode 100644 index 0000000000..ff4ecae4b3 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/weighted_candidate_source_ranker/WeightedCandidateSourceRankerParams.scala @@ -0,0 +1,8 @@ +package com.twitter.follow_recommendations.common.rankers.weighted_candidate_source_ranker + +import com.twitter.timelines.configapi.FSParam + +object WeightedCandidateSourceRankerParams { + case object ScribeRankingInfoInWeightedRanker + extends FSParam[Boolean]("weighted_ranker_scribe_ranking_info", false) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/stores/BUILD b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/stores/BUILD new file mode 100644 index 0000000000..3b7a46db79 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/stores/BUILD @@ -0,0 +1,19 @@ +scala_library( + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/com/google/inject:guice", + "3rdparty/jvm/com/google/inject/extensions:guice-assistedinject", + "3rdparty/jvm/net/codingwell:scala-guice", + "3rdparty/jvm/org/slf4j:slf4j-api", + "finatra/inject/inject-core/src/main/scala", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/socialgraph", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/strato", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/constants", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models", + "strato/config/columns/onboarding/userrecs:userrecs-strato-client", + "util/util-slf4j-api/src/main/scala", + ], +) diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/stores/LowTweepCredFollowStore.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/stores/LowTweepCredFollowStore.scala new file mode 100644 index 0000000000..d2f4e035b4 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/stores/LowTweepCredFollowStore.scala @@ -0,0 +1,39 @@ +package com.twitter.follow_recommendations.common.stores + +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.follow_recommendations.common.models.HasRecentFollowedUserIds +import com.twitter.stitch.Stitch +import com.twitter.strato.generated.client.onboarding.userrecs.TweepCredOnUserClientColumn +import javax.inject.Inject +import javax.inject.Singleton + +// Not a candidate source since it's a intermediary. +@Singleton +class LowTweepCredFollowStore @Inject() (tweepCredOnUserClientColumn: TweepCredOnUserClientColumn) { + + def getLowTweepCredUsers(target: HasRecentFollowedUserIds): Stitch[Seq[CandidateUser]] = { + val newFollowings = + target.recentFollowedUserIds.getOrElse(Nil).take(LowTweepCredFollowStore.NumFlockToRetrieve) + + val validTweepScoreUserIdsStitch: Stitch[Seq[Long]] = Stitch + .traverse(newFollowings) { newFollowingUserId => + val tweepCredScoreOptStitch = tweepCredOnUserClientColumn.fetcher + .fetch(newFollowingUserId) + .map(_.v) + tweepCredScoreOptStitch.map(_.flatMap(tweepCred => + if (tweepCred < LowTweepCredFollowStore.TweepCredThreshold) { + Some(newFollowingUserId) + } else { + None + })) + }.map(_.flatten) + + validTweepScoreUserIdsStitch + .map(_.map(CandidateUser(_, Some(CandidateUser.DefaultCandidateScore)))) + } +} + +object LowTweepCredFollowStore { + val NumFlockToRetrieve = 500 + val TweepCredThreshold = 40 +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/transforms/dedup/BUILD b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/transforms/dedup/BUILD new file mode 100644 index 0000000000..35534b064a --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/transforms/dedup/BUILD @@ -0,0 +1,8 @@ +scala_library( + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base", + ], +) diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/transforms/dedup/DedupTransform.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/transforms/dedup/DedupTransform.scala new file mode 100644 index 0000000000..64f73d6aed --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/transforms/dedup/DedupTransform.scala @@ -0,0 +1,14 @@ +package com.twitter.follow_recommendations.common.transforms.dedup + +import com.twitter.follow_recommendations.common.base.Transform +import com.twitter.product_mixer.core.model.common.UniversalNoun +import com.twitter.stitch.Stitch +import scala.collection.mutable + +class DedupTransform[Request, Candidate <: UniversalNoun[Long]]() + extends Transform[Request, Candidate] { + override def transform(target: Request, candidates: Seq[Candidate]): Stitch[Seq[Candidate]] = { + val seen = mutable.HashSet[Long]() + Stitch.value(candidates.filter(candidate => seen.add(candidate.id))) + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/transforms/modify_social_proof/BUILD b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/transforms/modify_social_proof/BUILD new file mode 100644 index 0000000000..79da9c2592 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/transforms/modify_social_proof/BUILD @@ -0,0 +1,22 @@ +scala_library( + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/com/google/inject:guice", + "3rdparty/jvm/com/google/inject/extensions:guice-assistedinject", + "3rdparty/jvm/net/codingwell:scala-guice", + "3rdparty/jvm/org/slf4j:slf4j-api", + "configapi/configapi-core", + "finatra/inject/inject-core/src/main/scala", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/graph_feature_service", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/socialgraph", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/constants", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/utils", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/deciders", + "snowflake/src/main/scala/com/twitter/snowflake/id", + "util/util-slf4j-api/src/main/scala", + ], +) diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/transforms/modify_social_proof/ModifySocialProofTransform.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/transforms/modify_social_proof/ModifySocialProofTransform.scala new file mode 100644 index 0000000000..306578a4d3 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/transforms/modify_social_proof/ModifySocialProofTransform.scala @@ -0,0 +1,202 @@ +package com.twitter.follow_recommendations.common.transforms.modify_social_proof + +import com.twitter.conversions.DurationOps._ +import com.twitter.decider.Decider +import com.twitter.decider.RandomRecipient +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.finagle.util.DefaultTimer +import com.twitter.follow_recommendations.common.base.GatedTransform +import com.twitter.follow_recommendations.common.clients.graph_feature_service.GraphFeatureServiceClient +import com.twitter.follow_recommendations.common.clients.socialgraph.SocialGraphClient +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.follow_recommendations.common.models.FollowProof +import com.twitter.follow_recommendations.configapi.deciders.DeciderKey +import com.twitter.graph_feature_service.thriftscala.EdgeType +import com.twitter.product_mixer.core.model.marshalling.request.HasClientContext +import com.twitter.snowflake.id.SnowflakeId +import com.twitter.stitch.Stitch +import com.twitter.timelines.configapi.HasParams +import com.twitter.util.logging.Logging +import com.twitter.util.Duration +import com.twitter.util.Time +import javax.inject.Inject +import javax.inject.Singleton + +object ModifySocialProof { + val GfsLagDuration: Duration = 14.days + val GfsIntersectionIds: Int = 3 + val SgsIntersectionIds: Int = 10 + val LeftEdgeTypes: Set[EdgeType] = Set(EdgeType.Following) + val RightEdgeTypes: Set[EdgeType] = Set(EdgeType.FollowedBy) + + /** + * Given the intersection ID's for a particular candidate, update the candidate's social proof + * @param candidate candidate object + * @param followProof follow proof to be added (includes id's and count) + * @param stats stats for tracking + * @return updated candidate object + */ + def addIntersectionIdsToCandidate( + candidate: CandidateUser, + followProof: FollowProof, + stats: StatsReceiver + ): CandidateUser = { + // create updated set of social proof + val updatedFollowedByOpt = candidate.followedBy match { + case Some(existingFollowedBy) => Some((followProof.followedBy ++ existingFollowedBy).distinct) + case None if followProof.followedBy.nonEmpty => Some(followProof.followedBy.distinct) + case _ => None + } + + val updatedFollowProof = updatedFollowedByOpt.map { updatedFollowedBy => + val updatedCount = followProof.numIds.max(updatedFollowedBy.size) + // track stats + val numSocialProofAdded = updatedFollowedBy.size - candidate.followedBy.size + addCandidatesWithSocialContextCountStat(stats, numSocialProofAdded) + FollowProof(updatedFollowedBy, updatedCount) + } + + candidate.setFollowProof(updatedFollowProof) + } + + private def addCandidatesWithSocialContextCountStat( + statsReceiver: StatsReceiver, + count: Int + ): Unit = { + if (count > 3) { + statsReceiver.counter("4_and_more").incr() + } else { + statsReceiver.counter(count.toString).incr() + } + } +} + +/** + * This class makes a request to gfs/sgs for hydrating additional social proof on each of the + * provided candidates. + */ +@Singleton +class ModifySocialProof @Inject() ( + gfsClient: GraphFeatureServiceClient, + socialGraphClient: SocialGraphClient, + statsReceiver: StatsReceiver, + decider: Decider = Decider.True) + extends Logging { + import ModifySocialProof._ + + private val stats = statsReceiver.scope(this.getClass.getSimpleName) + private val addedStats = stats.scope("num_social_proof_added_per_candidate") + private val gfsStats = stats.scope("graph_feature_service") + private val sgsStats = stats.scope("social_graph_service") + private val previousProofEmptyCounter = stats.counter("previous_proof_empty") + private val emptyFollowProofCounter = stats.counter("empty_followed_proof") + + /** + * For each candidate provided, we get the intersectionIds between the user and the candidate, + * appending the unique results to the social proof (followedBy field) if not already previously + * seen we query GFS for all users, except for cases specified via the mustCallSgs field or for + * very new users, who would not have any data in GFS, due to the lag duration of the service's + * processing. this is determined by GfsLagDuration + * @param userId id of the target user whom we provide recommendations for + * @param candidates list of candidates + * @param intersectionIdsNum if provided, determines the maximum number of accounts we want to be hydrated for social proof + * @param mustCallSgs Determines if we should query SGS regardless of user age or not. + * @return list of candidates updated with additional social proof + */ + def hydrateSocialProof( + userId: Long, + candidates: Seq[CandidateUser], + intersectionIdsNum: Option[Int] = None, + mustCallSgs: Boolean = false, + callSgsCachedColumn: Boolean = false, + gfsLagDuration: Duration = GfsLagDuration, + gfsIntersectionIds: Int = GfsIntersectionIds, + sgsIntersectionIds: Int = SgsIntersectionIds, + ): Stitch[Seq[CandidateUser]] = { + addCandidatesWithSocialContextCountStat( + stats.scope("social_context_count_before_hydration"), + candidates.count(_.followedBy.isDefined) + ) + val candidateIds = candidates.map(_.id) + val userAgeOpt = SnowflakeId.timeFromIdOpt(userId).map(Time.now - _) + + // this decider gate is used to determine what % of requests is allowed to call + // Graph Feature Service. this is useful for ramping down requests to Graph Feature Service + // when necessary + val deciderKey: String = DeciderKey.EnableGraphFeatureServiceRequests.toString + val enableGfsRequests: Boolean = decider.isAvailable(deciderKey, Some(RandomRecipient)) + + // if new query sgs + val (candidateToIntersectionIdsMapFut, isGfs) = + if (!enableGfsRequests || mustCallSgs || userAgeOpt.exists(_ < gfsLagDuration)) { + ( + if (callSgsCachedColumn) + socialGraphClient.getIntersectionsFromCachedColumn( + userId, + candidateIds, + intersectionIdsNum.getOrElse(sgsIntersectionIds) + ) + else + socialGraphClient.getIntersections( + userId, + candidateIds, + intersectionIdsNum.getOrElse(sgsIntersectionIds)), + false) + } else { + ( + gfsClient.getIntersections( + userId, + candidateIds, + intersectionIdsNum.getOrElse(gfsIntersectionIds)), + true) + } + val finalCandidates = candidateToIntersectionIdsMapFut + .map { candidateToIntersectionIdsMap => + { + previousProofEmptyCounter.incr(candidates.count(_.followedBy.exists(_.isEmpty))) + candidates.map { candidate => + addIntersectionIdsToCandidate( + candidate, + candidateToIntersectionIdsMap.getOrElse(candidate.id, FollowProof(Seq.empty, 0)), + addedStats) + } + } + } + .within(250.milliseconds)(DefaultTimer) + .rescue { + case e: Exception => + error(e.getMessage) + if (isGfs) { + gfsStats.scope("rescued").counter(e.getClass.getSimpleName).incr() + } else { + sgsStats.scope("rescued").counter(e.getClass.getSimpleName).incr() + } + Stitch.value(candidates) + } + + finalCandidates.onSuccess { candidatesSeq => + emptyFollowProofCounter.incr(candidatesSeq.count(_.followedBy.exists(_.isEmpty))) + addCandidatesWithSocialContextCountStat( + stats.scope("social_context_count_after_hydration"), + candidatesSeq.count(_.followedBy.isDefined) + ) + } + } +} + +/** + * This transform uses ModifySocialProof (which makes a request to gfs/sgs) for hydrating additional + * social proof on each of the provided candidates. + */ +@Singleton +class ModifySocialProofTransform @Inject() (modifySocialProof: ModifySocialProof) + extends GatedTransform[HasClientContext with HasParams, CandidateUser] + with Logging { + + override def transform( + target: HasClientContext with HasParams, + candidates: Seq[CandidateUser] + ): Stitch[Seq[CandidateUser]] = + target.getOptionalUserId + .map(modifySocialProof.hydrateSocialProof(_, candidates)).getOrElse(Stitch.value(candidates)) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/transforms/modify_social_proof/RemoveAccountProofTransform.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/transforms/modify_social_proof/RemoveAccountProofTransform.scala new file mode 100644 index 0000000000..8face1164c --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/transforms/modify_social_proof/RemoveAccountProofTransform.scala @@ -0,0 +1,27 @@ +package com.twitter.follow_recommendations.common.transforms.modify_social_proof + +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.follow_recommendations.common.base.GatedTransform +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.product_mixer.core.model.marshalling.request.HasClientContext +import com.twitter.stitch.Stitch +import com.twitter.timelines.configapi.HasParams +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class RemoveAccountProofTransform @Inject() (statsReceiver: StatsReceiver) + extends GatedTransform[HasClientContext with HasParams, CandidateUser] { + + private val stats = statsReceiver.scope(this.getClass.getSimpleName) + private val removedProofsCounter = stats.counter("num_removed_proofs") + + override def transform( + target: HasClientContext with HasParams, + items: Seq[CandidateUser] + ): Stitch[Seq[CandidateUser]] = + Stitch.value(items.map { candidate => + removedProofsCounter.incr() + candidate.copy(reason = None) + }) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/transforms/ranker_id/BUILD b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/transforms/ranker_id/BUILD new file mode 100644 index 0000000000..d6dcd85223 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/transforms/ranker_id/BUILD @@ -0,0 +1,19 @@ +scala_library( + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/com/google/inject:guice", + "3rdparty/jvm/com/google/inject/extensions:guice-assistedinject", + "3rdparty/jvm/net/codingwell:scala-guice", + "3rdparty/jvm/org/slf4j:slf4j-api", + "finatra/inject/inject-core/src/main/scala", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/constants", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/common", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/utils", + "hermit/hermit-core/src/main/scala/com/twitter/hermit/constants", + "util/util-slf4j-api/src/main/scala", + ], +) diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/transforms/ranker_id/RandomRankerIdTransform.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/transforms/ranker_id/RandomRankerIdTransform.scala new file mode 100644 index 0000000000..03639da26d --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/transforms/ranker_id/RandomRankerIdTransform.scala @@ -0,0 +1,24 @@ +package com.twitter.follow_recommendations.common.transforms.ranker_id + +import com.google.inject.Inject +import com.google.inject.Singleton +import com.twitter.follow_recommendations.common.base.GatedTransform +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.follow_recommendations.common.models.Score +import com.twitter.stitch.Stitch +import com.twitter.timelines.configapi.HasParams + +/** + * This class appends each candidate's rankerIds with the RandomRankerId. + * This is primarily for determining if a candidate was generated via random shuffling. + */ +@Singleton +class RandomRankerIdTransform @Inject() () extends GatedTransform[HasParams, CandidateUser] { + + override def transform( + target: HasParams, + candidates: Seq[CandidateUser] + ): Stitch[Seq[CandidateUser]] = { + Stitch.value(candidates.map(_.addScore(Score.RandomScore))) + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/transforms/recommendation_flow_identifier/AddRecommendationFlowIdentifierTransform.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/transforms/recommendation_flow_identifier/AddRecommendationFlowIdentifierTransform.scala new file mode 100644 index 0000000000..87c111a6c2 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/transforms/recommendation_flow_identifier/AddRecommendationFlowIdentifierTransform.scala @@ -0,0 +1,20 @@ +package com.twitter.follow_recommendations.common.transforms.recommendation_flow_identifier + +import com.google.inject.Inject +import com.twitter.follow_recommendations.common.base.Transform +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.follow_recommendations.common.models.HasRecommendationFlowIdentifier +import com.twitter.stitch.Stitch + +class AddRecommendationFlowIdentifierTransform @Inject() + extends Transform[HasRecommendationFlowIdentifier, CandidateUser] { + + override def transform( + target: HasRecommendationFlowIdentifier, + items: Seq[CandidateUser] + ): Stitch[Seq[CandidateUser]] = { + Stitch.value(items.map { candidateUser => + candidateUser.copy(recommendationFlowIdentifier = target.recommendationFlowIdentifier) + }) + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/transforms/recommendation_flow_identifier/BUILD b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/transforms/recommendation_flow_identifier/BUILD new file mode 100644 index 0000000000..820e2df66e --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/transforms/recommendation_flow_identifier/BUILD @@ -0,0 +1,9 @@ +scala_library( + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models", + ], +) diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/transforms/tracking_token/BUILD b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/transforms/tracking_token/BUILD new file mode 100644 index 0000000000..d9b257348e --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/transforms/tracking_token/BUILD @@ -0,0 +1,18 @@ +scala_library( + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/com/google/inject:guice", + "3rdparty/jvm/com/google/inject/extensions:guice-assistedinject", + "3rdparty/jvm/net/codingwell:scala-guice", + "3rdparty/jvm/org/slf4j:slf4j-api", + "finatra/inject/inject-core/src/main/scala", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/constants", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/utils", + "hermit/hermit-core/src/main/scala/com/twitter/hermit/constants", + "util/util-slf4j-api/src/main/scala", + ], +) diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/transforms/tracking_token/TrackingTokenTransform.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/transforms/tracking_token/TrackingTokenTransform.scala new file mode 100644 index 0000000000..5a30c9cb17 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/transforms/tracking_token/TrackingTokenTransform.scala @@ -0,0 +1,76 @@ +package com.twitter.follow_recommendations.common.transforms.tracking_token + +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.follow_recommendations.common.base.Transform +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.follow_recommendations.common.models.HasDisplayLocation +import com.twitter.follow_recommendations.common.models.Session +import com.twitter.follow_recommendations.common.models.TrackingToken +import com.twitter.hermit.constants.AlgorithmFeedbackTokens.AlgorithmToFeedbackTokenMap +import com.twitter.hermit.model.Algorithm +import com.twitter.product_mixer.core.model.common.identifier.CandidateSourceIdentifier +import com.twitter.product_mixer.core.model.marshalling.request.HasClientContext +import com.twitter.stitch.Stitch +import com.twitter.util.logging.Logging + +import javax.inject.Inject +import javax.inject.Singleton + +/** + * This transform adds the tracking token for all candidates + * Since this happens in the same request, we use the same trace-id for all candidates + * There are no RPC calls in this transform so it's safe to chain it with `andThen` at the end of + * all other product-specific transforms + */ +@Singleton +class TrackingTokenTransform @Inject() (baseStatsReceiver: StatsReceiver) + extends Transform[HasDisplayLocation with HasClientContext, CandidateUser] + with Logging { + + def profileResults( + target: HasDisplayLocation with HasClientContext, + candidates: Seq[CandidateUser] + ) = { + // Metrics to track # results per candidate source + val stats = baseStatsReceiver.scope(target.displayLocation.toString + "/final_results") + stats.stat("total").add(candidates.size) + + stats.counter(target.displayLocation.toString).incr() + + val flattenedCandidates: Seq[(CandidateSourceIdentifier, CandidateUser)] = for { + candidate <- candidates + identifier <- candidate.getPrimaryCandidateSource + } yield (identifier, candidate) + val candidatesGroupedBySource: Map[CandidateSourceIdentifier, Seq[CandidateUser]] = + flattenedCandidates.groupBy(_._1).mapValues(_.map(_._2)) + candidatesGroupedBySource map { + case (source, candidates) => stats.stat(source.name).add(candidates.size) + } + } + + override def transform( + target: HasDisplayLocation with HasClientContext, + candidates: Seq[CandidateUser] + ): Stitch[Seq[CandidateUser]] = { + profileResults(target, candidates) + + Stitch.value( + target.getOptionalUserId + .map { _ => + candidates.map { + candidate => + val token = Some(TrackingToken( + sessionId = Session.getSessionId, + displayLocation = Some(target.displayLocation), + controllerData = None, + algorithmId = candidate.userCandidateSourceDetails.flatMap(_.primaryCandidateSource + .flatMap { identifier => + Algorithm.withNameOpt(identifier.name).flatMap(AlgorithmToFeedbackTokenMap.get) + }) + )) + candidate.copy(trackingToken = token) + } + }.getOrElse(candidates)) + + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/transforms/weighted_sampling/BUILD b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/transforms/weighted_sampling/BUILD new file mode 100644 index 0000000000..606e8edfae --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/transforms/weighted_sampling/BUILD @@ -0,0 +1,10 @@ +scala_library( + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/utils", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/common", + ], +) diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/transforms/weighted_sampling/SamplingTransform.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/transforms/weighted_sampling/SamplingTransform.scala new file mode 100644 index 0000000000..269a39a489 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/transforms/weighted_sampling/SamplingTransform.scala @@ -0,0 +1,138 @@ +package com.twitter.follow_recommendations.common.transforms.weighted_sampling +import com.twitter.follow_recommendations.common.base.GatedTransform +import com.twitter.stitch.Stitch +import com.twitter.timelines.configapi.HasParams +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.follow_recommendations.common.models.HasDebugOptions +import com.twitter.follow_recommendations.common.models.Score +import com.twitter.follow_recommendations.common.models.Scores +import com.twitter.follow_recommendations.common.rankers.common.RankerId +import com.twitter.follow_recommendations.common.rankers.utils.Utils +import com.twitter.product_mixer.core.model.marshalling.request.HasClientContext +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class SamplingTransform @Inject() () + extends GatedTransform[HasClientContext with HasParams with HasDebugOptions, CandidateUser] { + + val name: String = this.getClass.getSimpleName + + /* + Description: This function takes in a set of candidate users and ranks them for a who-to-follow + request by sampling from the Placket-Luce distribution + (https://cran.rstudio.com/web/packages/PlackettLuce/vignettes/Overview.html) with a three + variations. The first variation is that the scores of the candidates are multiplied by + multiplicativeFactor before sampling. The second variation is that the scores are + exponentiated before sampling. The third variation is that depending on how many who-to-follow + positions are being requested, the first k positions are reserved for the candidates with the + highest scores (and they are sorted in decreasing order of score) and the remaining positions + are sampled from a Placket-Luce. We use the efficient algorithm proposed in this blog + https://medium.com/swlh/going-old-school-designing-algorithms-for-fast-weighted-sampling-in-production-c48fc1f40051 + to sample from a Plackett-Luce. Because of numerical stability reasons, before sampling from this + distribution, (1) we subtract off the maximum score from all the scores and (2) if after + this subtraction and multiplication by the multiplicative factor the resulting score is <= -10, + we force the candidate's transformed score under the above algorithm to be 0 (so r^(1/w) = 0) + where r is a random number and w is the transformed score. + + inputs: + - target: HasClientContext (WTF request) + - candidates: sequence of CandidateUsers (users that need to be ranked from a who-to-follow + request) each of which has a score + + inputs accessed through feature switches, i.e. through target.params (see the following file: + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/ + transforms/weighted_sampling/SamplingTransformParams.scala"): + - topKFixed: the first k positions of the who-to-follow ranking correspond to the users with the k + highest scores and are not sampled from the Placket-Luce distribution + - multiplicativeFactor: multiplicativeFactor is used to transform the scores of each candidate by + multiplying that user's score by multiplicativeFactor + + output: + - Sequence of CandidateUser whose order represents the ranking of users in a who-to-follow request + This ranking is sampled from a Placket-Luce distribution. + */ + override def transform( + target: HasClientContext with HasParams with HasDebugOptions, + candidates: Seq[CandidateUser] + ): Stitch[Seq[CandidateUser]] = { + + // the first k positions of the who-to-follow ranking correspond to the users with the k + // highest scores and are not sampled from the Placket-Luce distribution + val topKFixed = target.params(SamplingTransformParams.TopKFixed) + + // multiplicativeFactor is used to transform the scores of each candidate by + // multiplying that user's score by multiplicativeFactor + val multiplicativeFactor = target.params(SamplingTransformParams.MultiplicativeFactor) + + // sort candidates by their score + val candidatesSorted = candidates.sortBy(-1 * _.score.getOrElse(0.0)) + + // pick the top K candidates by score and the remaining candidates + val (topKFixedCandidates, candidatesOutsideOfTopK) = + candidatesSorted.zipWithIndex.partition { case (value, index) => index < topKFixed } + + val randomNumGenerator = + new scala.util.Random(target.getRandomizationSeed.getOrElse(System.currentTimeMillis)) + + // we need to subtract the maximum score off the scores for numerical stability reasons + // subtracting the max score off does not effect the underlying distribution we are sampling + // the candidates from + // we need the if statement since you cannot take the max of an empty sequence + val maximum_score = if (candidatesOutsideOfTopK.nonEmpty) { + candidatesOutsideOfTopK.map(x => x._1.score.getOrElse(0.0)).max + } else { + 0.0 + } + + // for candidates in candidatesOutsideOfTopK, we transform their score by subtracting off + // maximum_score and then multiply by multiplicativeFactor + val candidatesOutsideOfTopKTransformedScore = candidatesOutsideOfTopK.map(x => + (x._1, multiplicativeFactor * (x._1.score.getOrElse(0.0) - maximum_score))) + + // for each candidate with score transformed and clip score w, sample a random number r, + // create a new score r^(1/w) and sort the candidates to get the final ranking. + // for numerical stability reasons if the score is <=-10, we force r^(1/w) = 0. + // this samples the candidates from the modified Plackett-Luce distribution. See + // https://medium.com/swlh/going-old-school-designing-algorithms-for-fast-weighted-sampling-in-production-c48fc1f40051 + + val candidatesOutsideOfTopKSampled = candidatesOutsideOfTopKTransformedScore + .map(x => + ( + x._1, + if (x._2 <= -10.0) + 0.0 + else + scala.math.pow( + randomNumGenerator.nextFloat(), + 1 / (scala.math + .exp(x._2))))).sortBy(-1 * _._2) + + val topKCandidates: Seq[CandidateUser] = topKFixedCandidates.map(_._1) + + val scribeRankingInfo: Boolean = + target.params(SamplingTransformParams.ScribeRankingInfoInSamplingTransform) + + val transformedCandidates: Seq[CandidateUser] = if (scribeRankingInfo) { + val topKCandidatesWithRankingInfo: Seq[CandidateUser] = + Utils.addRankingInfo(topKCandidates, name) + val candidatesOutsideOfTopKSampledWithRankingInfo: Seq[CandidateUser] = + candidatesOutsideOfTopKSampled.zipWithIndex.map { + case ((candidate, score), rank) => + val newScore = Seq(Score(score, Some(RankerId.PlacketLuceSamplingTransformer))) + val newScores: Option[Scores] = candidate.scores + .map { scores => + scores.copy(scores = scores.scores ++ newScore) + }.orElse(Some(Scores(newScore, Some(RankerId.PlacketLuceSamplingTransformer)))) + val globalRank = rank + topKFixed + 1 + candidate.addInfoPerRankingStage(name, newScores, globalRank) + } + + topKCandidatesWithRankingInfo ++ candidatesOutsideOfTopKSampledWithRankingInfo + } else { + topKCandidates ++ candidatesOutsideOfTopKSampled.map(_._1) + } + + Stitch.value(transformedCandidates) + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/transforms/weighted_sampling/SamplingTransformFSConfig.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/transforms/weighted_sampling/SamplingTransformFSConfig.scala new file mode 100644 index 0000000000..b97251f93a --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/transforms/weighted_sampling/SamplingTransformFSConfig.scala @@ -0,0 +1,19 @@ +package com.twitter.follow_recommendations.common.transforms.weighted_sampling + +import com.twitter.follow_recommendations.configapi.common.FeatureSwitchConfig +import com.twitter.timelines.configapi.FSBoundedParam +import com.twitter.timelines.configapi.FSParam + +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class SamplingTransformFSConfig @Inject() () extends FeatureSwitchConfig { + override val intFSParams: Seq[FSBoundedParam[Int]] = Seq(SamplingTransformParams.TopKFixed) + + override val doubleFSParams: Seq[FSBoundedParam[Double]] = Seq( + SamplingTransformParams.MultiplicativeFactor) + + override val booleanFSParams: Seq[FSParam[Boolean]] = Seq( + SamplingTransformParams.ScribeRankingInfoInSamplingTransform) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/transforms/weighted_sampling/SamplingTransformParams.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/transforms/weighted_sampling/SamplingTransformParams.scala new file mode 100644 index 0000000000..363487a9be --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/transforms/weighted_sampling/SamplingTransformParams.scala @@ -0,0 +1,25 @@ +package com.twitter.follow_recommendations.common.transforms.weighted_sampling + +import com.twitter.timelines.configapi.FSBoundedParam +import com.twitter.timelines.configapi.FSParam + +object SamplingTransformParams { + + case object TopKFixed // indicates how many of the fisrt K who-to-follow recommendations are reserved for the candidates with largest K CandidateUser.score where these candidates are sorted in decreasing order of score + extends FSBoundedParam[Int]( + name = "post_nux_ml_flow_weighted_sampling_top_k_fixed", + default = 0, + min = 0, + max = 100) + + case object MultiplicativeFactor // CandidateUser.score gets transformed to multiplicativeFactor*CandidateUser.score before sampling from the Plackett-Luce distribution + extends FSBoundedParam[Double]( + name = "post_nux_ml_flow_weighted_sampling_multiplicative_factor", + default = 1.0, + min = -1000.0, + max = 1000.0) + + case object ScribeRankingInfoInSamplingTransform + extends FSParam[Boolean]("sampling_transform_scribe_ranking_info", false) + +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/utils/BUILD b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/utils/BUILD new file mode 100644 index 0000000000..7075167e3c --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/utils/BUILD @@ -0,0 +1,13 @@ +scala_library( + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "finatra/inject/inject-core/src/main/scala", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/request", + "stitch/stitch-core", + "util/util-slf4j-api/src/main/scala", + ], +) diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/utils/CollectionUtil.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/utils/CollectionUtil.scala new file mode 100644 index 0000000000..db9a1f9f50 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/utils/CollectionUtil.scala @@ -0,0 +1,22 @@ +package com.twitter.follow_recommendations.common.utils + +object CollectionUtil { + + /** + * Transposes a sequence of sequences. As opposed to the Scala collection library version + * of transpose, the sequences do not have to have the same length. + * + * Example: + * transpose(immutable.Seq(immutable.Seq(1,2,3), immutable.Seq(4,5), immutable.Seq(6,7))) + * => immutable.Seq(immutable.Seq(1, 4, 6), immutable.Seq(2, 5, 7), immutable.Seq(3)) + * + * @param seq a sequence of sequences + * @tparam A the type of elements in the seq + * @return the transposed sequence of sequences + */ + def transposeLazy[A](seq: Seq[Seq[A]]): Stream[Seq[A]] = + seq.filter(_.nonEmpty) match { + case Nil => Stream.empty + case ys => ys.map(_.head) #:: transposeLazy(ys.map(_.tail)) + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/utils/DisplayLocationProductConverterUtil.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/utils/DisplayLocationProductConverterUtil.scala new file mode 100644 index 0000000000..2f6db39b61 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/utils/DisplayLocationProductConverterUtil.scala @@ -0,0 +1,27 @@ +package com.twitter.follow_recommendations.common.utils + +import com.twitter.follow_recommendations.common.models.DisplayLocation +import com.twitter.follow_recommendations.common.models.Product +import com.twitter.product_mixer.core.model.marshalling.request.Product + +object DisplayLocationProductConverterUtil { + def productToDisplayLocation(product: Product): DisplayLocation = { + product match { + case Product.MagicRecs => DisplayLocation.MagicRecs + case _ => + throw UnconvertibleProductMixerProductException( + s"Cannot convert Product Mixer Product ${product.identifier.name} into a FRS DisplayLocation.") + } + } + + def displayLocationToProduct(displayLocation: DisplayLocation): Product = { + displayLocation match { + case DisplayLocation.MagicRecs => Product.MagicRecs + case _ => + throw UnconvertibleProductMixerProductException( + s"Cannot convert DisplayLocation ${displayLocation.toFsName} into a Product Mixer Product.") + } + } +} + +case class UnconvertibleProductMixerProductException(message: String) extends Exception(message) diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/utils/MergeUtil.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/utils/MergeUtil.scala new file mode 100644 index 0000000000..6aaee4c459 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/utils/MergeUtil.scala @@ -0,0 +1,51 @@ +package com.twitter.follow_recommendations.common.utils + +object MergeUtil { + + /** + * Takes a seq of items which have weights. Returns an infinite stream of each item + * by their weights. All weights need to be greater than or equal to zero. In addition, + * the sum of weights should be greater than zero. + * + * Example usage of this function: + * Input weighted Item {{CS1, 3}, {CS2, 2}, {CS3, 5}} + * Output stream: (CS1, CS1, CS1, CS2, CS2, CS3, CS3, CS3, CS3, CS3, CS1, CS1, CS1, CS2,...} + * + * @param items items + * @param weighted provides weights for items + * @tparam T type of item + * + * @return Stream of Ts + */ + def weightedRoundRobin[T]( + items: Seq[T] + )( + implicit weighted: Weighted[T] + ): Stream[T] = { + if (items.isEmpty) { + Stream.empty + } else { + val weights = items.map { i => weighted(i) } + assert( + weights.forall { + _ >= 0 + }, + "Negative weight exists for sampling") + val cumulativeWeight = weights.scanLeft(0.0)(_ + _).tail + assert(cumulativeWeight.last > 0, "Sum of the sampling weights is not positive") + + var weightIdx = 0 + var weight = 0 + + def next(): Stream[T] = { + val tmpIdx = weightIdx + weight = weight + 1 + weight = if (weight >= weights(weightIdx)) 0 else weight + weightIdx = if (weight == 0) weightIdx + 1 else weightIdx + weightIdx = if (weightIdx == weights.length) 0 else weightIdx + items(tmpIdx) #:: next() + } + next() + } + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/utils/RandomUtil.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/utils/RandomUtil.scala new file mode 100644 index 0000000000..9d66e8debd --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/utils/RandomUtil.scala @@ -0,0 +1,88 @@ +package com.twitter.follow_recommendations.common.utils +import scala.util.Random + +object RandomUtil { + + /** + * Takes a seq of items which have weights. Returns an infinite stream that is + * sampled with replacement using the weights for each item. All weights need + * to be greater than or equal to zero. In addition, the sum of weights + * should be greater than zero. + * + * @param items items + * @param weighted provides weights for items + * @tparam T type of item + * @return Stream of Ts + */ + def weightedRandomSamplingWithReplacement[T]( + items: Seq[T], + random: Option[Random] = None + )( + implicit weighted: Weighted[T] + ): Stream[T] = { + if (items.isEmpty) { + Stream.empty + } else { + val weights = items.map { i => weighted(i) } + assert(weights.forall { _ >= 0 }, "Negative weight exists for sampling") + val cumulativeWeight = weights.scanLeft(0.0)(_ + _).tail + assert(cumulativeWeight.last > 0, "Sum of the sampling weights is not positive") + val cumulativeProbability = cumulativeWeight map (_ / cumulativeWeight.last) + def next(): Stream[T] = { + val rand = random.getOrElse(Random).nextDouble() + val idx = cumulativeProbability.indexWhere(_ >= rand) + items(if (idx == -1) items.length - 1 else idx) #:: next() + } + next() + } + } + + /** + * Takes a seq of items and their weights. Returns a lazy weighted shuffle of + * the elements in the list. All weights should be greater than zero. + * + * @param items items + * @param weighted provides weights for items + * @tparam T type of item + * @return Stream of Ts + */ + def weightedRandomShuffle[T]( + items: Seq[T], + random: Option[Random] = None + )( + implicit weighted: Weighted[T] + ): Stream[T] = { + assert(items.forall { i => weighted(i) > 0 }, "Non-positive weight exists for shuffling") + def next(it: Seq[T]): Stream[T] = { + if (it.isEmpty) + Stream.empty + else { + val cumulativeWeight = it.scanLeft(0.0)((acc: Double, curr: T) => acc + weighted(curr)).tail + val cutoff = random.getOrElse(Random).nextDouble() * cumulativeWeight.last + val idx = cumulativeWeight.indexWhere(_ >= cutoff) + val (left, right) = it.splitAt(idx) + it(if (idx == -1) it.size - 1 else idx) #:: next(left ++ right.drop(1)) + } + } + next(items) + } + + /** + * Takes a seq of items and a weight function, returns a lazy weighted shuffle of + * the elements in the list.The weight function is based on the rank of the element + * in the original lst. + * @param items + * @param rankToWeight + * @param random + * @tparam T + * @return + */ + def weightedRandomShuffleByRank[T]( + items: Seq[T], + rankToWeight: Int => Double, + random: Option[Random] = None + ): Stream[T] = { + val candWeights = items.zipWithIndex.map { case (item, rank) => (item, rankToWeight(rank)) } + RandomUtil.weightedRandomShuffle(candWeights, random).map(_._1) + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/utils/RescueWithStatsUtils.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/utils/RescueWithStatsUtils.scala new file mode 100644 index 0000000000..8275228d6d --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/utils/RescueWithStatsUtils.scala @@ -0,0 +1,50 @@ +package com.twitter.follow_recommendations.common.utils + +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.follow_recommendations.common.base.StatsUtil +import com.twitter.stitch.Stitch +import com.twitter.util.Duration +import com.twitter.util.TimeoutException + +object RescueWithStatsUtils { + def rescueWithStats[T]( + s: Stitch[Seq[T]], + stats: StatsReceiver, + source: String + ): Stitch[Seq[T]] = { + StatsUtil.profileStitchSeqResults(s, stats.scope(source)).rescue { + case _: Exception => Stitch.Nil + } + } + + def rescueOptionalWithStats[T]( + s: Stitch[Option[T]], + stats: StatsReceiver, + source: String + ): Stitch[Option[T]] = { + StatsUtil.profileStitchOptionalResults(s, stats.scope(source)).rescue { + case _: Exception => Stitch.None + } + } + + def rescueWithStatsWithin[T]( + s: Stitch[Seq[T]], + stats: StatsReceiver, + source: String, + timeout: Duration + ): Stitch[Seq[T]] = { + val hydratedScopeSource = stats.scope(source) + StatsUtil + .profileStitchSeqResults( + s.within(timeout)(com.twitter.finagle.util.DefaultTimer), + hydratedScopeSource) + .rescue { + case _: TimeoutException => + hydratedScopeSource.counter("timeout").incr() + Stitch.Nil + case _: Exception => + hydratedScopeSource.counter("exception").incr() + Stitch.Nil + } + } +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/utils/UserSignupUtil.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/utils/UserSignupUtil.scala new file mode 100644 index 0000000000..73d90a85b0 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/utils/UserSignupUtil.scala @@ -0,0 +1,14 @@ +package com.twitter.follow_recommendations.common.utils + +import com.twitter.product_mixer.core.model.marshalling.request.HasClientContext +import com.twitter.snowflake.id.SnowflakeId +import com.twitter.util.Duration +import com.twitter.util.Time + +object UserSignupUtil { + def signupTime(hasClientContext: HasClientContext): Option[Time] = + hasClientContext.clientContext.userId.flatMap(SnowflakeId.timeFromIdOpt) + + def userSignupAge(hasClientContext: HasClientContext): Option[Duration] = + signupTime(hasClientContext).map(Time.now - _) +} diff --git a/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/utils/Weighted.scala b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/utils/Weighted.scala new file mode 100644 index 0000000000..adb95e5f56 --- /dev/null +++ b/follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/utils/Weighted.scala @@ -0,0 +1,21 @@ +package com.twitter.follow_recommendations.common.utils + +/** + * Typeclass for any Recommendation type that has a weight + * + */ +trait Weighted[-Rec] { + def apply(rec: Rec): Double +} + +object Weighted { + implicit object WeightedTuple extends Weighted[(_, Double)] { + override def apply(rec: (_, Double)): Double = rec._2 + } + + def fromFunction[Rec](f: Rec => Double): Weighted[Rec] = { + new Weighted[Rec] { + override def apply(rec: Rec): Double = f(rec) + } + } +} diff --git a/follow-recommendations-service/server/src/main/resources/BUILD b/follow-recommendations-service/server/src/main/resources/BUILD new file mode 100644 index 0000000000..610607ec5c --- /dev/null +++ b/follow-recommendations-service/server/src/main/resources/BUILD @@ -0,0 +1,20 @@ +resources( + sources = [ + "*.tsv", + "*.xml", + "**/*", + "config/*.yml", + ], +) + +# Created for Bazel compatibility. +# In Bazel, loose files must be part of a target to be included into a bundle. +files( + name = "frs_resources", + sources = [ + "*.tsv", + "*.xml", + "*.yml", + "**/*", + ], +) diff --git a/follow-recommendations-service/server/src/main/resources/config/decider.yml b/follow-recommendations-service/server/src/main/resources/config/decider.yml new file mode 100644 index 0000000000..a466260949 --- /dev/null +++ b/follow-recommendations-service/server/src/main/resources/config/decider.yml @@ -0,0 +1,129 @@ +enable_recommendations: + comment: Proportion of requests where we return an actual response as part. Decreasing the value will increase the portion of empty responses (in order to disable the service) as part of the graceful degradation. + default_availability: 10000 +enable_score_user_candidates: + comment: Proportion of requests where score user candidates from the scoreUserCandidates endpoint + default_availability: 10000 +enable_profile_sidebar_product: + comment: Proportion of requests where we return an actual response for profile sidebar product + default_availability: 10000 +enable_magic_recs_product: + comment: Proportion of requests where we return an actual response for magic recs product + default_availability: 10000 +enable_rux_landing_page_product: + comment: Proportion of requests where we return an actual response for rux landing page product + default_availability: 10000 +enable_rux_pymk_product: + comment: Proportion of requests where we return an actual response for rux pymk product + default_availability: 10000 +enable_profile_bonus_follow_product: + comment: Proportion of requests where we return an actual response for profile bonus follow product + default_availability: 10000 +enable_election_explore_wtf_product: + comment: Proportion of requests where we return an actual response for election explore wtf product + default_availability: 10000 +enable_cluster_follow_product: + comment: Proportion of requests where we return an actual response for cluster follow product + default_availability: 10000 +enable_home_timeline_product: + comment: Proportion of requests where we return an actual response for htl wtf product + default_availability: 10000 +enable_htl_bonus_follow_product: + comment: Proportion of requests where we return an actual response for htl bonus follow product + default_availability: 10000 +enable_explore_tab_product: + comment: Proportion of requests where we return an actual response for explore tab product + default_availability: 10000 +enable_sidebar_product: + comment: Proportion of requests where we return an actual response for sidebar product + default_availability: 10000 +enable_campaign_form_product: + comment: Proportion of requests where we return an actual response for campaign form product + default_availability: 10000 +enable_reactive_follow_product: + comment: Proportion of requests where we return an actual response for reactive follow product + default_availability: 10000 +enable_nux_pymk_product: + comment: Proportion of requests where we return an actual response for nux pymk product + default_availability: 10000 +enable_nux_interests_product: + comment: Proportion of requests where we return an actual response for nux interests product + default_availability: 10000 +enable_nux_topic_bonus_follow_product: + comment: Proportion of requests where we return an actual response for nux topic-based bonus follow product + default_availability: 10000 +enable_india_covid19_curated_accounts_wtf_product: + comment: Proportion of requests where we return an actual response for india covid19 curated accounts wtf product + default_availability: 10000 +enable_ab_upload_product: + comment: Proportion of requests where we return an actual response for the address book upload product + default_availability: 10000 +enable_people_plus_plus_product: + comment: Proportion of requests where we return an actual response for the PeoplePlusPlus/Connect Tab product + default_availability: 10000 +enable_tweet_notification_recs_product: + comment: Proportion of requests where we return an actual response for the Tweet Notification Recommendations product + default_availability: 10000 +enable_profile_device_follow_product: + comment: Proportion of requests where we return an actual response for the ProfileDeviceFollow product + default_availability: 10000 +enable_diffy_module_dark_reading: + comment: Percentage of dark read traffic routed to diffy thrift + default_availability: 0 +enable_recos_backfill_product: + comment: Proportion of requests where we return an actual response for the RecosBackfill product + default_availability: 10000 +enable_post_nux_follow_task_product: + comment: Proportion of requests where we return an actual response for post NUX follow task product + default_availability: 10000 +enable_curated_space_hosts_product: + comment: Proportion of requests where we return an actual response for curated space hosts product + default_availability: 10000 +enable_nux_geo_category_product: + comment: Proportion of requests where we return an actual response for nux geo category product + default_availability: 10000 +enable_nux_interests_category_product: + comment: Proportion of requests where we return an actual response for nux interests category product + default_availability: 10000 +enable_nux_pymk_category_product: + comment: Proportion of requests where we return an actual response for nux pymk category product + default_availability: 10000 +enable_home_timeline_tweet_recs_product: + comment: Proportion of requests where we return an actual response for the Home Timeline Tweet Recs product + default_availability: 10000 +enable_htl_bulk_friend_follows_product: + comment: Proportion of requests where we return an actual response for the HTL bulk friend follows product + default_availability: 10000 +enable_nux_auto_follow_product: + comment: Proportion of requests where we return an actual response for the NUX auto follow product + default_availability: 10000 +enable_search_bonus_follow_product: + comment: Proportion of requests where we return an actual response for search bonus follow product + default_availability: 10000 +enable_fetch_user_in_request_builder: + comment: Proportion of requests where we fetch user object from gizmoduck in request builder + default_availability: 0 +enable_product_mixer_magic_recs_product: + comment: Proportion of requests where we enable the product mixer magic recs product + default_availability: 10000 +enable_home_timeline_reverse_chron_product: + comment: Proportion of requests where we return an actual response for Home timeline reverse chron product + default_availability: 10000 +enable_product_mixer_pipeline_magic_recs_dark_read: + comment: Compare product mixer pipeline responses to current FRS pipeline responses for Magic Recs + default_availability: 0 +enable_experimental_caching: + comment: Proportion of requests we use experimental caching for data caching + default_availability: 0 +enable_distributed_caching: + comment: Proportion of requests we use a distributed cache cluster for data caching + default_availability: 10000 +enable_gizmoduck_caching: + comment: Proportion of requests we use a distributed cache cluster for data caching in Gizmoduck + default_availability: 10000 +enable_traffic_dark_reading: + comment: Proportion of requests where we replicate the request for traffic dark reading + default_availability: 0 +enable_graph_feature_service_requests: + comment: Proportion of requests where we allow request calls to Graph Feature Service + default_availability: 10000 diff --git a/follow-recommendations-service/server/src/main/resources/logback.xml b/follow-recommendations-service/server/src/main/resources/logback.xml new file mode 100644 index 0000000000..96348c8e4a --- /dev/null +++ b/follow-recommendations-service/server/src/main/resources/logback.xml @@ -0,0 +1,133 @@ + + + + + + + + + + + + + + + + + true + + + + + ${log.service.output} + + ${log.service.output}.%i + 1 + 5 + + + 50MB + + + %date %.-3level ${DEFAULT_SERVICE_PATTERN}%n + + + + + + ${log.access.output} + + ${log.access.output}.%i + 1 + 5 + + + 50MB + + + ${DEFAULT_ACCESS_PATTERN}%n + + + + + + true + ${log.lens.category} + ${log.lens.index} + ${log.lens.tag}/service + + %msg + + + + + + true + ${log.lens.category} + ${log.lens.index} + ${log.lens.tag}/access + + %msg + + + + + + + + + + + + ${async_queue_size} + ${async_max_flush_time} + + + + + ${async_queue_size} + ${async_max_flush_time} + + + + + ${async_queue_size} + ${async_max_flush_time} + + + + + ${async_queue_size} + ${async_max_flush_time} + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/follow-recommendations-service/server/src/main/resources/quality/stp_models/20141223/epModel b/follow-recommendations-service/server/src/main/resources/quality/stp_models/20141223/epModel new file mode 100644 index 0000000000..a08d9723cc --- /dev/null +++ b/follow-recommendations-service/server/src/main/resources/quality/stp_models/20141223/epModel @@ -0,0 +1,8 @@ +# OWNER = jdeng +# Date = 20141223_153423 +# Training Size = 16744473 +# Testing Size = 16767335 +# trained with ElasticNetCV alpha=0.05 cv_folds=5 best_lambda=1.0E-7 +# num base features: 10 +# num nonzero weights: 30 +{bias:-5.67151,featureMetadataMap:["fwd_email":{metadata:{featureWeight:{weight:2.47389}}},"rev_phone":{metadata:{featureWeight:{weight:1.88433}}},"mutual_follow_path":{metadata:{featureWeight:{intervalWeights:[{left:47.0,weight:6.31809},{left:11.0,right:16.0,weight:4.52959},{left:31.0,right:47.0,weight:5.7101},{right:2.0,weight:0.383515},{left:24.0,right:31.0,weight:5.26515},{left:3.0,right:4.0,weight:2.91751},{left:2.0,right:3.0,weight:2.22851},{left:4.0,right:5.0,weight:3.28515},{left:8.0,right:11.0,weight:4.14731},{left:5.0,right:8.0,weight:3.73588},{left:16.0,right:24.0,weight:4.90908}]}}},"fwd_phone":{metadata:{featureWeight:{weight:2.07327}}},"fwd_email_path":{metadata:{featureWeight:{weight:0.961773}}},"rev_phone_path":{metadata:{featureWeight:{weight:0.354484}}},"low_tweepcred_follow_path":{metadata:{featureWeight:{intervalWeights:[{left:4.0,right:5.0,weight:0.177209},{left:7.0,right:8.0,weight:0.12378},{left:3.0,right:4.0,weight:0.197566},{left:5.0,right:6.0,weight:0.15867},{left:2.0,right:3.0,weight:0.196539},{right:2.0,weight:0.1805},{left:75.0,weight:-0.424598},{left:6.0,right:7.0,weight:0.143698},{left:10.0,right:13.0,weight:0.0458502},{left:8.0,right:10.0,weight:0.0919314},{left:13.0,right:75.0,weight:-0.111484}]}}},"rev_email_path":{metadata:{featureWeight:{weight:0.654451}}},"rev_email":{metadata:{featureWeight:{weight:2.33859}}},"fwd_phone_path":{metadata:{featureWeight:{weight:0.210418}}}]} diff --git a/follow-recommendations-service/server/src/main/resources/quality/stp_models/20141223/trainingConfig b/follow-recommendations-service/server/src/main/resources/quality/stp_models/20141223/trainingConfig new file mode 100644 index 0000000000..ab38990dda --- /dev/null +++ b/follow-recommendations-service/server/src/main/resources/quality/stp_models/20141223/trainingConfig @@ -0,0 +1 @@ +{input:{context:"discover.prod",startDateTime:"",endDateTime:"",trainingFeatures:["STP_FEATURES":["fwd_email","mutual_follow_path","fwd_email_path","rev_phone_path","low_tweepcred_follow_path","rev_phone","fwd_phone","rev_email_path","rev_email","fwd_phone_path"]],engagementActions:["click","favorite","open_link","open","send_tweet","send_reply","retweet","reply","profile_click","follow"],impressionActions:["discard","results","impression"],dataFormat:1,dataPath:"",isLabeled:0},sample:{positiveSampleRatio:1.0,negativeSampleRatio:1.0,sampleType:1},split:{trainingDataSplitSize:0.5,testingDataSplitSize:0.5,splitType:2},transform:{},filter:{featureOptions:[]},join:{engagementRules:["discover"],contentIdType:"tweet",groupBucketSize:3600000},discretize:{}} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/BUILD b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/BUILD new file mode 100644 index 0000000000..4dcbaab445 --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/BUILD @@ -0,0 +1,48 @@ +scala_library( + compiler_option_sets = ["fatal_warnings"], + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/org/slf4j:slf4j-api", + "finagle/finagle-http/src/main/scala", + "finagle/finagle-thriftmux/src/main/scala", + "finatra-internal/decider/src/main/scala", + "finatra-internal/international/src/main/scala/com/twitter/finatra/international/modules", + "finatra-internal/mtls-http/src/main/scala", + "finatra-internal/mtls-thriftmux/src/main/scala", + "finatra/http-core/src/main/java/com/twitter/finatra/http", + "finatra/inject/inject-app/src/main/scala", + "finatra/inject/inject-core/src/main/scala", + "finatra/inject/inject-server/src/main/scala", + "finatra/inject/inject-thrift-client", + "finatra/jackson/src/main/scala/com/twitter/finatra/jackson/modules", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/addressbook", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/adserver", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/cache", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/deepbirdv2", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/email_storage_service", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/geoduck", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/gizmoduck", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/graph_feature_service", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/phone_storage_service", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/socialgraph", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/strato", + "follow-recommendations-service/server/src/main/resources", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/controllers", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/modules", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/services", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/services/exceptions", + "follow-recommendations-service/thrift/src/main/thrift:thrift-scala", + "geoduck/service/src/main/scala/com/twitter/geoduck/service/common/clientmodules", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/controllers", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/configapi", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/module", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/module/stringcenter", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/product/guice", + "twitter-server/server/src/main/scala", + "util/util-app/src/main/scala", + "util/util-core:scala", + "util/util-slf4j-api/src/main/scala", + ], +) diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/FollowRecommendationsServiceThriftServer.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/FollowRecommendationsServiceThriftServer.scala new file mode 100644 index 0000000000..fba889c2fb --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/FollowRecommendationsServiceThriftServer.scala @@ -0,0 +1,118 @@ +package com.twitter.follow_recommendations + +import com.google.inject.Module +import com.twitter.finagle.ThriftMux +import com.twitter.finatra.decider.modules.DeciderModule +import com.twitter.finatra.http.HttpServer +import com.twitter.finatra.http.routing.HttpRouter +import com.twitter.finatra.international.modules.I18nFactoryModule +import com.twitter.finatra.international.modules.LanguagesModule +import com.twitter.finatra.jackson.modules.ScalaObjectMapperModule +import com.twitter.finatra.mtls.http.{Mtls => HttpMtls} +import com.twitter.finatra.mtls.thriftmux.Mtls +import com.twitter.finatra.thrift.ThriftServer +import com.twitter.finatra.thrift.filters._ +import com.twitter.finagle.thrift.Protocols +import com.twitter.finatra.thrift.routing.ThriftRouter +import com.twitter.follow_recommendations.common.clients.addressbook.AddressbookModule +import com.twitter.follow_recommendations.common.clients.adserver.AdserverModule +import com.twitter.follow_recommendations.common.clients.cache.MemcacheModule +import com.twitter.follow_recommendations.common.clients.deepbirdv2.DeepBirdV2PredictionServiceClientModule +import com.twitter.follow_recommendations.common.clients.email_storage_service.EmailStorageServiceModule +import com.twitter.follow_recommendations.common.clients.geoduck.LocationServiceModule +import com.twitter.follow_recommendations.common.clients.gizmoduck.GizmoduckModule +import com.twitter.follow_recommendations.common.clients.graph_feature_service.GraphFeatureStoreModule +import com.twitter.follow_recommendations.common.clients.impression_store.ImpressionStoreModule +import com.twitter.follow_recommendations.common.clients.phone_storage_service.PhoneStorageServiceModule +import com.twitter.follow_recommendations.common.clients.socialgraph.SocialGraphModule +import com.twitter.follow_recommendations.common.clients.strato.StratoClientModule +import com.twitter.follow_recommendations.common.constants.ServiceConstants._ +import com.twitter.follow_recommendations.common.feature_hydration.sources.HydrationSourcesModule +import com.twitter.follow_recommendations.controllers.ThriftController +import com.twitter.follow_recommendations.modules._ +import com.twitter.follow_recommendations.service.exceptions.UnknownLoggingExceptionMapper +import com.twitter.follow_recommendations.services.FollowRecommendationsServiceWarmupHandler +import com.twitter.follow_recommendations.thriftscala.FollowRecommendationsThriftService +import com.twitter.geoduck.service.common.clientmodules.ReverseGeocoderThriftClientModule +import com.twitter.inject.thrift.filters.DarkTrafficFilter +import com.twitter.inject.thrift.modules.ThriftClientIdModule +import com.twitter.product_mixer.core.controllers.ProductMixerController +import com.twitter.product_mixer.core.module.PipelineExecutionLoggerModule +import com.twitter.product_mixer.core.module.product_mixer_flags.ProductMixerFlagModule +import com.twitter.product_mixer.core.module.stringcenter.ProductScopeStringCenterModule +import com.twitter.product_mixer.core.product.guice.ProductScopeModule + +object FollowRecommendationsServiceThriftServerMain extends FollowRecommendationsServiceThriftServer + +class FollowRecommendationsServiceThriftServer + extends ThriftServer + with Mtls + with HttpServer + with HttpMtls { + override val name: String = "follow-recommendations-service-server" + + override val modules: Seq[Module] = + Seq( + ABDeciderModule, + AddressbookModule, + AdserverModule, + ConfigApiModule, + DeciderModule, + DeepBirdV2PredictionServiceClientModule, + DiffyModule, + EmailStorageServiceModule, + FeaturesSwitchesModule, + FlagsModule, + GizmoduckModule, + GraphFeatureStoreModule, + HydrationSourcesModule, + I18nFactoryModule, + ImpressionStoreModule, + LanguagesModule, + LocationServiceModule, + MemcacheModule, + PhoneStorageServiceModule, + PipelineExecutionLoggerModule, + ProductMixerFlagModule, + ProductRegistryModule, + new ProductScopeModule(), + new ProductScopeStringCenterModule(), + new ReverseGeocoderThriftClientModule, + ScalaObjectMapperModule, + ScorerModule, + ScribeModule, + SocialGraphModule, + StratoClientModule, + ThriftClientIdModule, + TimerModule, + ) + + def configureThrift(router: ThriftRouter): Unit = { + router + .filter[LoggingMDCFilter] + .filter[TraceIdMDCFilter] + .filter[ThriftMDCFilter] + .filter[StatsFilter] + .filter[AccessLoggingFilter] + .filter[ExceptionMappingFilter] + .exceptionMapper[UnknownLoggingExceptionMapper] + .filter[DarkTrafficFilter[FollowRecommendationsThriftService.ReqRepServicePerEndpoint]] + .add[ThriftController] + } + + override def configureThriftServer(server: ThriftMux.Server): ThriftMux.Server = { + server.withProtocolFactory( + Protocols.binaryFactory( + stringLengthLimit = StringLengthLimit, + containerLengthLimit = ContainerLengthLimit)) + } + + override def configureHttp(router: HttpRouter): Unit = router.add( + ProductMixerController[FollowRecommendationsThriftService.MethodPerEndpoint]( + this.injector, + FollowRecommendationsThriftService.ExecutePipeline)) + + override def warmup(): Unit = { + handle[FollowRecommendationsServiceWarmupHandler]() + } +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/assembler/models/Action.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/assembler/models/Action.scala new file mode 100644 index 0000000000..cd3d889677 --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/assembler/models/Action.scala @@ -0,0 +1,9 @@ +package com.twitter.follow_recommendations.assembler.models + +import com.twitter.follow_recommendations.{thriftscala => t} + +case class Action(text: String, actionURL: String) { + lazy val toThrift: t.Action = { + t.Action(text, actionURL) + } +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/assembler/models/BUILD b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/assembler/models/BUILD new file mode 100644 index 0000000000..65d2654804 --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/assembler/models/BUILD @@ -0,0 +1,12 @@ +scala_library( + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models", + "follow-recommendations-service/thrift/src/main/thrift:thrift-scala", + "stringcenter/client", + ], + exports = [ + ], +) diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/assembler/models/Config.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/assembler/models/Config.scala new file mode 100644 index 0000000000..3346a05e1c --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/assembler/models/Config.scala @@ -0,0 +1,8 @@ +package com.twitter.follow_recommendations.assembler.models + +import com.twitter.stringcenter.client.core.ExternalString + +case class HeaderConfig(title: TitleConfig) +case class TitleConfig(text: ExternalString) +case class FooterConfig(actionConfig: Option[ActionConfig]) +case class ActionConfig(footerText: ExternalString, actionURL: String) diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/assembler/models/FeedbackAction.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/assembler/models/FeedbackAction.scala new file mode 100644 index 0000000000..25caa933a1 --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/assembler/models/FeedbackAction.scala @@ -0,0 +1,13 @@ +package com.twitter.follow_recommendations.assembler.models + +import com.twitter.follow_recommendations.{thriftscala => t} + +trait FeedbackAction { + def toThrift: t.FeedbackAction +} + +case class DismissUserId() extends FeedbackAction { + override lazy val toThrift: t.FeedbackAction = { + t.FeedbackAction.DismissUserId(t.DismissUserId()) + } +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/assembler/models/Footer.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/assembler/models/Footer.scala new file mode 100644 index 0000000000..f62368431e --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/assembler/models/Footer.scala @@ -0,0 +1,9 @@ +package com.twitter.follow_recommendations.assembler.models + +import com.twitter.follow_recommendations.{thriftscala => t} + +case class Footer(action: Option[Action]) { + lazy val toThrift: t.Footer = { + t.Footer(action.map(_.toThrift)) + } +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/assembler/models/Header.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/assembler/models/Header.scala new file mode 100644 index 0000000000..58c60c789d --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/assembler/models/Header.scala @@ -0,0 +1,9 @@ +package com.twitter.follow_recommendations.assembler.models + +import com.twitter.follow_recommendations.{thriftscala => t} + +case class Header(title: Title) { + lazy val toThrift: t.Header = { + t.Header(title.toThrift) + } +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/assembler/models/Layout.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/assembler/models/Layout.scala new file mode 100644 index 0000000000..f0dc9630af --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/assembler/models/Layout.scala @@ -0,0 +1,16 @@ +package com.twitter.follow_recommendations.assembler.models + +sealed trait Layout + +case class UserListLayout( + header: Option[HeaderConfig], + userListOptions: UserListOptions, + socialProofs: Option[Seq[SocialProof]], + footer: Option[FooterConfig]) + extends Layout + +case class CarouselLayout( + header: Option[HeaderConfig], + carouselOptions: CarouselOptions, + socialProofs: Option[Seq[SocialProof]]) + extends Layout diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/assembler/models/RecommendationOptions.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/assembler/models/RecommendationOptions.scala new file mode 100644 index 0000000000..72351e0339 --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/assembler/models/RecommendationOptions.scala @@ -0,0 +1,11 @@ +package com.twitter.follow_recommendations.assembler.models + +sealed trait RecommendationOptions + +case class UserListOptions( + userBioEnabled: Boolean, + userBioTruncated: Boolean, + userBioMaxLines: Option[Long], +) extends RecommendationOptions + +case class CarouselOptions() extends RecommendationOptions diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/assembler/models/SocialProof.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/assembler/models/SocialProof.scala new file mode 100644 index 0000000000..fd5878af85 --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/assembler/models/SocialProof.scala @@ -0,0 +1,16 @@ +package com.twitter.follow_recommendations.assembler.models + +import com.twitter.stringcenter.client.core.ExternalString + +sealed trait SocialProof + +case class GeoContextProof(popularInCountryText: ExternalString) extends SocialProof +case class FollowedByUsersProof(text1: ExternalString, text2: ExternalString, textN: ExternalString) + extends SocialProof + +sealed trait SocialText { + def text: String +} + +case class GeoSocialText(text: String) extends SocialText +case class FollowedByUsersText(text: String) extends SocialText diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/assembler/models/Title.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/assembler/models/Title.scala new file mode 100644 index 0000000000..3d128e7c8a --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/assembler/models/Title.scala @@ -0,0 +1,9 @@ +package com.twitter.follow_recommendations.assembler.models + +import com.twitter.follow_recommendations.{thriftscala => t} + +case class Title(text: String) { + lazy val toThrift: t.Title = { + t.Title(text) + } +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/assembler/models/WTFPresentation.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/assembler/models/WTFPresentation.scala new file mode 100644 index 0000000000..7cfda18462 --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/assembler/models/WTFPresentation.scala @@ -0,0 +1,47 @@ +package com.twitter.follow_recommendations.assembler.models + +import com.twitter.follow_recommendations.{thriftscala => t} + +trait WTFPresentation { + def toThrift: t.WTFPresentation +} + +case class UserList( + userBioEnabled: Boolean, + userBioTruncated: Boolean, + userBioMaxLines: Option[Long], + feedbackAction: Option[FeedbackAction]) + extends WTFPresentation { + def toThrift: t.WTFPresentation = { + t.WTFPresentation.UserBioList( + t.UserList(userBioEnabled, userBioTruncated, userBioMaxLines, feedbackAction.map(_.toThrift))) + } +} + +object UserList { + def fromUserListOptions( + userListOptions: UserListOptions + ): UserList = { + UserList( + userListOptions.userBioEnabled, + userListOptions.userBioTruncated, + userListOptions.userBioMaxLines, + None) + } +} + +case class Carousel( + feedbackAction: Option[FeedbackAction]) + extends WTFPresentation { + def toThrift: t.WTFPresentation = { + t.WTFPresentation.Carousel(t.Carousel(feedbackAction.map(_.toThrift))) + } +} + +object Carousel { + def fromCarouselOptions( + carouselOptions: CarouselOptions + ): Carousel = { + Carousel(None) + } +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/blenders/BUILD b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/blenders/BUILD new file mode 100644 index 0000000000..81d912a998 --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/blenders/BUILD @@ -0,0 +1,16 @@ +scala_library( + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/com/google/inject:guice", + "3rdparty/jvm/com/google/inject/extensions:guice-assistedinject", + "3rdparty/jvm/net/codingwell:scala-guice", + "3rdparty/jvm/org/slf4j:slf4j-api", + "finatra/inject/inject-core/src/main/scala", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/models", + "util/util-slf4j-api/src/main/scala", + ], +) diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/blenders/PromotedAccountsBlender.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/blenders/PromotedAccountsBlender.scala new file mode 100644 index 0000000000..8516de53d8 --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/blenders/PromotedAccountsBlender.scala @@ -0,0 +1,138 @@ +package com.twitter.follow_recommendations.blenders + +import com.google.common.annotations.VisibleForTesting +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.follow_recommendations.common.base.Transform +import com.twitter.follow_recommendations.common.models.AdMetadata +import com.twitter.follow_recommendations.common.models.Recommendation +import com.twitter.inject.Logging +import com.twitter.stitch.Stitch +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class PromotedAccountsBlender @Inject() (statsReceiver: StatsReceiver) + extends Transform[Int, Recommendation] + with Logging { + + import PromotedAccountsBlender._ + val stats = statsReceiver.scope(Name) + val inputOrganicAccounts = stats.counter(InputOrganic) + val inputPromotedAccounts = stats.counter(InputPromoted) + val outputOrganicAccounts = stats.counter(OutputOrganic) + val outputPromotedAccounts = stats.counter(OutputPromoted) + val promotedAccountsStats = stats.scope(NumPromotedAccounts) + + override def transform( + maxResults: Int, + items: Seq[Recommendation] + ): Stitch[Seq[Recommendation]] = { + val (promoted, organic) = items.partition(_.isPromotedAccount) + val promotedIds = promoted.map(_.id).toSet + val dedupedOrganic = organic.filterNot(u => promotedIds.contains(u.id)) + val blended = blendPromotedAccount(dedupedOrganic, promoted, maxResults) + val (outputPromoted, outputOrganic) = blended.partition(_.isPromotedAccount) + inputOrganicAccounts.incr(dedupedOrganic.size) + inputPromotedAccounts.incr(promoted.size) + outputOrganicAccounts.incr(outputOrganic.size) + val size = outputPromoted.size + outputPromotedAccounts.incr(size) + if (size <= 5) { + promotedAccountsStats.counter(outputPromoted.size.toString).incr() + } else { + promotedAccountsStats.counter(MoreThan5Promoted).incr() + } + Stitch.value(blended) + } + + /** + * Merge Promoted results and organic results. Promoted result dictates the position + * in the merge list. + * + * merge a list of positioned users, aka. promoted, and a list of organic + * users. The positioned promoted users are pre-sorted with regards to their + * position ascendingly. Only requirement about position is to be within the + * range, i.e, can not exceed the combined length if merge is successful, ok + * to be at the last position, but not beyond. + * For more detailed description of location position: + * http://confluence.local.twitter.com/display/ADS/Promoted+Tweets+in+Timeline+Design+Document + */ + @VisibleForTesting + private[blenders] def mergePromotedAccounts( + organicUsers: Seq[Recommendation], + promotedUsers: Seq[Recommendation] + ): Seq[Recommendation] = { + def mergeAccountWithIndex( + organicUsers: Seq[Recommendation], + promotedUsers: Seq[Recommendation], + index: Int + ): Stream[Recommendation] = { + if (promotedUsers.isEmpty) organicUsers.toStream + else { + val promotedHead = promotedUsers.head + val promotedTail = promotedUsers.tail + promotedHead.adMetadata match { + case Some(AdMetadata(position, _)) => + if (position < 0) mergeAccountWithIndex(organicUsers, promotedTail, index) + else if (position == index) + promotedHead #:: mergeAccountWithIndex(organicUsers, promotedTail, index) + else if (organicUsers.isEmpty) organicUsers.toStream + else { + val organicHead = organicUsers.head + val organicTail = organicUsers.tail + organicHead #:: mergeAccountWithIndex(organicTail, promotedUsers, index + 1) + } + case _ => + logger.error("Unknown Candidate type in mergePromotedAccounts") + Stream.empty + } + } + } + + mergeAccountWithIndex(organicUsers, promotedUsers, 0) + } + + private[this] def blendPromotedAccount( + organic: Seq[Recommendation], + promoted: Seq[Recommendation], + maxResults: Int + ): Seq[Recommendation] = { + + val merged = mergePromotedAccounts(organic, promoted) + val mergedServed = merged.take(maxResults) + val promotedServed = promoted.intersect(mergedServed) + + if (isBlendPromotedNeeded( + mergedServed.size - promotedServed.size, + promotedServed.size, + maxResults + )) { + mergedServed + } else { + organic.take(maxResults) + } + } + + @VisibleForTesting + private[blenders] def isBlendPromotedNeeded( + organicSize: Int, + promotedSize: Int, + maxResults: Int + ): Boolean = { + (organicSize > 1) && + (promotedSize > 0) && + (promotedSize < organicSize) && + (promotedSize <= 2) && + (maxResults > 1) + } +} + +object PromotedAccountsBlender { + val Name = "promoted_accounts_blender" + val InputOrganic = "input_organic_accounts" + val InputPromoted = "input_promoted_accounts" + val OutputOrganic = "output_organic_accounts" + val OutputPromoted = "output_promoted_accounts" + val NumPromotedAccounts = "num_promoted_accounts" + val MoreThan5Promoted = "more_than_5" +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/BUILD b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/BUILD new file mode 100644 index 0000000000..f64fcda692 --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/BUILD @@ -0,0 +1,28 @@ +scala_library( + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/com/google/inject:guice", + "configapi/configapi-core", + "configapi/configapi-decider", + "configapi/configapi-featureswitches:v2", + "featureswitches/featureswitches-core", + "featureswitches/featureswitches-core:v2", + "featureswitches/featureswitches-core/src/main/scala/com/twitter/featureswitches/v2/builder", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/crowd_search_accounts", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/real_graph", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/recent_engagement", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/top_organic_follows_accounts", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/triangular_loops", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/ml_ranker/ranking", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/common", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/deciders", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/params", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/content_recommender_flow", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products", + ], +) diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/ConfigBuilder.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/ConfigBuilder.scala new file mode 100644 index 0000000000..818f4402cd --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/ConfigBuilder.scala @@ -0,0 +1,16 @@ +package com.twitter.follow_recommendations.configapi + +import com.twitter.timelines.configapi.CompositeConfig +import com.twitter.timelines.configapi.Config +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class ConfigBuilder @Inject() ( + deciderConfigs: DeciderConfigs, + featureSwitchConfigs: FeatureSwitchConfigs) { + // The order of configs added to `CompositeConfig` is important. The config will be matched with + // the first possible rule. So, current setup will give priority to Deciders instead of FS + def build(): Config = + new CompositeConfig(Seq(deciderConfigs.config, featureSwitchConfigs.config)) +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/DeciderConfigs.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/DeciderConfigs.scala new file mode 100644 index 0000000000..0154a17033 --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/DeciderConfigs.scala @@ -0,0 +1,52 @@ +package com.twitter.follow_recommendations.configapi + +import com.twitter.decider.Recipient +import com.twitter.decider.SimpleRecipient +import com.twitter.follow_recommendations.configapi.deciders.DeciderKey +import com.twitter.follow_recommendations.configapi.deciders.DeciderParams +import com.twitter.follow_recommendations.products.home_timeline_tweet_recs.configapi.HomeTimelineTweetRecsParams +import com.twitter.servo.decider.DeciderGateBuilder +import com.twitter.timelines.configapi._ +import com.twitter.timelines.configapi.decider.DeciderSwitchOverrideValue +import com.twitter.timelines.configapi.decider.GuestRecipient +import com.twitter.timelines.configapi.decider.RecipientBuilder +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class DeciderConfigs @Inject() (deciderGateBuilder: DeciderGateBuilder) { + val overrides: Seq[OptionalOverride[_]] = DeciderConfigs.ParamsToDeciderMap.map { + case (params, deciderKey) => + params.optionalOverrideValue( + DeciderSwitchOverrideValue( + feature = deciderGateBuilder.keyToFeature(deciderKey), + enabledValue = true, + recipientBuilder = DeciderConfigs.UserOrGuestOrRequest + ) + ) + }.toSeq + + val config: BaseConfig = BaseConfigBuilder(overrides).build("FollowRecommendationServiceDeciders") +} + +object DeciderConfigs { + val ParamsToDeciderMap = Map( + DeciderParams.EnableRecommendations -> DeciderKey.EnableRecommendations, + DeciderParams.EnableScoreUserCandidates -> DeciderKey.EnableScoreUserCandidates, + HomeTimelineTweetRecsParams.EnableProduct -> DeciderKey.EnableHomeTimelineTweetRecsProduct, + ) + + object UserOrGuestOrRequest extends RecipientBuilder { + + def apply(requestContext: BaseRequestContext): Option[Recipient] = requestContext match { + case c: WithUserId if c.userId.isDefined => + c.userId.map(SimpleRecipient) + case c: WithGuestId if c.guestId.isDefined => + c.guestId.map(GuestRecipient) + case c: WithGuestId => + RecipientBuilder.Request(c) + case _ => + throw new UndefinedUserIdNorGuestIDException(requestContext) + } + } +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/FeatureSwitchConfigs.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/FeatureSwitchConfigs.scala new file mode 100644 index 0000000000..c7f7f6d9e6 --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/FeatureSwitchConfigs.scala @@ -0,0 +1,138 @@ +package com.twitter.follow_recommendations.configapi + +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.follow_recommendations.common.candidate_sources.base.SocialProofEnforcedCandidateSourceFSConfig +import com.twitter.follow_recommendations.common.candidate_sources.crowd_search_accounts.CrowdSearchAccountsFSConfig +import com.twitter.follow_recommendations.common.candidate_sources.geo.PopGeoQualityFollowSourceFSConfig +import com.twitter.follow_recommendations.common.candidate_sources.top_organic_follows_accounts.TopOrganicFollowsAccountsFSConfig +import com.twitter.follow_recommendations.common.candidate_sources.geo.PopGeoSourceFSConfig +import com.twitter.follow_recommendations.common.candidate_sources.ppmi_locale_follow.PPMILocaleFollowSourceFSConfig +import com.twitter.follow_recommendations.common.candidate_sources.real_graph.RealGraphOonFSConfig +import com.twitter.follow_recommendations.common.candidate_sources.recent_engagement.RepeatedProfileVisitsFSConfig +import com.twitter.follow_recommendations.common.candidate_sources.sims.SimsSourceFSConfig +import com.twitter.follow_recommendations.common.candidate_sources.sims_expansion.RecentEngagementSimilarUsersFSConfig +import com.twitter.follow_recommendations.common.candidate_sources.sims_expansion.SimsExpansionFSConfig +import com.twitter.follow_recommendations.common.candidate_sources.socialgraph.RecentFollowingRecentFollowingExpansionSourceFSConfig +import com.twitter.follow_recommendations.common.candidate_sources.stp.OfflineStpSourceFsConfig +import com.twitter.follow_recommendations.common.candidate_sources.stp.OnlineSTPSourceFSConfig +import com.twitter.follow_recommendations.common.candidate_sources.triangular_loops.TriangularLoopsFSConfig +import com.twitter.follow_recommendations.common.candidate_sources.user_user_graph.UserUserGraphFSConfig +import com.twitter.follow_recommendations.common.feature_hydration.sources.FeatureHydrationSourcesFSConfig +import com.twitter.follow_recommendations.common.rankers.weighted_candidate_source_ranker.WeightedCandidateSourceRankerFSConfig +import com.twitter.follow_recommendations.configapi.common.FeatureSwitchConfig +import com.twitter.follow_recommendations.flows.content_recommender_flow.ContentRecommenderFlowFSConfig +import com.twitter.follow_recommendations.common.predicates.gizmoduck.GizmoduckPredicateFSConfig +import com.twitter.follow_recommendations.common.predicates.hss.HssPredicateFSConfig +import com.twitter.follow_recommendations.common.predicates.sgs.SgsPredicateFSConfig +import com.twitter.follow_recommendations.flows.post_nux_ml.PostNuxMlFlowFSConfig +import com.twitter.logging.Logger +import com.twitter.timelines.configapi.BaseConfigBuilder +import com.twitter.timelines.configapi.FeatureSwitchOverrideUtil + +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class FeatureSwitchConfigs @Inject() ( + globalFeatureSwitchConfig: GlobalFeatureSwitchConfig, + featureHydrationSourcesFSConfig: FeatureHydrationSourcesFSConfig, + weightedCandidateSourceRankerFSConfig: WeightedCandidateSourceRankerFSConfig, + // Flow related config + contentRecommenderFlowFSConfig: ContentRecommenderFlowFSConfig, + postNuxMlFlowFSConfig: PostNuxMlFlowFSConfig, + // Candidate source related config + crowdSearchAccountsFSConfig: CrowdSearchAccountsFSConfig, + offlineStpSourceFsConfig: OfflineStpSourceFsConfig, + onlineSTPSourceFSConfig: OnlineSTPSourceFSConfig, + popGeoSourceFSConfig: PopGeoSourceFSConfig, + popGeoQualityFollowFSConfig: PopGeoQualityFollowSourceFSConfig, + realGraphOonFSConfig: RealGraphOonFSConfig, + repeatedProfileVisitsFSConfig: RepeatedProfileVisitsFSConfig, + recentEngagementSimilarUsersFSConfig: RecentEngagementSimilarUsersFSConfig, + recentFollowingRecentFollowingExpansionSourceFSConfig: RecentFollowingRecentFollowingExpansionSourceFSConfig, + simsExpansionFSConfig: SimsExpansionFSConfig, + simsSourceFSConfig: SimsSourceFSConfig, + socialProofEnforcedCandidateSourceFSConfig: SocialProofEnforcedCandidateSourceFSConfig, + triangularLoopsFSConfig: TriangularLoopsFSConfig, + userUserGraphFSConfig: UserUserGraphFSConfig, + // Predicate related configs + gizmoduckPredicateFSConfig: GizmoduckPredicateFSConfig, + hssPredicateFSConfig: HssPredicateFSConfig, + sgsPredicateFSConfig: SgsPredicateFSConfig, + ppmiLocaleSourceFSConfig: PPMILocaleFollowSourceFSConfig, + topOrganicFollowsAccountsFSConfig: TopOrganicFollowsAccountsFSConfig, + statsReceiver: StatsReceiver) { + + val logger = Logger(classOf[FeatureSwitchConfigs]) + + val mergedFSConfig = + FeatureSwitchConfig.merge( + Seq( + globalFeatureSwitchConfig, + featureHydrationSourcesFSConfig, + weightedCandidateSourceRankerFSConfig, + // Flow related config + contentRecommenderFlowFSConfig, + postNuxMlFlowFSConfig, + // Candidate source related config + crowdSearchAccountsFSConfig, + offlineStpSourceFsConfig, + onlineSTPSourceFSConfig, + popGeoSourceFSConfig, + popGeoQualityFollowFSConfig, + realGraphOonFSConfig, + repeatedProfileVisitsFSConfig, + recentEngagementSimilarUsersFSConfig, + recentFollowingRecentFollowingExpansionSourceFSConfig, + simsExpansionFSConfig, + simsSourceFSConfig, + socialProofEnforcedCandidateSourceFSConfig, + triangularLoopsFSConfig, + userUserGraphFSConfig, + // Predicate related configs: + gizmoduckPredicateFSConfig, + hssPredicateFSConfig, + sgsPredicateFSConfig, + ppmiLocaleSourceFSConfig, + topOrganicFollowsAccountsFSConfig, + ) + ) + + /** + * enum params have to be listed in this main file together as otherwise we'll have to pass in + * some signature like `Seq[FSEnumParams[_]]` which are generics of generics and won't compile. + * we only have enumFsParams from globalFeatureSwitchConfig at the moment + */ + val enumOverrides = globalFeatureSwitchConfig.enumFsParams.flatMap { enumParam => + FeatureSwitchOverrideUtil.getEnumFSOverrides(statsReceiver, logger, enumParam) + } + + val gatedOverrides = mergedFSConfig.gatedOverridesMap.flatMap { + case (fsName, overrides) => + FeatureSwitchOverrideUtil.gatedOverrides(fsName, overrides: _*) + } + + val enumSeqOverrides = globalFeatureSwitchConfig.enumSeqFsParams.flatMap { enumSeqParam => + FeatureSwitchOverrideUtil.getEnumSeqFSOverrides(statsReceiver, logger, enumSeqParam) + } + + val overrides = + FeatureSwitchOverrideUtil + .getBooleanFSOverrides(mergedFSConfig.booleanFSParams: _*) ++ + FeatureSwitchOverrideUtil + .getBoundedIntFSOverrides(mergedFSConfig.intFSParams: _*) ++ + FeatureSwitchOverrideUtil + .getBoundedLongFSOverrides(mergedFSConfig.longFSParams: _*) ++ + FeatureSwitchOverrideUtil + .getBoundedDoubleFSOverrides(mergedFSConfig.doubleFSParams: _*) ++ + FeatureSwitchOverrideUtil + .getDurationFSOverrides(mergedFSConfig.durationFSParams: _*) ++ + FeatureSwitchOverrideUtil + .getBoundedOptionalDoubleOverrides(mergedFSConfig.optionalDoubleFSParams: _*) ++ + FeatureSwitchOverrideUtil.getStringSeqFSOverrides(mergedFSConfig.stringSeqFSParams: _*) ++ + enumOverrides ++ + gatedOverrides ++ + enumSeqOverrides + + val config = BaseConfigBuilder(overrides).build("FollowRecommendationServiceFeatureSwitches") +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/GlobalFeatureSwitchConfig.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/GlobalFeatureSwitchConfig.scala new file mode 100644 index 0000000000..537ed660cf --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/GlobalFeatureSwitchConfig.scala @@ -0,0 +1,49 @@ +package com.twitter.follow_recommendations.configapi + +import com.twitter.follow_recommendations.common.candidate_sources.crowd_search_accounts.CrowdSearchAccountsParams.AccountsFilteringAndRankingLogics +import com.twitter.follow_recommendations.common.candidate_sources.top_organic_follows_accounts.TopOrganicFollowsAccountsParams.{ + AccountsFilteringAndRankingLogics => OrganicAccountsFilteringAndRankingLogics +} +import com.twitter.follow_recommendations.common.candidate_sources.sims_expansion.RecentEngagementSimilarUsersParams +import com.twitter.follow_recommendations.common.candidate_sources.sims_expansion.SimsExpansionSourceParams +import com.twitter.follow_recommendations.common.rankers.ml_ranker.ranking.MlRankerParams.CandidateScorerIdParam +import com.twitter.follow_recommendations.configapi.common.FeatureSwitchConfig +import com.twitter.follow_recommendations.configapi.params.GlobalParams.CandidateSourcesToFilter +import com.twitter.follow_recommendations.configapi.params.GlobalParams.EnableCandidateParamHydrations +import com.twitter.follow_recommendations.configapi.params.GlobalParams.EnableGFSSocialProofTransform +import com.twitter.follow_recommendations.configapi.params.GlobalParams.EnableRecommendationFlowLogs +import com.twitter.follow_recommendations.configapi.params.GlobalParams.EnableWhoToFollowProducts +import com.twitter.follow_recommendations.configapi.params.GlobalParams.KeepSocialUserCandidate +import com.twitter.follow_recommendations.configapi.params.GlobalParams.KeepUserCandidate +import com.twitter.timelines.configapi.FSName +import com.twitter.timelines.configapi.Param +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class GlobalFeatureSwitchConfig @Inject() () extends FeatureSwitchConfig { + override val booleanFSParams: Seq[Param[Boolean] with FSName] = { + Seq( + EnableCandidateParamHydrations, + KeepUserCandidate, + KeepSocialUserCandidate, + EnableGFSSocialProofTransform, + EnableWhoToFollowProducts, + EnableRecommendationFlowLogs + ) + } + + val enumFsParams = + Seq( + CandidateScorerIdParam, + SimsExpansionSourceParams.Aggregator, + RecentEngagementSimilarUsersParams.Aggregator, + CandidateSourcesToFilter, + ) + + val enumSeqFsParams = + Seq( + AccountsFilteringAndRankingLogics, + OrganicAccountsFilteringAndRankingLogics + ) +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/ParamsFactory.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/ParamsFactory.scala new file mode 100644 index 0000000000..847d699624 --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/ParamsFactory.scala @@ -0,0 +1,29 @@ +package com.twitter.follow_recommendations.configapi + +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.follow_recommendations.common.models.DisplayLocation +import com.twitter.product_mixer.core.model.marshalling.request.ClientContext +import com.twitter.servo.util.MemoizingStatsReceiver +import com.twitter.timelines.configapi.Config +import com.twitter.timelines.configapi.FeatureValue +import com.twitter.timelines.configapi.Params +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class ParamsFactory @Inject() ( + config: Config, + requestContextFactory: RequestContextFactory, + statsReceiver: StatsReceiver) { + + private val stats = new MemoizingStatsReceiver(statsReceiver.scope("configapi")) + def apply(followRecommendationServiceRequestContext: RequestContext): Params = + config(followRecommendationServiceRequestContext, stats) + + def apply( + clientContext: ClientContext, + displayLocation: DisplayLocation, + featureOverrides: Map[String, FeatureValue] + ): Params = + apply(requestContextFactory(clientContext, displayLocation, featureOverrides)) +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/RequestContext.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/RequestContext.scala new file mode 100644 index 0000000000..ebc8abf3ce --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/RequestContext.scala @@ -0,0 +1,19 @@ +package com.twitter.follow_recommendations.configapi + +import com.twitter.timelines.configapi.BaseRequestContext +import com.twitter.timelines.configapi.FeatureContext +import com.twitter.timelines.configapi.NullFeatureContext +import com.twitter.timelines.configapi.GuestId +import com.twitter.timelines.configapi.UserId +import com.twitter.timelines.configapi.WithFeatureContext +import com.twitter.timelines.configapi.WithGuestId +import com.twitter.timelines.configapi.WithUserId + +case class RequestContext( + userId: Option[UserId], + guestId: Option[GuestId], + featureContext: FeatureContext = NullFeatureContext) + extends BaseRequestContext + with WithUserId + with WithGuestId + with WithFeatureContext diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/RequestContextFactory.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/RequestContextFactory.scala new file mode 100644 index 0000000000..89d8617a3a --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/RequestContextFactory.scala @@ -0,0 +1,74 @@ +package com.twitter.follow_recommendations.configapi + +import com.google.common.annotations.VisibleForTesting +import com.google.inject.Inject +import com.twitter.decider.Decider +import com.twitter.featureswitches.v2.FeatureSwitches +import com.twitter.featureswitches.{Recipient => FeatureSwitchRecipient} +import com.twitter.follow_recommendations.common.models.DisplayLocation +import com.twitter.product_mixer.core.model.marshalling.request.ClientContext +import com.twitter.snowflake.id.SnowflakeId +import com.twitter.timelines.configapi.FeatureContext +import com.twitter.timelines.configapi.FeatureValue +import com.twitter.timelines.configapi.ForcedFeatureContext +import com.twitter.timelines.configapi.OrElseFeatureContext +import com.twitter.timelines.configapi.featureswitches.v2.FeatureSwitchResultsFeatureContext +import javax.inject.Singleton + +/* + * Request Context Factory is used to build RequestContext objects which are used + * by the config api to determine the param overrides to apply to the request. + * The param overrides are determined per request by configs which specify which + * FS/Deciders/AB translate to what param overrides. + */ +@Singleton +class RequestContextFactory @Inject() (featureSwitches: FeatureSwitches, decider: Decider) { + def apply( + clientContext: ClientContext, + displayLocation: DisplayLocation, + featureOverrides: Map[String, FeatureValue] + ): RequestContext = { + val featureContext = getFeatureContext(clientContext, displayLocation, featureOverrides) + RequestContext(clientContext.userId, clientContext.guestId, featureContext) + } + + private[configapi] def getFeatureContext( + clientContext: ClientContext, + displayLocation: DisplayLocation, + featureOverrides: Map[String, FeatureValue] + ): FeatureContext = { + val recipient = + getFeatureSwitchRecipient(clientContext) + .withCustomFields("display_location" -> displayLocation.toFsName) + + // userAgeOpt is going to be set to None for logged out users and defaulted to Some(Int.MaxValue) for non-snowflake users + val userAgeOpt = clientContext.userId.map { userId => + SnowflakeId.timeFromIdOpt(userId).map(_.untilNow.inDays).getOrElse(Int.MaxValue) + } + val recipientWithAccountAge = + userAgeOpt + .map(age => recipient.withCustomFields("account_age_in_days" -> age)).getOrElse(recipient) + + val results = featureSwitches.matchRecipient(recipientWithAccountAge) + OrElseFeatureContext( + ForcedFeatureContext(featureOverrides), + new FeatureSwitchResultsFeatureContext(results)) + } + + @VisibleForTesting + private[configapi] def getFeatureSwitchRecipient( + clientContext: ClientContext + ): FeatureSwitchRecipient = { + FeatureSwitchRecipient( + userId = clientContext.userId, + userRoles = clientContext.userRoles, + deviceId = clientContext.deviceId, + guestId = clientContext.guestId, + languageCode = clientContext.languageCode, + countryCode = clientContext.countryCode, + isVerified = None, + clientApplicationId = clientContext.appId, + isTwoffice = clientContext.isTwoffice + ) + } +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/candidates/BUILD b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/candidates/BUILD new file mode 100644 index 0000000000..5470c9bf43 --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/candidates/BUILD @@ -0,0 +1,18 @@ +scala_library( + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/com/google/inject:guice", + "configapi/configapi-core", + "configapi/configapi-decider", + "configapi/configapi-featureswitches:v2", + "featureswitches/featureswitches-core", + "featureswitches/featureswitches-core:v2", + "featureswitches/featureswitches-core/src/main/scala/com/twitter/featureswitches/v2/builder", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/deciders", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/params", + ], +) diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/candidates/CandidateUserContext.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/candidates/CandidateUserContext.scala new file mode 100644 index 0000000000..6b954a9116 --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/candidates/CandidateUserContext.scala @@ -0,0 +1,19 @@ +package com.twitter.follow_recommendations.configapi.candidates + +import com.twitter.timelines.configapi.BaseRequestContext +import com.twitter.timelines.configapi.FeatureContext +import com.twitter.timelines.configapi.NullFeatureContext +import com.twitter.timelines.configapi.WithFeatureContext +import com.twitter.timelines.configapi.WithUserId + +/** + * represent the context for a recommendation candidate (producer side) + * @param userId id of the recommended user + * @param featureContext feature context + */ +case class CandidateUserContext( + override val userId: Option[Long], + featureContext: FeatureContext = NullFeatureContext) + extends BaseRequestContext + with WithUserId + with WithFeatureContext diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/candidates/CandidateUserContextFactory.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/candidates/CandidateUserContextFactory.scala new file mode 100644 index 0000000000..4f30cf17a7 --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/candidates/CandidateUserContextFactory.scala @@ -0,0 +1,55 @@ +package com.twitter.follow_recommendations.configapi.candidates + +import com.google.common.annotations.VisibleForTesting +import com.google.inject.Inject +import com.twitter.decider.Decider +import com.twitter.featureswitches.v2.FeatureSwitches +import com.twitter.featureswitches.{Recipient => FeatureSwitchRecipient} +import com.twitter.follow_recommendations.common.constants.GuiceNamedConstants.PRODUCER_SIDE_FEATURE_SWITCHES +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.follow_recommendations.common.models.DisplayLocation +import com.twitter.timelines.configapi.FeatureContext +import com.twitter.timelines.configapi.featureswitches.v2.FeatureSwitchResultsFeatureContext +import javax.inject.Named +import javax.inject.Singleton + +@Singleton +class CandidateUserContextFactory @Inject() ( + @Named(PRODUCER_SIDE_FEATURE_SWITCHES) featureSwitches: FeatureSwitches, + decider: Decider) { + def apply( + candidateUser: CandidateUser, + displayLocation: DisplayLocation + ): CandidateUserContext = { + val featureContext = getFeatureContext(candidateUser, displayLocation) + + CandidateUserContext(Some(candidateUser.id), featureContext) + } + + private[configapi] def getFeatureContext( + candidateUser: CandidateUser, + displayLocation: DisplayLocation + ): FeatureContext = { + + val recipient = getFeatureSwitchRecipient(candidateUser).withCustomFields( + "display_location" -> displayLocation.toFsName) + new FeatureSwitchResultsFeatureContext(featureSwitches.matchRecipient(recipient)) + } + + @VisibleForTesting + private[configapi] def getFeatureSwitchRecipient( + candidateUser: CandidateUser + ): FeatureSwitchRecipient = { + FeatureSwitchRecipient( + userId = Some(candidateUser.id), + userRoles = None, + deviceId = None, + guestId = None, + languageCode = None, + countryCode = None, + isVerified = None, + clientApplicationId = None, + isTwoffice = None + ) + } +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/candidates/CandidateUserParamsFactory.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/candidates/CandidateUserParamsFactory.scala new file mode 100644 index 0000000000..5afd09a632 --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/candidates/CandidateUserParamsFactory.scala @@ -0,0 +1,35 @@ +package com.twitter.follow_recommendations.configapi.candidates + +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.follow_recommendations.common.models.HasDisplayLocation +import com.twitter.follow_recommendations.configapi.params.GlobalParams +import com.twitter.servo.util.MemoizingStatsReceiver +import com.twitter.timelines.configapi.Config +import com.twitter.timelines.configapi.HasParams +import com.twitter.timelines.configapi.Params +import javax.inject.Inject +import javax.inject.Singleton + +/** + * CandidateParamsFactory is primarily used for "producer side" experiments, don't use it on consumer side experiments + */ +@Singleton +class CandidateUserParamsFactory[T <: HasParams with HasDisplayLocation] @Inject() ( + config: Config, + candidateContextFactory: CandidateUserContextFactory, + statsReceiver: StatsReceiver) { + private val stats = new MemoizingStatsReceiver(statsReceiver.scope("configapi_candidate_params")) + def apply(candidateContext: CandidateUser, request: T): CandidateUser = { + if (candidateContext.params == Params.Invalid) { + if (request.params(GlobalParams.EnableCandidateParamHydrations)) { + candidateContext.copy(params = + config(candidateContextFactory(candidateContext, request.displayLocation), stats)) + } else { + candidateContext.copy(params = Params.Empty) + } + } else { + candidateContext + } + } +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/candidates/HydrateCandidateParamsTransform.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/candidates/HydrateCandidateParamsTransform.scala new file mode 100644 index 0000000000..deadd1d705 --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/candidates/HydrateCandidateParamsTransform.scala @@ -0,0 +1,21 @@ +package com.twitter.follow_recommendations.configapi.candidates + +import com.google.inject.Inject +import com.google.inject.Singleton +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.follow_recommendations.common.models.HasDisplayLocation +import com.twitter.follow_recommendations.common.base.Transform +import com.twitter.stitch.Stitch +import com.twitter.timelines.configapi.HasParams +import com.twitter.util.logging.Logging + +@Singleton +class HydrateCandidateParamsTransform[Target <: HasParams with HasDisplayLocation] @Inject() ( + candidateParamsFactory: CandidateUserParamsFactory[Target]) + extends Transform[Target, CandidateUser] + with Logging { + + def transform(target: Target, candidates: Seq[CandidateUser]): Stitch[Seq[CandidateUser]] = { + Stitch.value(candidates.map(candidateParamsFactory.apply(_, target))) + } +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/common/BUILD b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/common/BUILD new file mode 100644 index 0000000000..6fee24f895 --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/common/BUILD @@ -0,0 +1,8 @@ +scala_library( + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "configapi/configapi-core", + ], +) diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/common/FeatureSwitchConfig.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/common/FeatureSwitchConfig.scala new file mode 100644 index 0000000000..798b026704 --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/common/FeatureSwitchConfig.scala @@ -0,0 +1,60 @@ +package com.twitter.follow_recommendations.configapi.common + +import com.twitter.timelines.configapi.FeatureSwitchOverrideUtil.DefinedFeatureName +import com.twitter.timelines.configapi.FeatureSwitchOverrideUtil.ValueFeatureName +import com.twitter.timelines.configapi.BoundedParam +import com.twitter.timelines.configapi.FSBoundedParam +import com.twitter.timelines.configapi.FSName +import com.twitter.timelines.configapi.HasDurationConversion +import com.twitter.timelines.configapi.OptionalOverride +import com.twitter.timelines.configapi.Param +import com.twitter.util.Duration + +trait FeatureSwitchConfig { + def booleanFSParams: Seq[Param[Boolean] with FSName] = Nil + + def intFSParams: Seq[FSBoundedParam[Int]] = Nil + + def longFSParams: Seq[FSBoundedParam[Long]] = Nil + + def doubleFSParams: Seq[FSBoundedParam[Double]] = Nil + + def durationFSParams: Seq[FSBoundedParam[Duration] with HasDurationConversion] = Nil + + def optionalDoubleFSParams: Seq[ + (BoundedParam[Option[Double]], DefinedFeatureName, ValueFeatureName) + ] = Nil + + def stringSeqFSParams: Seq[Param[Seq[String]] with FSName] = Nil + + /** + * Apply overrides in list when the given FS Key is enabled. + * This override type does NOT work with experiments. Params here will be evaluated for every + * request IMMEDIATELY, not upon param.apply. If you would like to use an experiment pls use + * the primitive type or ENUM overrides. + */ + def gatedOverridesMap: Map[String, Seq[OptionalOverride[_]]] = Map.empty +} + +object FeatureSwitchConfig { + def merge(configs: Seq[FeatureSwitchConfig]): FeatureSwitchConfig = new FeatureSwitchConfig { + override def booleanFSParams: Seq[Param[Boolean] with FSName] = + configs.flatMap(_.booleanFSParams) + override def intFSParams: Seq[FSBoundedParam[Int]] = + configs.flatMap(_.intFSParams) + override def longFSParams: Seq[FSBoundedParam[Long]] = + configs.flatMap(_.longFSParams) + override def durationFSParams: Seq[FSBoundedParam[Duration] with HasDurationConversion] = + configs.flatMap(_.durationFSParams) + override def gatedOverridesMap: Map[String, Seq[OptionalOverride[_]]] = + configs.flatMap(_.gatedOverridesMap).toMap + override def doubleFSParams: Seq[FSBoundedParam[Double]] = + configs.flatMap(_.doubleFSParams) + override def optionalDoubleFSParams: Seq[ + (BoundedParam[Option[Double]], DefinedFeatureName, ValueFeatureName) + ] = + configs.flatMap(_.optionalDoubleFSParams) + override def stringSeqFSParams: Seq[Param[Seq[String]] with FSName] = + configs.flatMap(_.stringSeqFSParams) + } +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/deciders/BUILD b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/deciders/BUILD new file mode 100644 index 0000000000..e4982dc0f0 --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/deciders/BUILD @@ -0,0 +1,10 @@ +scala_library( + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/com/google/inject:guice", + "configapi/configapi-core", + "configapi/configapi-decider", + ], +) diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/deciders/DeciderKey.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/deciders/DeciderKey.scala new file mode 100644 index 0000000000..f4c069a63b --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/deciders/DeciderKey.scala @@ -0,0 +1,51 @@ +package com.twitter.follow_recommendations.configapi.deciders + +import com.twitter.servo.decider.DeciderKeyEnum + +object DeciderKey extends DeciderKeyEnum { + val EnableDiffyModuleDarkReading = Value("enable_diffy_module_dark_reading") + val EnableRecommendations = Value("enable_recommendations") + val EnableScoreUserCandidates = Value("enable_score_user_candidates") + val EnableProfileSidebarProduct = Value("enable_profile_sidebar_product") + val EnableMagicRecsProduct = Value("enable_magic_recs_product") + val EnableRuxLandingPageProduct = Value("enable_rux_landing_page_product") + val EnableRuxPymkProduct = Value("enable_rux_pymk_product") + val EnableProfileBonusFollowProduct = Value("enable_profile_bonus_follow_product") + val EnableElectionExploreWtfProduct = Value("enable_election_explore_wtf_product") + val EnableClusterFollowProduct = Value("enable_cluster_follow_product") + val EnableHomeTimelineProduct = Value("enable_home_timeline_product") + val EnableHtlBonusFollowProduct = Value("enable_htl_bonus_follow_product") + val EnableExploreTabProduct = Value("enable_explore_tab_product") + val EnableSidebarProduct = Value("enable_sidebar_product") + val EnableNuxPymkProduct = Value("enable_nux_pymk_product") + val EnableNuxInterestsProduct = Value("enable_nux_interests_product") + val EnableNuxTopicBonusFollowProduct = Value("enable_nux_topic_bonus_follow_product") + val EnableCampaignFormProduct = Value("enable_campaign_form_product") + val EnableReactiveFollowProduct = Value("enable_reactive_follow_product") + val EnableIndiaCovid19CuratedAccountsWtfProduct = Value( + "enable_india_covid19_curated_accounts_wtf_product") + val EnableAbUploadProduct = Value("enable_ab_upload_product") + val EnablePeolePlusPlusProduct = Value("enable_people_plus_plus_product") + val EnableTweetNotificationRecsProduct = Value("enable_tweet_notification_recs_product") + val EnableProfileDeviceFollow = Value("enable_profile_device_follow_product") + val EnableRecosBackfillProduct = Value("enable_recos_backfill_product") + val EnablePostNuxFollowTaskProduct = Value("enable_post_nux_follow_task_product") + val EnableCuratedSpaceHostsProduct = Value("enable_curated_space_hosts_product") + val EnableNuxGeoCategoryProduct = Value("enable_nux_geo_category_product") + val EnableNuxInterestsCategoryProduct = Value("enable_nux_interests_category_product") + val EnableNuxPymkCategoryProduct = Value("enable_nux_pymk_category_product") + val EnableHomeTimelineTweetRecsProduct = Value("enable_home_timeline_tweet_recs_product") + val EnableHtlBulkFriendFollowsProduct = Value("enable_htl_bulk_friend_follows_product") + val EnableNuxAutoFollowProduct = Value("enable_nux_auto_follow_product") + val EnableSearchBonusFollowProduct = Value("enable_search_bonus_follow_product") + val EnableFetchUserInRequestBuilder = Value("enable_fetch_user_in_request_builder") + val EnableProductMixerMagicRecsProduct = Value("enable_product_mixer_magic_recs_product") + val EnableHomeTimelineReverseChronProduct = Value("enable_home_timeline_reverse_chron_product") + val EnableProductMixerPipelineMagicRecsDarkRead = Value( + "enable_product_mixer_pipeline_magic_recs_dark_read") + val EnableExperimentalCaching = Value("enable_experimental_caching") + val EnableDistributedCaching = Value("enable_distributed_caching") + val EnableGizmoduckCaching = Value("enable_gizmoduck_caching") + val EnableTrafficDarkReading = Value("enable_traffic_dark_reading") + val EnableGraphFeatureServiceRequests = Value("enable_graph_feature_service_requests") +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/deciders/DeciderParams.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/deciders/DeciderParams.scala new file mode 100644 index 0000000000..07bf3e14d1 --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/deciders/DeciderParams.scala @@ -0,0 +1,8 @@ +package com.twitter.follow_recommendations.configapi.deciders + +import com.twitter.timelines.configapi.Param + +object DeciderParams { + object EnableRecommendations extends Param[Boolean](false) + object EnableScoreUserCandidates extends Param[Boolean](false) +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/params/BUILD b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/params/BUILD new file mode 100644 index 0000000000..1bb357b2c9 --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/params/BUILD @@ -0,0 +1,13 @@ +scala_library( + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/com/google/inject:guice", + "configapi/configapi-core", + "configapi/configapi-decider", + "configapi/configapi-featureswitches:v2", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/deciders", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/models", + ], +) diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/params/GlobalParams.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/params/GlobalParams.scala new file mode 100644 index 0000000000..cf1905c41e --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/params/GlobalParams.scala @@ -0,0 +1,35 @@ +package com.twitter.follow_recommendations.configapi.params + +import com.twitter.follow_recommendations.models.CandidateSourceType +import com.twitter.timelines.configapi.FSEnumParam +import com.twitter.timelines.configapi.FSParam + +/** + * When adding Producer side experiments, make sure to register the FS Key in [[ProducerFeatureFilter]] + * in [[FeatureSwitchesModule]], otherwise, the FS will not work. + */ +object GlobalParams { + + object EnableCandidateParamHydrations + extends FSParam[Boolean]("frs_receiver_enable_candidate_params", false) + + object KeepUserCandidate + extends FSParam[Boolean]("frs_receiver_holdback_keep_user_candidate", true) + + object KeepSocialUserCandidate + extends FSParam[Boolean]("frs_receiver_holdback_keep_social_user_candidate", true) + + case object EnableGFSSocialProofTransform + extends FSParam("social_proof_transform_use_graph_feature_service", true) + + case object EnableWhoToFollowProducts extends FSParam("who_to_follow_product_enabled", true) + + case object CandidateSourcesToFilter + extends FSEnumParam[CandidateSourceType.type]( + "candidate_sources_type_filter_id", + CandidateSourceType.None, + CandidateSourceType) + + object EnableRecommendationFlowLogs + extends FSParam[Boolean]("frs_recommendation_flow_logs_enabled", false) +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/controllers/BUILD b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/controllers/BUILD new file mode 100644 index 0000000000..2a625d8567 --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/controllers/BUILD @@ -0,0 +1,29 @@ +scala_library( + compiler_option_sets = ["fatal_warnings"], + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/com/google/inject:guice", + "3rdparty/jvm/com/google/inject/extensions:guice-assistedinject", + "3rdparty/jvm/javax/inject:javax.inject", + "3rdparty/jvm/net/codingwell:scala-guice", + "3rdparty/jvm/org/slf4j:slf4j-api", + "decider/src/main/scala", + "finagle/finagle-core/src/main", + "finatra/inject/inject-core/src/main/scala", + "finatra/thrift/src/main/scala/com/twitter/finatra/thrift", + "finatra/thrift/src/main/scala/com/twitter/finatra/thrift:controller", + "finatra/thrift/src/main/scala/com/twitter/finatra/thrift/exceptions", + "finatra/thrift/src/main/scala/com/twitter/finatra/thrift/filters", + "finatra/thrift/src/main/scala/com/twitter/finatra/thrift/modules", + "finatra/thrift/src/main/scala/com/twitter/finatra/thrift/response", + "finatra/thrift/src/main/scala/com/twitter/finatra/thrift/routing", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/models", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/services", + "follow-recommendations-service/thrift/src/main/thrift:thrift-scala", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/debug_query", + "scrooge/scrooge-core/src/main/scala", + "util/util-core:scala", + "util/util-slf4j-api/src/main/scala", + ], +) diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/controllers/CandidateUserDebugParamsBuilder.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/controllers/CandidateUserDebugParamsBuilder.scala new file mode 100644 index 0000000000..1695c464d9 --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/controllers/CandidateUserDebugParamsBuilder.scala @@ -0,0 +1,25 @@ +package com.twitter.follow_recommendations.controllers + +import com.twitter.follow_recommendations.common.models._ +import com.twitter.follow_recommendations.configapi.ParamsFactory +import com.twitter.follow_recommendations.models.CandidateUserDebugParams +import com.twitter.follow_recommendations.models.FeatureValue +import com.twitter.follow_recommendations.{thriftscala => t} +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class CandidateUserDebugParamsBuilder @Inject() (paramsFactory: ParamsFactory) { + def fromThrift(req: t.ScoringUserRequest): CandidateUserDebugParams = { + val clientContext = ClientContextConverter.fromThrift(req.clientContext) + val displayLocation = DisplayLocation.fromThrift(req.displayLocation) + + CandidateUserDebugParams(req.candidates.map { candidate => + candidate.userId -> paramsFactory( + clientContext, + displayLocation, + candidate.featureOverrides + .map(_.mapValues(FeatureValue.fromThrift).toMap).getOrElse(Map.empty)) + }.toMap) + } +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/controllers/RecommendationRequestBuilder.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/controllers/RecommendationRequestBuilder.scala new file mode 100644 index 0000000000..bc21fd6a3e --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/controllers/RecommendationRequestBuilder.scala @@ -0,0 +1,41 @@ +package com.twitter.follow_recommendations.controllers + +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.follow_recommendations.common.models.ClientContextConverter +import com.twitter.follow_recommendations.common.models.DisplayLocation +import com.twitter.follow_recommendations.models.DebugParams +import com.twitter.follow_recommendations.models.DisplayContext +import com.twitter.follow_recommendations.models.RecommendationRequest +import com.twitter.follow_recommendations.{thriftscala => t} +import com.twitter.gizmoduck.thriftscala.UserType +import com.twitter.stitch.Stitch +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class RecommendationRequestBuilder @Inject() ( + requestBuilderUserFetcher: RequestBuilderUserFetcher, + statsReceiver: StatsReceiver) { + private val scopedStats = statsReceiver.scope(this.getClass.getSimpleName) + private val isSoftUserCounter = scopedStats.counter("is_soft_user") + + def fromThrift(tRequest: t.RecommendationRequest): Stitch[RecommendationRequest] = { + requestBuilderUserFetcher.fetchUser(tRequest.clientContext.userId).map { userOpt => + val isSoftUser = userOpt.exists(_.userType == UserType.Soft) + if (isSoftUser) isSoftUserCounter.incr() + RecommendationRequest( + clientContext = ClientContextConverter.fromThrift(tRequest.clientContext), + displayLocation = DisplayLocation.fromThrift(tRequest.displayLocation), + displayContext = tRequest.displayContext.map(DisplayContext.fromThrift), + maxResults = tRequest.maxResults, + cursor = tRequest.cursor, + excludedIds = tRequest.excludedIds, + fetchPromotedContent = tRequest.fetchPromotedContent, + debugParams = tRequest.debugParams.map(DebugParams.fromThrift), + userLocationState = tRequest.userLocationState, + isSoftUser = isSoftUser + ) + } + + } +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/controllers/RequestBuilderUserFetcher.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/controllers/RequestBuilderUserFetcher.scala new file mode 100644 index 0000000000..3953b5ef34 --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/controllers/RequestBuilderUserFetcher.scala @@ -0,0 +1,48 @@ +package com.twitter.follow_recommendations.controllers + +import com.twitter.decider.Decider +import com.twitter.decider.SimpleRecipient +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.follow_recommendations.common.base.StatsUtil +import com.twitter.follow_recommendations.configapi.deciders.DeciderKey +import com.twitter.gizmoduck.thriftscala.LookupContext +import com.twitter.gizmoduck.thriftscala.User +import com.twitter.stitch.Stitch +import com.twitter.stitch.gizmoduck.Gizmoduck +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class RequestBuilderUserFetcher @Inject() ( + gizmoduck: Gizmoduck, + statsReceiver: StatsReceiver, + decider: Decider) { + private val scopedStats = statsReceiver.scope(this.getClass.getSimpleName) + + def fetchUser(userIdOpt: Option[Long]): Stitch[Option[User]] = { + userIdOpt match { + case Some(userId) if enableDecider(userId) => + val stitch = gizmoduck + .getUserById( + userId = userId, + context = LookupContext( + forUserId = Some(userId), + includeProtected = true, + includeSoftUsers = true + ) + ).map(user => Some(user)) + StatsUtil + .profileStitch(stitch, scopedStats) + .handle { + case _: Throwable => None + } + case _ => Stitch.None + } + } + + private def enableDecider(userId: Long): Boolean = { + decider.isAvailable( + DeciderKey.EnableFetchUserInRequestBuilder.toString, + Some(SimpleRecipient(userId))) + } +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/controllers/ScoringUserRequestBuilder.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/controllers/ScoringUserRequestBuilder.scala new file mode 100644 index 0000000000..4a45a19f77 --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/controllers/ScoringUserRequestBuilder.scala @@ -0,0 +1,53 @@ +package com.twitter.follow_recommendations.controllers + +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.follow_recommendations.common.models.ClientContextConverter +import com.twitter.follow_recommendations.common.models.DebugOptions +import com.twitter.follow_recommendations.common.models.DisplayLocation +import com.twitter.follow_recommendations.models.DebugParams +import com.twitter.follow_recommendations.models.ScoringUserRequest +import com.twitter.timelines.configapi.Params +import javax.inject.Inject +import javax.inject.Singleton +import com.twitter.follow_recommendations.{thriftscala => t} +import com.twitter.gizmoduck.thriftscala.UserType +import com.twitter.stitch.Stitch + +@Singleton +class ScoringUserRequestBuilder @Inject() ( + requestBuilderUserFetcher: RequestBuilderUserFetcher, + candidateUserDebugParamsBuilder: CandidateUserDebugParamsBuilder, + statsReceiver: StatsReceiver) { + private val scopedStats = statsReceiver.scope(this.getClass.getSimpleName) + private val isSoftUserCounter = scopedStats.counter("is_soft_user") + + def fromThrift(req: t.ScoringUserRequest): Stitch[ScoringUserRequest] = { + requestBuilderUserFetcher.fetchUser(req.clientContext.userId).map { userOpt => + val isSoftUser = userOpt.exists(_.userType == UserType.Soft) + if (isSoftUser) isSoftUserCounter.incr() + + val candidateUsersParamsMap = candidateUserDebugParamsBuilder.fromThrift(req) + val candidates = req.candidates.map { candidate => + CandidateUser + .fromUserRecommendation(candidate).copy(params = + candidateUsersParamsMap.paramsMap.getOrElse(candidate.userId, Params.Invalid)) + } + + ScoringUserRequest( + clientContext = ClientContextConverter.fromThrift(req.clientContext), + displayLocation = DisplayLocation.fromThrift(req.displayLocation), + params = Params.Empty, + debugOptions = req.debugParams.map(DebugOptions.fromDebugParamsThrift), + recentFollowedUserIds = None, + recentFollowedByUserIds = None, + wtfImpressions = None, + similarToUserIds = Nil, + candidates = candidates, + debugParams = req.debugParams.map(DebugParams.fromThrift), + isSoftUser = isSoftUser + ) + } + } + +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/controllers/ThriftController.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/controllers/ThriftController.scala new file mode 100644 index 0000000000..f3014982e6 --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/controllers/ThriftController.scala @@ -0,0 +1,41 @@ +package com.twitter.follow_recommendations.controllers + +import com.twitter.finatra.thrift.Controller +import com.twitter.follow_recommendations.configapi.ParamsFactory +import com.twitter.follow_recommendations.services.ProductPipelineSelector +import com.twitter.follow_recommendations.services.UserScoringService +import com.twitter.follow_recommendations.thriftscala.FollowRecommendationsThriftService +import com.twitter.follow_recommendations.thriftscala.FollowRecommendationsThriftService._ +import com.twitter.stitch.Stitch +import javax.inject.Inject + +class ThriftController @Inject() ( + userScoringService: UserScoringService, + recommendationRequestBuilder: RecommendationRequestBuilder, + scoringUserRequestBuilder: ScoringUserRequestBuilder, + productPipelineSelector: ProductPipelineSelector, + paramsFactory: ParamsFactory) + extends Controller(FollowRecommendationsThriftService) { + + handle(GetRecommendations) { args: GetRecommendations.Args => + val stitch = recommendationRequestBuilder.fromThrift(args.request).flatMap { request => + val params = paramsFactory( + request.clientContext, + request.displayLocation, + request.debugParams.flatMap(_.featureOverrides).getOrElse(Map.empty)) + productPipelineSelector.selectPipeline(request, params).map(_.toThrift) + } + Stitch.run(stitch) + } + + handle(ScoreUserCandidates) { args: ScoreUserCandidates.Args => + val stitch = scoringUserRequestBuilder.fromThrift(args.request).flatMap { request => + val params = paramsFactory( + request.clientContext, + request.displayLocation, + request.debugParams.flatMap(_.featureOverrides).getOrElse(Map.empty)) + userScoringService.get(request.copy(params = params)).map(_.toThrift) + } + Stitch.run(stitch) + } +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/ads/BUILD b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/ads/BUILD new file mode 100644 index 0000000000..aa775c6a70 --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/ads/BUILD @@ -0,0 +1,19 @@ +scala_library( + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/com/google/inject:guice", + "3rdparty/jvm/com/google/inject/extensions:guice-assistedinject", + "3rdparty/jvm/net/codingwell:scala-guice", + "3rdparty/jvm/org/slf4j:slf4j-api", + "finatra/inject/inject-core/src/main/scala", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/promoted_accounts", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/transforms/tracking_token", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/common", + "util/util-slf4j-api/src/main/scala", + ], +) diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/ads/PromotedAccountsFlow.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/ads/PromotedAccountsFlow.scala new file mode 100644 index 0000000000..dd93724847 --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/ads/PromotedAccountsFlow.scala @@ -0,0 +1,112 @@ +package com.twitter.follow_recommendations.flows.ads + +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.follow_recommendations.common.base.EnrichedCandidateSource +import com.twitter.follow_recommendations.common.base.IdentityRanker +import com.twitter.follow_recommendations.common.base.IdentityTransform +import com.twitter.follow_recommendations.common.base.ParamPredicate +import com.twitter.follow_recommendations.common.base.Predicate +import com.twitter.follow_recommendations.common.base.Ranker +import com.twitter.follow_recommendations.common.base.RecommendationFlow +import com.twitter.follow_recommendations.common.base.RecommendationResultsConfig +import com.twitter.follow_recommendations.common.base.Transform +import com.twitter.follow_recommendations.common.base.TruePredicate +import com.twitter.follow_recommendations.common.candidate_sources.promoted_accounts.PromotedAccountsCandidateSource +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.follow_recommendations.common.predicates.ExcludedUserIdPredicate +import com.twitter.follow_recommendations.common.transforms.tracking_token.TrackingTokenTransform +import com.twitter.inject.annotations.Flag +import com.twitter.product_mixer.core.functional_component.candidate_source.CandidateSource +import com.twitter.util.Duration +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class PromotedAccountsFlow @Inject() ( + promotedAccountsCandidateSource: PromotedAccountsCandidateSource, + trackingTokenTransform: TrackingTokenTransform, + baseStatsReceiver: StatsReceiver, + @Flag("fetch_prod_promoted_accounts") fetchProductionPromotedAccounts: Boolean) + extends RecommendationFlow[PromotedAccountsFlowRequest, CandidateUser] { + + protected override def targetEligibility: Predicate[PromotedAccountsFlowRequest] = + new ParamPredicate[PromotedAccountsFlowRequest]( + PromotedAccountsFlowParams.TargetEligibility + ) + + protected override def candidateSources( + target: PromotedAccountsFlowRequest + ): Seq[CandidateSource[PromotedAccountsFlowRequest, CandidateUser]] = { + import EnrichedCandidateSource._ + val candidateSourceStats = statsReceiver.scope("candidate_sources") + val budget: Duration = target.params(PromotedAccountsFlowParams.FetchCandidateSourceBudget) + val candidateSources = Seq( + promotedAccountsCandidateSource + .mapKeys[PromotedAccountsFlowRequest](r => + Seq(r.toAdsRequest(fetchProductionPromotedAccounts))) + .mapValue(PromotedAccountsUtil.toCandidateUser) + ).map { candidateSource => + candidateSource + .failOpenWithin(budget, candidateSourceStats).observe(candidateSourceStats) + } + candidateSources + } + + protected override def preRankerCandidateFilter: Predicate[ + (PromotedAccountsFlowRequest, CandidateUser) + ] = { + val preRankerFilterStats = statsReceiver.scope("pre_ranker") + ExcludedUserIdPredicate.observe(preRankerFilterStats.scope("exclude_user_id_predicate")) + } + + /** + * rank the candidates + */ + protected override def selectRanker( + target: PromotedAccountsFlowRequest + ): Ranker[PromotedAccountsFlowRequest, CandidateUser] = { + new IdentityRanker[PromotedAccountsFlowRequest, CandidateUser] + } + + /** + * transform the candidates after ranking (e.g. dedupping, grouping and etc) + */ + protected override def postRankerTransform: Transform[ + PromotedAccountsFlowRequest, + CandidateUser + ] = { + new IdentityTransform[PromotedAccountsFlowRequest, CandidateUser] + } + + /** + * filter invalid candidates before returning the results. + * + * Some heavy filters e.g. SGS filter could be applied in this step + */ + protected override def validateCandidates: Predicate[ + (PromotedAccountsFlowRequest, CandidateUser) + ] = { + new TruePredicate[(PromotedAccountsFlowRequest, CandidateUser)] + } + + /** + * transform the candidates into results and return + */ + protected override def transformResults: Transform[PromotedAccountsFlowRequest, CandidateUser] = { + trackingTokenTransform + } + + /** + * configuration for recommendation results + */ + protected override def resultsConfig( + target: PromotedAccountsFlowRequest + ): RecommendationResultsConfig = { + RecommendationResultsConfig( + target.params(PromotedAccountsFlowParams.ResultSizeParam), + target.params(PromotedAccountsFlowParams.BatchSizeParam) + ) + } + + override val statsReceiver: StatsReceiver = baseStatsReceiver.scope("promoted_accounts_flow") +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/ads/PromotedAccountsFlowParams.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/ads/PromotedAccountsFlowParams.scala new file mode 100644 index 0000000000..010aea509a --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/ads/PromotedAccountsFlowParams.scala @@ -0,0 +1,19 @@ +package com.twitter.follow_recommendations.flows.ads +import com.twitter.conversions.DurationOps._ +import com.twitter.timelines.configapi.Param +import com.twitter.util.Duration + +abstract class PromotedAccountsFlowParams[A](default: A) extends Param[A](default) { + override val statName: String = "ads/" + this.getClass.getSimpleName +} + +object PromotedAccountsFlowParams { + + // number of total slots returned to the end user, available to put ads + case object TargetEligibility extends PromotedAccountsFlowParams[Boolean](true) + case object ResultSizeParam extends PromotedAccountsFlowParams[Int](Int.MaxValue) + case object BatchSizeParam extends PromotedAccountsFlowParams[Int](Int.MaxValue) + case object FetchCandidateSourceBudget + extends PromotedAccountsFlowParams[Duration](1000.millisecond) + +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/ads/PromotedAccountsFlowRequest.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/ads/PromotedAccountsFlowRequest.scala new file mode 100644 index 0000000000..61afb3049e --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/ads/PromotedAccountsFlowRequest.scala @@ -0,0 +1,33 @@ +package com.twitter.follow_recommendations.flows.ads +import com.twitter.follow_recommendations.common.clients.adserver.AdRequest +import com.twitter.follow_recommendations.common.models.DisplayLocation +import com.twitter.follow_recommendations.common.models.HasDisplayLocation +import com.twitter.follow_recommendations.common.models.HasExcludedUserIds +import com.twitter.product_mixer.core.model.marshalling.request.ClientContext +import com.twitter.product_mixer.core.model.marshalling.request.HasClientContext +import com.twitter.timelines.configapi.HasParams +import com.twitter.timelines.configapi.Params + +case class PromotedAccountsFlowRequest( + override val clientContext: ClientContext, + override val params: Params, + displayLocation: DisplayLocation, + profileId: Option[Long], + // note we also add userId and profileId to excludeUserIds + excludeIds: Seq[Long]) + extends HasParams + with HasClientContext + with HasExcludedUserIds + with HasDisplayLocation { + def toAdsRequest(fetchProductionPromotedAccounts: Boolean): AdRequest = { + AdRequest( + clientContext = clientContext, + displayLocation = displayLocation, + isTest = Some(!fetchProductionPromotedAccounts), + profileUserId = profileId + ) + } + override val excludedUserIds: Seq[Long] = { + excludeIds ++ clientContext.userId.toSeq ++ profileId.toSeq + } +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/ads/PromotedAccountsUtil.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/ads/PromotedAccountsUtil.scala new file mode 100644 index 0000000000..5327911c21 --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/ads/PromotedAccountsUtil.scala @@ -0,0 +1,28 @@ +package com.twitter.follow_recommendations.flows.ads +import com.twitter.follow_recommendations.common.candidate_sources.promoted_accounts.PromotedCandidateUser +import com.twitter.follow_recommendations.common.models.AccountProof +import com.twitter.follow_recommendations.common.models.AdMetadata +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.follow_recommendations.common.models.Reason +import com.twitter.follow_recommendations.common.models.UserCandidateSourceDetails + +object PromotedAccountsUtil { + def toCandidateUser(promotedCandidateUser: PromotedCandidateUser): CandidateUser = { + CandidateUser( + id = promotedCandidateUser.id, + score = None, + adMetadata = + Some(AdMetadata(promotedCandidateUser.position, promotedCandidateUser.adImpression)), + reason = Some( + Reason( + accountProof = Some(AccountProof(followProof = Some(promotedCandidateUser.followProof)))) + ), + userCandidateSourceDetails = Some( + UserCandidateSourceDetails( + promotedCandidateUser.primaryCandidateSource, + Map.empty, + Map.empty, + None)) + ) + } +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/content_recommender_flow/BUILD b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/content_recommender_flow/BUILD new file mode 100644 index 0000000000..886c0fe5a9 --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/content_recommender_flow/BUILD @@ -0,0 +1,32 @@ +scala_library( + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/com/google/inject:guice", + "3rdparty/jvm/com/google/inject/extensions:guice-assistedinject", + "3rdparty/jvm/net/codingwell:scala-guice", + "3rdparty/jvm/org/slf4j:slf4j-api", + "finatra/inject/inject-core/src/main/scala", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/addressbook", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/crowd_search_accounts", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/real_graph", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/top_organic_follows_accounts", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/user_user_graph", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/gizmoduck", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/sgs", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/weighted_candidate_source_ranker", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/transforms/dedup", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/transforms/tracking_token", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/candidates", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/common", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/params", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/common", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/utils", + "util/util-slf4j-api/src/main/scala", + ], +) diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/content_recommender_flow/ContentRecommenderFlow.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/content_recommender_flow/ContentRecommenderFlow.scala new file mode 100644 index 0000000000..30dfa0d424 --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/content_recommender_flow/ContentRecommenderFlow.scala @@ -0,0 +1,202 @@ +package com.twitter.follow_recommendations.flows.content_recommender_flow + +import com.twitter.conversions.DurationOps._ +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.follow_recommendations.common.base.EnrichedCandidateSource +import com.twitter.follow_recommendations.common.base.GatedPredicateBase +import com.twitter.follow_recommendations.common.base.ParamPredicate +import com.twitter.follow_recommendations.common.base.Predicate +import com.twitter.follow_recommendations.common.base.Ranker +import com.twitter.follow_recommendations.common.base.RecommendationFlow +import com.twitter.follow_recommendations.common.base.RecommendationResultsConfig +import com.twitter.follow_recommendations.common.base.Transform +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.follow_recommendations.common.predicates.ExcludedUserIdPredicate +import com.twitter.follow_recommendations.common.predicates.InactivePredicate +import com.twitter.follow_recommendations.common.predicates.gizmoduck.GizmoduckPredicate +import com.twitter.follow_recommendations.common.predicates.sgs.InvalidRelationshipPredicate +import com.twitter.follow_recommendations.common.predicates.sgs.InvalidTargetCandidateRelationshipTypesPredicate +import com.twitter.follow_recommendations.common.predicates.sgs.RecentFollowingPredicate +import com.twitter.follow_recommendations.common.rankers.weighted_candidate_source_ranker.WeightedCandidateSourceRanker +import com.twitter.follow_recommendations.common.transforms.dedup.DedupTransform +import com.twitter.follow_recommendations.common.transforms.tracking_token.TrackingTokenTransform +import com.twitter.follow_recommendations.utils.CandidateSourceHoldbackUtil +import com.twitter.follow_recommendations.utils.RecommendationFlowBaseSideEffectsUtil +import com.twitter.product_mixer.core.functional_component.candidate_source.CandidateSource +import com.twitter.product_mixer.core.quality_factor.BoundsWithDefault +import com.twitter.product_mixer.core.quality_factor.LinearLatencyQualityFactor +import com.twitter.product_mixer.core.quality_factor.LinearLatencyQualityFactorConfig +import com.twitter.product_mixer.core.quality_factor.LinearLatencyQualityFactorObserver +import com.twitter.product_mixer.core.quality_factor.QualityFactorObserver + +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class ContentRecommenderFlow @Inject() ( + contentRecommenderFlowCandidateSourceRegistry: ContentRecommenderFlowCandidateSourceRegistry, + recentFollowingPredicate: RecentFollowingPredicate, + gizmoduckPredicate: GizmoduckPredicate, + inactivePredicate: InactivePredicate, + sgsPredicate: InvalidTargetCandidateRelationshipTypesPredicate, + invalidRelationshipPredicate: InvalidRelationshipPredicate, + trackingTokenTransform: TrackingTokenTransform, + baseStatsReceiver: StatsReceiver) + extends RecommendationFlow[ContentRecommenderRequest, CandidateUser] + with RecommendationFlowBaseSideEffectsUtil[ContentRecommenderRequest, CandidateUser] + with CandidateSourceHoldbackUtil { + + override val statsReceiver: StatsReceiver = baseStatsReceiver.scope("content_recommender_flow") + + override val qualityFactorObserver: Option[QualityFactorObserver] = { + val config = LinearLatencyQualityFactorConfig( + qualityFactorBounds = + BoundsWithDefault(minInclusive = 0.1, maxInclusive = 1.0, default = 1.0), + initialDelay = 60.seconds, + targetLatency = 100.milliseconds, + targetLatencyPercentile = 95.0, + delta = 0.001 + ) + val qualityFactor = LinearLatencyQualityFactor(config) + val observer = LinearLatencyQualityFactorObserver(qualityFactor) + statsReceiver.provideGauge("quality_factor")(qualityFactor.currentValue.toFloat) + Some(observer) + } + + protected override def targetEligibility: Predicate[ContentRecommenderRequest] = + new ParamPredicate[ContentRecommenderRequest]( + ContentRecommenderParams.TargetEligibility + ) + + protected override def candidateSources( + target: ContentRecommenderRequest + ): Seq[CandidateSource[ContentRecommenderRequest, CandidateUser]] = { + import EnrichedCandidateSource._ + val identifiers = ContentRecommenderFlowCandidateSourceWeights.getWeights(target.params).keySet + val selected = contentRecommenderFlowCandidateSourceRegistry.select(identifiers) + val budget = + target.params(ContentRecommenderParams.FetchCandidateSourceBudgetInMillisecond).millisecond + filterCandidateSources(target, selected.map(c => c.failOpenWithin(budget, statsReceiver)).toSeq) + } + + protected override val preRankerCandidateFilter: Predicate[ + (ContentRecommenderRequest, CandidateUser) + ] = { + val preRankerFilterStats = statsReceiver.scope("pre_ranker") + val recentFollowingPredicateStats = preRankerFilterStats.scope("recent_following_predicate") + val invalidRelationshipPredicateStats = + preRankerFilterStats.scope("invalid_relationship_predicate") + + object recentFollowingGatedPredicate + extends GatedPredicateBase[(ContentRecommenderRequest, CandidateUser)]( + recentFollowingPredicate, + recentFollowingPredicateStats + ) { + override def gate(item: (ContentRecommenderRequest, CandidateUser)): Boolean = + item._1.params(ContentRecommenderParams.EnableRecentFollowingPredicate) + } + + object invalidRelationshipGatedPredicate + extends GatedPredicateBase[(ContentRecommenderRequest, CandidateUser)]( + invalidRelationshipPredicate, + invalidRelationshipPredicateStats + ) { + override def gate(item: (ContentRecommenderRequest, CandidateUser)): Boolean = + item._1.params(ContentRecommenderParams.EnableInvalidRelationshipPredicate) + } + + ExcludedUserIdPredicate + .observe(preRankerFilterStats.scope("exclude_user_id_predicate")) + .andThen(recentFollowingGatedPredicate.observe(recentFollowingPredicateStats)) + .andThen(invalidRelationshipGatedPredicate.observe(invalidRelationshipPredicateStats)) + } + + /** + * rank the candidates + */ + protected override def selectRanker( + target: ContentRecommenderRequest + ): Ranker[ContentRecommenderRequest, CandidateUser] = { + val rankersStatsReceiver = statsReceiver.scope("rankers") + WeightedCandidateSourceRanker + .build[ContentRecommenderRequest]( + ContentRecommenderFlowCandidateSourceWeights.getWeights(target.params), + randomSeed = target.getRandomizationSeed + ).observe(rankersStatsReceiver.scope("weighted_candidate_source_ranker")) + } + + /** + * transform the candidates after ranking + */ + protected override def postRankerTransform: Transform[ + ContentRecommenderRequest, + CandidateUser + ] = { + new DedupTransform[ContentRecommenderRequest, CandidateUser] + .observe(statsReceiver.scope("dedupping")) + } + + protected override def validateCandidates: Predicate[ + (ContentRecommenderRequest, CandidateUser) + ] = { + val stats = statsReceiver.scope("validate_candidates") + val gizmoduckPredicateStats = stats.scope("gizmoduck_predicate") + val inactivePredicateStats = stats.scope("inactive_predicate") + val sgsPredicateStats = stats.scope("sgs_predicate") + + val includeGizmoduckPredicate = + new ParamPredicate[ContentRecommenderRequest]( + ContentRecommenderParams.EnableGizmoduckPredicate) + .map[(ContentRecommenderRequest, CandidateUser)] { + case (request: ContentRecommenderRequest, _) => + request + } + + val includeInactivePredicate = + new ParamPredicate[ContentRecommenderRequest]( + ContentRecommenderParams.EnableInactivePredicate) + .map[(ContentRecommenderRequest, CandidateUser)] { + case (request: ContentRecommenderRequest, _) => + request + } + + val includeInvalidTargetCandidateRelationshipTypesPredicate = + new ParamPredicate[ContentRecommenderRequest]( + ContentRecommenderParams.EnableInvalidTargetCandidateRelationshipPredicate) + .map[(ContentRecommenderRequest, CandidateUser)] { + case (request: ContentRecommenderRequest, _) => + request + } + + Predicate + .andConcurrently[(ContentRecommenderRequest, CandidateUser)]( + Seq( + gizmoduckPredicate.observe(gizmoduckPredicateStats).gate(includeGizmoduckPredicate), + inactivePredicate.observe(inactivePredicateStats).gate(includeInactivePredicate), + sgsPredicate + .observe(sgsPredicateStats).gate( + includeInvalidTargetCandidateRelationshipTypesPredicate), + ) + ) + } + + /** + * transform the candidates into results and return + */ + protected override def transformResults: Transform[ContentRecommenderRequest, CandidateUser] = { + trackingTokenTransform + } + + /** + * configuration for recommendation results + */ + protected override def resultsConfig( + target: ContentRecommenderRequest + ): RecommendationResultsConfig = { + RecommendationResultsConfig( + target.maxResults.getOrElse(target.params(ContentRecommenderParams.ResultSizeParam)), + target.params(ContentRecommenderParams.BatchSizeParam) + ) + } + +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/content_recommender_flow/ContentRecommenderFlowCandidateSourceRegistry.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/content_recommender_flow/ContentRecommenderFlowCandidateSourceRegistry.scala new file mode 100644 index 0000000000..4a6c610425 --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/content_recommender_flow/ContentRecommenderFlowCandidateSourceRegistry.scala @@ -0,0 +1,78 @@ +package com.twitter.follow_recommendations.flows.content_recommender_flow + +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.follow_recommendations.common.base.CandidateSourceRegistry +import com.twitter.follow_recommendations.common.candidate_sources.addressbook.ForwardEmailBookSource +import com.twitter.follow_recommendations.common.candidate_sources.addressbook.ForwardPhoneBookSource +import com.twitter.follow_recommendations.common.candidate_sources.addressbook.ReverseEmailBookSource +import com.twitter.follow_recommendations.common.candidate_sources.addressbook.ReversePhoneBookSource +import com.twitter.follow_recommendations.common.candidate_sources.geo.PopCountryBackFillSource +import com.twitter.follow_recommendations.common.candidate_sources.geo.PopCountrySource +import com.twitter.follow_recommendations.common.candidate_sources.geo.PopGeohashSource +import com.twitter.follow_recommendations.common.candidate_sources.crowd_search_accounts.CrowdSearchAccountsSource +import com.twitter.follow_recommendations.common.candidate_sources.ppmi_locale_follow.PPMILocaleFollowSource +import com.twitter.follow_recommendations.common.candidate_sources.top_organic_follows_accounts.TopOrganicFollowsAccountsSource +import com.twitter.follow_recommendations.common.candidate_sources.real_graph.RealGraphOonV2Source +import com.twitter.follow_recommendations.common.candidate_sources.recent_engagement.RepeatedProfileVisitsSource +import com.twitter.follow_recommendations.common.candidate_sources.sims_expansion.RecentEngagementSimilarUsersSource +import com.twitter.follow_recommendations.common.candidate_sources.sims_expansion.RecentFollowingSimilarUsersSource +import com.twitter.follow_recommendations.common.candidate_sources.socialgraph.RecentFollowingRecentFollowingExpansionSource +import com.twitter.follow_recommendations.common.candidate_sources.stp.OfflineStrongTiePredictionSource +import com.twitter.follow_recommendations.common.candidate_sources.triangular_loops.TriangularLoopsSource +import com.twitter.follow_recommendations.common.candidate_sources.user_user_graph.UserUserGraphCandidateSource +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.product_mixer.core.functional_component.candidate_source.CandidateSource + +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class ContentRecommenderFlowCandidateSourceRegistry @Inject() ( + // social based + forwardPhoneBookSource: ForwardPhoneBookSource, + forwardEmailBookSource: ForwardEmailBookSource, + reversePhoneBookSource: ReversePhoneBookSource, + reverseEmailBookSource: ReverseEmailBookSource, + offlineStrongTiePredictionSource: OfflineStrongTiePredictionSource, + triangularLoopsSource: TriangularLoopsSource, + userUserGraphCandidateSource: UserUserGraphCandidateSource, + realGraphOonSource: RealGraphOonV2Source, + recentFollowingRecentFollowingExpansionSource: RecentFollowingRecentFollowingExpansionSource, + // activity based + recentFollowingSimilarUsersSource: RecentFollowingSimilarUsersSource, + recentEngagementSimilarUsersSource: RecentEngagementSimilarUsersSource, + repeatedProfileVisitsSource: RepeatedProfileVisitsSource, + // geo based + popCountrySource: PopCountrySource, + popGeohashSource: PopGeohashSource, + popCountryBackFillSource: PopCountryBackFillSource, + crowdSearchAccountsSource: CrowdSearchAccountsSource, + topOrganicFollowsAccountsSource: TopOrganicFollowsAccountsSource, + ppmiLocaleFollowSource: PPMILocaleFollowSource, + baseStatsReceiver: StatsReceiver) + extends CandidateSourceRegistry[ContentRecommenderRequest, CandidateUser] { + + override val statsReceiver = baseStatsReceiver + .scope("content_recommender_flow", "candidate_sources") + + override val sources: Set[CandidateSource[ContentRecommenderRequest, CandidateUser]] = Seq( + forwardPhoneBookSource, + forwardEmailBookSource, + reversePhoneBookSource, + reverseEmailBookSource, + offlineStrongTiePredictionSource, + triangularLoopsSource, + userUserGraphCandidateSource, + realGraphOonSource, + recentFollowingRecentFollowingExpansionSource, + recentFollowingSimilarUsersSource, + recentEngagementSimilarUsersSource, + repeatedProfileVisitsSource, + popCountrySource, + popGeohashSource, + popCountryBackFillSource, + crowdSearchAccountsSource, + topOrganicFollowsAccountsSource, + ppmiLocaleFollowSource, + ).toSet +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/content_recommender_flow/ContentRecommenderFlowCandidateSourceWeights.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/content_recommender_flow/ContentRecommenderFlowCandidateSourceWeights.scala new file mode 100644 index 0000000000..845a6ec0a5 --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/content_recommender_flow/ContentRecommenderFlowCandidateSourceWeights.scala @@ -0,0 +1,71 @@ +package com.twitter.follow_recommendations.flows.content_recommender_flow + +import com.twitter.follow_recommendations.common.candidate_sources.addressbook.ForwardEmailBookSource +import com.twitter.follow_recommendations.common.candidate_sources.addressbook.ForwardPhoneBookSource +import com.twitter.follow_recommendations.common.candidate_sources.addressbook.ReverseEmailBookSource +import com.twitter.follow_recommendations.common.candidate_sources.addressbook.ReversePhoneBookSource +import com.twitter.follow_recommendations.common.candidate_sources.geo.PopCountryBackFillSource +import com.twitter.follow_recommendations.common.candidate_sources.geo.PopCountrySource +import com.twitter.follow_recommendations.common.candidate_sources.geo.PopGeohashSource +import com.twitter.follow_recommendations.common.candidate_sources.real_graph.RealGraphOonV2Source +import com.twitter.follow_recommendations.common.candidate_sources.recent_engagement.RepeatedProfileVisitsSource +import com.twitter.follow_recommendations.common.candidate_sources.sims_expansion.RecentEngagementSimilarUsersSource +import com.twitter.follow_recommendations.common.candidate_sources.sims_expansion.RecentFollowingSimilarUsersSource +import com.twitter.follow_recommendations.common.candidate_sources.stp.OfflineStrongTiePredictionSource +import com.twitter.follow_recommendations.common.candidate_sources.triangular_loops.TriangularLoopsSource +import com.twitter.follow_recommendations.common.candidate_sources.user_user_graph.UserUserGraphCandidateSource +import com.twitter.product_mixer.core.model.common.identifier.CandidateSourceIdentifier +import com.twitter.follow_recommendations.common.candidate_sources.crowd_search_accounts.CrowdSearchAccountsSource +import com.twitter.follow_recommendations.common.candidate_sources.ppmi_locale_follow.PPMILocaleFollowSource +import com.twitter.follow_recommendations.common.candidate_sources.socialgraph.RecentFollowingRecentFollowingExpansionSource +import com.twitter.follow_recommendations.common.candidate_sources.top_organic_follows_accounts.TopOrganicFollowsAccountsSource +import com.twitter.timelines.configapi.Params + +object ContentRecommenderFlowCandidateSourceWeights { + + def getWeights( + params: Params + ): Map[CandidateSourceIdentifier, Double] = { + Map[CandidateSourceIdentifier, Double]( + // Social based + UserUserGraphCandidateSource.Identifier -> params( + ContentRecommenderFlowCandidateSourceWeightsParams.UserUserGraphSourceWeight), + ForwardPhoneBookSource.Identifier -> params( + ContentRecommenderFlowCandidateSourceWeightsParams.ForwardPhoneBookSourceWeight), + ReversePhoneBookSource.Identifier -> params( + ContentRecommenderFlowCandidateSourceWeightsParams.ReversePhoneBookSourceWeight), + ForwardEmailBookSource.Identifier -> params( + ContentRecommenderFlowCandidateSourceWeightsParams.ForwardEmailBookSourceWeight), + ReverseEmailBookSource.Identifier -> params( + ContentRecommenderFlowCandidateSourceWeightsParams.ReverseEmailBookSourceWeight), + TriangularLoopsSource.Identifier -> params( + ContentRecommenderFlowCandidateSourceWeightsParams.TriangularLoopsSourceWeight), + OfflineStrongTiePredictionSource.Identifier -> params( + ContentRecommenderFlowCandidateSourceWeightsParams.OfflineStrongTiePredictionSourceWeight), + RecentFollowingRecentFollowingExpansionSource.Identifier -> params( + ContentRecommenderFlowCandidateSourceWeightsParams.NewFollowingNewFollowingExpansionSourceWeight), + RecentFollowingSimilarUsersSource.Identifier -> params( + ContentRecommenderFlowCandidateSourceWeightsParams.NewFollowingSimilarUserSourceWeight), + // Activity based + RealGraphOonV2Source.Identifier -> params( + ContentRecommenderFlowCandidateSourceWeightsParams.RealGraphOonSourceWeight), + RecentEngagementSimilarUsersSource.Identifier -> params( + ContentRecommenderFlowCandidateSourceWeightsParams.RecentEngagementSimilarUserSourceWeight), + RepeatedProfileVisitsSource.Identifier -> params( + ContentRecommenderFlowCandidateSourceWeightsParams.RepeatedProfileVisitsSourceWeight), + // Geo based + PopCountrySource.Identifier -> params( + ContentRecommenderFlowCandidateSourceWeightsParams.PopCountrySourceWeight), + PopGeohashSource.Identifier -> params( + ContentRecommenderFlowCandidateSourceWeightsParams.PopGeohashSourceWeight), + PopCountryBackFillSource.Identifier -> params( + ContentRecommenderFlowCandidateSourceWeightsParams.PopCountryBackfillSourceWeight), + PPMILocaleFollowSource.Identifier -> params( + ContentRecommenderFlowCandidateSourceWeightsParams.PPMILocaleFollowSourceWeight), + CrowdSearchAccountsSource.Identifier -> params( + ContentRecommenderFlowCandidateSourceWeightsParams.CrowdSearchAccountSourceWeight), + TopOrganicFollowsAccountsSource.Identifier -> params( + ContentRecommenderFlowCandidateSourceWeightsParams.TopOrganicFollowsAccountsSourceWeight), + ) + } +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/content_recommender_flow/ContentRecommenderFlowCandidateSourceWeightsParams.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/content_recommender_flow/ContentRecommenderFlowCandidateSourceWeightsParams.scala new file mode 100644 index 0000000000..462de260b0 --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/content_recommender_flow/ContentRecommenderFlowCandidateSourceWeightsParams.scala @@ -0,0 +1,117 @@ +package com.twitter.follow_recommendations.flows.content_recommender_flow + +import com.twitter.timelines.configapi.FSBoundedParam + +object ContentRecommenderFlowCandidateSourceWeightsParams { + // Social based + case object ForwardPhoneBookSourceWeight + extends FSBoundedParam[Double]( + ContentRecommenderFlowFeatureSwitchKeys.ForwardPhoneBookSourceWeight, + 1d, + 0d, + 1000d) + case object ForwardEmailBookSourceWeight + extends FSBoundedParam[Double]( + ContentRecommenderFlowFeatureSwitchKeys.ForwardEmailBookSourceWeight, + 1d, + 0d, + 1000d) + case object ReversePhoneBookSourceWeight + extends FSBoundedParam[Double]( + ContentRecommenderFlowFeatureSwitchKeys.ReversePhoneBookSourceWeight, + 1d, + 0d, + 1000d) + case object ReverseEmailBookSourceWeight + extends FSBoundedParam[Double]( + ContentRecommenderFlowFeatureSwitchKeys.ReverseEmailBookSourceWeight, + 1d, + 0d, + 1000d) + case object OfflineStrongTiePredictionSourceWeight + extends FSBoundedParam[Double]( + ContentRecommenderFlowFeatureSwitchKeys.OfflineStrongTiePredictionSourceWeight, + 1d, + 0d, + 1000d) + case object TriangularLoopsSourceWeight + extends FSBoundedParam[Double]( + ContentRecommenderFlowFeatureSwitchKeys.TriangularLoopsSourceWeight, + 1d, + 0d, + 1000d) + case object UserUserGraphSourceWeight + extends FSBoundedParam[Double]( + ContentRecommenderFlowFeatureSwitchKeys.UserUserGraphSourceWeight, + 1d, + 0d, + 1000d) + case object NewFollowingNewFollowingExpansionSourceWeight + extends FSBoundedParam[Double]( + ContentRecommenderFlowFeatureSwitchKeys.NewFollowingNewFollowingExpansionSourceWeight, + 1d, + 0d, + 1000d) + // Activity based + case object NewFollowingSimilarUserSourceWeight + extends FSBoundedParam[Double]( + ContentRecommenderFlowFeatureSwitchKeys.NewFollowingSimilarUserSourceWeight, + 1d, + 0d, + 1000d) + case object RecentEngagementSimilarUserSourceWeight + extends FSBoundedParam[Double]( + ContentRecommenderFlowFeatureSwitchKeys.RecentEngagementSimilarUserSourceWeight, + 1d, + 0d, + 1000d) + case object RepeatedProfileVisitsSourceWeight + extends FSBoundedParam[Double]( + ContentRecommenderFlowFeatureSwitchKeys.RepeatedProfileVisitsSourceWeight, + 1d, + 0d, + 1000d) + case object RealGraphOonSourceWeight + extends FSBoundedParam[Double]( + ContentRecommenderFlowFeatureSwitchKeys.RealGraphOonSourceWeight, + 1d, + 0d, + 1000d) + // Geo based + case object PopCountrySourceWeight + extends FSBoundedParam[Double]( + ContentRecommenderFlowFeatureSwitchKeys.PopCountrySourceWeight, + 1d, + 0d, + 1000d) + case object PopGeohashSourceWeight + extends FSBoundedParam[Double]( + ContentRecommenderFlowFeatureSwitchKeys.PopGeohashSourceWeight, + 1d, + 0d, + 1000d) + case object PopCountryBackfillSourceWeight + extends FSBoundedParam[Double]( + ContentRecommenderFlowFeatureSwitchKeys.PopCountryBackfillSourceWeight, + 1d, + 0d, + 1000d) + case object PPMILocaleFollowSourceWeight + extends FSBoundedParam[Double]( + ContentRecommenderFlowFeatureSwitchKeys.PPMILocaleFollowSourceWeight, + 1d, + 0d, + 1000d) + case object TopOrganicFollowsAccountsSourceWeight + extends FSBoundedParam[Double]( + ContentRecommenderFlowFeatureSwitchKeys.TopOrganicFollowsAccountsSourceWeight, + 1d, + 0d, + 1000d) + case object CrowdSearchAccountSourceWeight + extends FSBoundedParam[Double]( + ContentRecommenderFlowFeatureSwitchKeys.CrowdSearchAccountSourceWeight, + 1d, + 0d, + 1000d) +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/content_recommender_flow/ContentRecommenderFlowFSConfig.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/content_recommender_flow/ContentRecommenderFlowFSConfig.scala new file mode 100644 index 0000000000..a24032c841 --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/content_recommender_flow/ContentRecommenderFlowFSConfig.scala @@ -0,0 +1,60 @@ +package com.twitter.follow_recommendations.flows.content_recommender_flow + +import com.twitter.follow_recommendations.configapi.common.FeatureSwitchConfig +import com.twitter.timelines.configapi.FSBoundedParam +import com.twitter.timelines.configapi.FSName +import com.twitter.timelines.configapi.Param + +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class ContentRecommenderFlowFSConfig @Inject() () extends FeatureSwitchConfig { + override val booleanFSParams: Seq[Param[Boolean] with FSName] = + Seq( + ContentRecommenderParams.IncludeActivityBasedCandidateSource, + ContentRecommenderParams.IncludeSocialBasedCandidateSource, + ContentRecommenderParams.IncludeGeoBasedCandidateSource, + ContentRecommenderParams.IncludeHomeTimelineTweetRecsCandidateSource, + ContentRecommenderParams.IncludeSocialProofEnforcedCandidateSource, + ContentRecommenderParams.EnableRecentFollowingPredicate, + ContentRecommenderParams.EnableGizmoduckPredicate, + ContentRecommenderParams.EnableInactivePredicate, + ContentRecommenderParams.EnableInvalidTargetCandidateRelationshipPredicate, + ContentRecommenderParams.IncludeNewFollowingNewFollowingExpansionCandidateSource, + ContentRecommenderParams.IncludeMoreGeoBasedCandidateSource, + ContentRecommenderParams.TargetEligibility, + ContentRecommenderParams.GetFollowersFromSgs, + ContentRecommenderParams.EnableInvalidRelationshipPredicate, + ) + + override val intFSParams: Seq[FSBoundedParam[Int]] = + Seq( + ContentRecommenderParams.ResultSizeParam, + ContentRecommenderParams.BatchSizeParam, + ContentRecommenderParams.FetchCandidateSourceBudgetInMillisecond, + ContentRecommenderParams.RecentFollowingPredicateBudgetInMillisecond, + ) + + override val doubleFSParams: Seq[FSBoundedParam[Double]] = + Seq( + ContentRecommenderFlowCandidateSourceWeightsParams.ForwardPhoneBookSourceWeight, + ContentRecommenderFlowCandidateSourceWeightsParams.ForwardEmailBookSourceWeight, + ContentRecommenderFlowCandidateSourceWeightsParams.ReversePhoneBookSourceWeight, + ContentRecommenderFlowCandidateSourceWeightsParams.ReverseEmailBookSourceWeight, + ContentRecommenderFlowCandidateSourceWeightsParams.OfflineStrongTiePredictionSourceWeight, + ContentRecommenderFlowCandidateSourceWeightsParams.TriangularLoopsSourceWeight, + ContentRecommenderFlowCandidateSourceWeightsParams.UserUserGraphSourceWeight, + ContentRecommenderFlowCandidateSourceWeightsParams.NewFollowingNewFollowingExpansionSourceWeight, + ContentRecommenderFlowCandidateSourceWeightsParams.NewFollowingSimilarUserSourceWeight, + ContentRecommenderFlowCandidateSourceWeightsParams.RecentEngagementSimilarUserSourceWeight, + ContentRecommenderFlowCandidateSourceWeightsParams.RepeatedProfileVisitsSourceWeight, + ContentRecommenderFlowCandidateSourceWeightsParams.RealGraphOonSourceWeight, + ContentRecommenderFlowCandidateSourceWeightsParams.PopCountrySourceWeight, + ContentRecommenderFlowCandidateSourceWeightsParams.PopGeohashSourceWeight, + ContentRecommenderFlowCandidateSourceWeightsParams.PopCountryBackfillSourceWeight, + ContentRecommenderFlowCandidateSourceWeightsParams.PPMILocaleFollowSourceWeight, + ContentRecommenderFlowCandidateSourceWeightsParams.TopOrganicFollowsAccountsSourceWeight, + ContentRecommenderFlowCandidateSourceWeightsParams.CrowdSearchAccountSourceWeight, + ) +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/content_recommender_flow/ContentRecommenderFlowFeatureSwitchKeys.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/content_recommender_flow/ContentRecommenderFlowFeatureSwitchKeys.scala new file mode 100644 index 0000000000..ff51dc9f64 --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/content_recommender_flow/ContentRecommenderFlowFeatureSwitchKeys.scala @@ -0,0 +1,70 @@ +package com.twitter.follow_recommendations.flows.content_recommender_flow + +object ContentRecommenderFlowFeatureSwitchKeys { + val TargetUserEligible = "content_recommender_flow_target_eligible" + val ResultSize = "content_recommender_flow_result_size" + val BatchSize = "content_recommender_flow_batch_size" + val RecentFollowingPredicateBudgetInMillisecond = + "content_recommender_flow_recent_following_predicate_budget_in_ms" + val CandidateGenerationBudgetInMillisecond = + "content_recommender_flow_candidate_generation_budget_in_ms" + val EnableRecentFollowingPredicate = "content_recommender_flow_enable_recent_following_predicate" + val EnableGizmoduckPredicate = "content_recommender_flow_enable_gizmoduck_predicate" + val EnableInactivePredicate = "content_recommender_flow_enable_inactive_predicate" + val EnableInvalidTargetCandidateRelationshipPredicate = + "content_recommender_flow_enable_invalid_target_candidate_relationship_predicate" + val IncludeActivityBasedCandidateSource = + "content_recommender_flow_include_activity_based_candidate_source" + val IncludeSocialBasedCandidateSource = + "content_recommender_flow_include_social_based_candidate_source" + val IncludeGeoBasedCandidateSource = + "content_recommender_flow_include_geo_based_candidate_source" + val IncludeHomeTimelineTweetRecsCandidateSource = + "content_recommender_flow_include_home_timeline_tweet_recs_candidate_source" + val IncludeSocialProofEnforcedCandidateSource = + "content_recommender_flow_include_social_proof_enforced_candidate_source" + val IncludeNewFollowingNewFollowingExpansionCandidateSource = + "content_recommender_flow_include_new_following_new_following_expansion_candidate_source" + val IncludeMoreGeoBasedCandidateSource = + "content_recommender_flow_include_more_geo_based_candidate_source" + val GetFollowersFromSgs = "content_recommender_flow_get_followers_from_sgs" + val EnableInvalidRelationshipPredicate = + "content_recommender_flow_enable_invalid_relationship_predicate" + + // Candidate source weight param keys + // Social based + val ForwardPhoneBookSourceWeight = + "content_recommender_flow_candidate_source_weight_forward_phone_book" + val ForwardEmailBookSourceWeight = + "content_recommender_flow_candidate_source_weight_forward_email_book" + val ReversePhoneBookSourceWeight = + "content_recommender_flow_candidate_source_weight_reverse_phone_book" + val ReverseEmailBookSourceWeight = + "content_recommender_flow_candidate_source_weight_reverse_email_book" + val OfflineStrongTiePredictionSourceWeight = + "content_recommender_flow_candidate_source_weight_offline_stp" + val TriangularLoopsSourceWeight = + "content_recommender_flow_candidate_source_weight_triangular_loops" + val UserUserGraphSourceWeight = "content_recommender_flow_candidate_source_weight_user_user_graph" + val NewFollowingNewFollowingExpansionSourceWeight = + "content_recommender_flow_candidate_source_weight_new_following_new_following_expansion" + // Activity based + val NewFollowingSimilarUserSourceWeight = + "content_recommender_flow_candidate_source_weight_new_following_similar_user" + val RecentEngagementSimilarUserSourceWeight = + "content_recommender_flow_candidate_source_weight_recent_engagement_similar_user" + val RepeatedProfileVisitsSourceWeight = + "content_recommender_flow_candidate_source_weight_repeated_profile_visits" + val RealGraphOonSourceWeight = "content_recommender_flow_candidate_source_weight_real_graph_oon" + // Geo based + val PopCountrySourceWeight = "content_recommender_flow_candidate_source_weight_pop_country" + val PopGeohashSourceWeight = "content_recommender_flow_candidate_source_weight_pop_geohash" + val PopCountryBackfillSourceWeight = + "content_recommender_flow_candidate_source_weight_pop_country_backfill" + val PPMILocaleFollowSourceWeight = + "content_recommender_flow_candidate_source_weight_ppmi_locale_follow" + val TopOrganicFollowsAccountsSourceWeight = + "content_recommender_flow_candidate_source_weight_top_organic_follow_account" + val CrowdSearchAccountSourceWeight = + "content_recommender_flow_candidate_source_weight_crowd_search_account" +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/content_recommender_flow/ContentRecommenderParams.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/content_recommender_flow/ContentRecommenderParams.scala new file mode 100644 index 0000000000..6b43325af5 --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/content_recommender_flow/ContentRecommenderParams.scala @@ -0,0 +1,85 @@ +package com.twitter.follow_recommendations.flows.content_recommender_flow + +import com.twitter.timelines.configapi.FSBoundedParam +import com.twitter.timelines.configapi.FSParam +import com.twitter.timelines.configapi.Param + +abstract class ContentRecommenderParams[A](default: A) extends Param[A](default) { + override val statName: String = "content_recommender/" + this.getClass.getSimpleName +} + +object ContentRecommenderParams { + + case object TargetEligibility + extends FSParam[Boolean](ContentRecommenderFlowFeatureSwitchKeys.TargetUserEligible, true) + + case object ResultSizeParam + extends FSBoundedParam[Int](ContentRecommenderFlowFeatureSwitchKeys.ResultSize, 15, 1, 500) + case object BatchSizeParam + extends FSBoundedParam[Int](ContentRecommenderFlowFeatureSwitchKeys.BatchSize, 15, 1, 500) + case object RecentFollowingPredicateBudgetInMillisecond + extends FSBoundedParam[Int]( + ContentRecommenderFlowFeatureSwitchKeys.RecentFollowingPredicateBudgetInMillisecond, + 8, + 1, + 50) + case object FetchCandidateSourceBudgetInMillisecond + extends FSBoundedParam[Int]( + ContentRecommenderFlowFeatureSwitchKeys.CandidateGenerationBudgetInMillisecond, + 60, + 1, + 80) + case object EnableRecentFollowingPredicate + extends FSParam[Boolean]( + ContentRecommenderFlowFeatureSwitchKeys.EnableRecentFollowingPredicate, + true) + case object EnableGizmoduckPredicate + extends FSParam[Boolean]( + ContentRecommenderFlowFeatureSwitchKeys.EnableGizmoduckPredicate, + false) + case object EnableInactivePredicate + extends FSParam[Boolean]( + ContentRecommenderFlowFeatureSwitchKeys.EnableInactivePredicate, + false) + case object EnableInvalidTargetCandidateRelationshipPredicate + extends FSParam[Boolean]( + ContentRecommenderFlowFeatureSwitchKeys.EnableInvalidTargetCandidateRelationshipPredicate, + false) + case object IncludeActivityBasedCandidateSource + extends FSParam[Boolean]( + ContentRecommenderFlowFeatureSwitchKeys.IncludeActivityBasedCandidateSource, + true) + case object IncludeSocialBasedCandidateSource + extends FSParam[Boolean]( + ContentRecommenderFlowFeatureSwitchKeys.IncludeSocialBasedCandidateSource, + true) + case object IncludeGeoBasedCandidateSource + extends FSParam[Boolean]( + ContentRecommenderFlowFeatureSwitchKeys.IncludeGeoBasedCandidateSource, + true) + case object IncludeHomeTimelineTweetRecsCandidateSource + extends FSParam[Boolean]( + ContentRecommenderFlowFeatureSwitchKeys.IncludeHomeTimelineTweetRecsCandidateSource, + false) + case object IncludeSocialProofEnforcedCandidateSource + extends FSParam[Boolean]( + ContentRecommenderFlowFeatureSwitchKeys.IncludeSocialProofEnforcedCandidateSource, + false) + case object IncludeNewFollowingNewFollowingExpansionCandidateSource + extends FSParam[Boolean]( + ContentRecommenderFlowFeatureSwitchKeys.IncludeNewFollowingNewFollowingExpansionCandidateSource, + false) + + case object IncludeMoreGeoBasedCandidateSource + extends FSParam[Boolean]( + ContentRecommenderFlowFeatureSwitchKeys.IncludeMoreGeoBasedCandidateSource, + false) + + case object GetFollowersFromSgs + extends FSParam[Boolean](ContentRecommenderFlowFeatureSwitchKeys.GetFollowersFromSgs, false) + + case object EnableInvalidRelationshipPredicate + extends FSParam[Boolean]( + ContentRecommenderFlowFeatureSwitchKeys.EnableInvalidRelationshipPredicate, + false) +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/content_recommender_flow/ContentRecommenderRequest.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/content_recommender_flow/ContentRecommenderRequest.scala new file mode 100644 index 0000000000..5952314e52 --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/content_recommender_flow/ContentRecommenderRequest.scala @@ -0,0 +1,45 @@ +package com.twitter.follow_recommendations.flows.content_recommender_flow + +import com.twitter.core_workflows.user_model.thriftscala.UserState +import com.twitter.follow_recommendations.common.models.DebugOptions +import com.twitter.follow_recommendations.common.models.DisplayLocation +import com.twitter.follow_recommendations.common.models.GeohashAndCountryCode +import com.twitter.follow_recommendations.common.models.HasDebugOptions +import com.twitter.follow_recommendations.common.models.HasDisplayLocation +import com.twitter.follow_recommendations.common.models.HasExcludedUserIds +import com.twitter.follow_recommendations.common.models.HasGeohashAndCountryCode +import com.twitter.follow_recommendations.common.models.HasInvalidRelationshipUserIds +import com.twitter.follow_recommendations.common.models.HasRecentFollowedByUserIds +import com.twitter.follow_recommendations.common.models.HasRecentFollowedUserIds +import com.twitter.follow_recommendations.common.models.HasUserState +import com.twitter.product_mixer.core.model.marshalling.request.ClientContext +import com.twitter.product_mixer.core.model.marshalling.request.HasClientContext +import com.twitter.timelines.configapi.HasParams +import com.twitter.timelines.configapi.Params + +case class ContentRecommenderRequest( + override val params: Params, + override val clientContext: ClientContext, + inputExcludeUserIds: Seq[Long], + override val recentFollowedUserIds: Option[Seq[Long]], + override val recentFollowedByUserIds: Option[Seq[Long]], + override val invalidRelationshipUserIds: Option[Set[Long]], + override val displayLocation: DisplayLocation, + maxResults: Option[Int] = None, + override val debugOptions: Option[DebugOptions] = None, + override val geohashAndCountryCode: Option[GeohashAndCountryCode] = None, + override val userState: Option[UserState] = None) + extends HasParams + with HasClientContext + with HasDisplayLocation + with HasDebugOptions + with HasRecentFollowedUserIds + with HasRecentFollowedByUserIds + with HasInvalidRelationshipUserIds + with HasExcludedUserIds + with HasUserState + with HasGeohashAndCountryCode { + override val excludedUserIds: Seq[Long] = { + inputExcludeUserIds ++ clientContext.userId.toSeq + } +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/content_recommender_flow/ContentRecommenderRequestBuilder.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/content_recommender_flow/ContentRecommenderRequestBuilder.scala new file mode 100644 index 0000000000..769f9ce517 --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/content_recommender_flow/ContentRecommenderRequestBuilder.scala @@ -0,0 +1,121 @@ +package com.twitter.follow_recommendations.flows.content_recommender_flow + +import com.twitter.conversions.DurationOps._ +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.follow_recommendations.common.clients.geoduck.UserLocationFetcher +import com.twitter.follow_recommendations.common.clients.socialgraph.SocialGraphClient +import com.twitter.follow_recommendations.common.clients.user_state.UserStateClient +import com.twitter.follow_recommendations.common.utils.RescueWithStatsUtils.rescueOptionalWithStats +import com.twitter.follow_recommendations.common.utils.RescueWithStatsUtils.rescueWithStats +import com.twitter.follow_recommendations.common.utils.RescueWithStatsUtils.rescueWithStatsWithin +import com.twitter.follow_recommendations.products.common.ProductRequest +import com.twitter.stitch.Stitch + +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class ContentRecommenderRequestBuilder @Inject() ( + socialGraph: SocialGraphClient, + userLocationFetcher: UserLocationFetcher, + userStateClient: UserStateClient, + statsReceiver: StatsReceiver) { + + val stats: StatsReceiver = statsReceiver.scope("content_recommender_request_builder") + val invalidRelationshipUsersStats: StatsReceiver = stats.scope("invalidRelationshipUserIds") + private val invalidRelationshipUsersMaxSizeCounter = + invalidRelationshipUsersStats.counter("maxSize") + private val invalidRelationshipUsersNotMaxSizeCounter = + invalidRelationshipUsersStats.counter("notMaxSize") + + def build(req: ProductRequest): Stitch[ContentRecommenderRequest] = { + val userStateStitch = Stitch + .collect(req.recommendationRequest.clientContext.userId.map(userId => + userStateClient.getUserState(userId))).map(_.flatten) + val recentFollowedUserIdsStitch = + Stitch + .collect(req.recommendationRequest.clientContext.userId.map { userId => + rescueWithStatsWithin( + socialGraph.getRecentFollowedUserIds(userId), + stats, + "recentFollowedUserIds", + req + .params( + ContentRecommenderParams.RecentFollowingPredicateBudgetInMillisecond).millisecond + ) + }) + val recentFollowedByUserIdsStitch = + if (req.params(ContentRecommenderParams.GetFollowersFromSgs)) { + Stitch + .collect( + req.recommendationRequest.clientContext.userId.map(userId => + rescueWithStatsWithin( + socialGraph.getRecentFollowedByUserIdsFromCachedColumn(userId), + stats, + "recentFollowedByUserIds", + req + .params(ContentRecommenderParams.RecentFollowingPredicateBudgetInMillisecond) + .millisecond + ))) + } else Stitch.None + val invalidRelationshipUserIdsStitch: Stitch[Option[Seq[Long]]] = + if (req.params(ContentRecommenderParams.EnableInvalidRelationshipPredicate)) { + Stitch + .collect( + req.recommendationRequest.clientContext.userId.map { userId => + rescueWithStats( + socialGraph + .getInvalidRelationshipUserIdsFromCachedColumn(userId) + .onSuccess(ids => + if (ids.size >= SocialGraphClient.MaxNumInvalidRelationship) { + invalidRelationshipUsersMaxSizeCounter.incr() + } else { + invalidRelationshipUsersNotMaxSizeCounter.incr() + }), + stats, + "invalidRelationshipUserIds" + ) + } + ) + } else { + Stitch.None + } + val locationStitch = + rescueOptionalWithStats( + userLocationFetcher.getGeohashAndCountryCode( + req.recommendationRequest.clientContext.userId, + req.recommendationRequest.clientContext.ipAddress + ), + stats, + "userLocation" + ) + Stitch + .join( + recentFollowedUserIdsStitch, + recentFollowedByUserIdsStitch, + invalidRelationshipUserIdsStitch, + locationStitch, + userStateStitch) + .map { + case ( + recentFollowedUserIds, + recentFollowedByUserIds, + invalidRelationshipUserIds, + location, + userState) => + ContentRecommenderRequest( + req.params, + req.recommendationRequest.clientContext, + req.recommendationRequest.excludedIds.getOrElse(Nil), + recentFollowedUserIds, + recentFollowedByUserIds, + invalidRelationshipUserIds.map(_.toSet), + req.recommendationRequest.displayLocation, + req.recommendationRequest.maxResults, + req.recommendationRequest.debugParams.flatMap(_.debugOptions), + location, + userState + ) + } + } +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/post_nux_ml/BUILD b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/post_nux_ml/BUILD new file mode 100644 index 0000000000..9129e17b88 --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/post_nux_ml/BUILD @@ -0,0 +1,58 @@ +scala_library( + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/com/google/inject:guice", + "3rdparty/jvm/com/google/inject/extensions:guice-assistedinject", + "3rdparty/jvm/net/codingwell:scala-guice", + "3rdparty/jvm/org/slf4j:slf4j-api", + "finatra/inject/inject-core/src/main/scala", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/addressbook", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/crowd_search_accounts", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/geo", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/real_graph", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/recent_engagement", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/salsa", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims_expansion", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/top_organic_follows_accounts", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/triangular_loops", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/two_hop_random_walk", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/user_user_graph", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/deepbirdv2", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/geoduck", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/impression_store", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/interests_service", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/user_state", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/common", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/sources", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/dismiss", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/gizmoduck", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/health", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/sgs", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/user_activity", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/fatigue_ranker", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/first_n_ranker", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/interleave_ranker", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/ml_ranker/ranking", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/ml_ranker/scoring", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/weighted_candidate_source_ranker", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/transforms/dedup", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/transforms/modify_social_proof", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/transforms/ranker_id", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/transforms/tracking_token", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/transforms/weighted_sampling", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/candidates", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/common", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/params", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/logging", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/common", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/utils", + "util/util-slf4j-api/src/main/scala", + ], +) diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/post_nux_ml/PostNuxMlCandidateSourceRegistry.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/post_nux_ml/PostNuxMlCandidateSourceRegistry.scala new file mode 100644 index 0000000000..ed15d566c3 --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/post_nux_ml/PostNuxMlCandidateSourceRegistry.scala @@ -0,0 +1,103 @@ +package com.twitter.follow_recommendations.flows.post_nux_ml + +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.follow_recommendations.common.base.CandidateSourceRegistry +import com.twitter.follow_recommendations.common.base.EnrichedCandidateSource +import com.twitter.follow_recommendations.common.candidate_sources.addressbook.ForwardEmailBookSource +import com.twitter.follow_recommendations.common.candidate_sources.addressbook.ForwardPhoneBookSource +import com.twitter.follow_recommendations.common.candidate_sources.addressbook.ReverseEmailBookSource +import com.twitter.follow_recommendations.common.candidate_sources.addressbook.ReversePhoneBookSource +import com.twitter.follow_recommendations.common.candidate_sources.crowd_search_accounts.CrowdSearchAccountsSource +import com.twitter.follow_recommendations.common.candidate_sources.top_organic_follows_accounts.TopOrganicFollowsAccountsSource +import com.twitter.follow_recommendations.common.candidate_sources.geo.PopCountrySource +import com.twitter.follow_recommendations.common.candidate_sources.geo.PopCountryBackFillSource +import com.twitter.follow_recommendations.common.candidate_sources.geo.PopGeohashQualityFollowSource +import com.twitter.follow_recommendations.common.candidate_sources.geo.PopGeohashSource +import com.twitter.follow_recommendations.common.candidate_sources.ppmi_locale_follow.PPMILocaleFollowSource +import com.twitter.follow_recommendations.common.candidate_sources.real_graph.RealGraphOonV2Source +import com.twitter.follow_recommendations.common.candidate_sources.recent_engagement.RecentEngagementNonDirectFollowSource +import com.twitter.follow_recommendations.common.candidate_sources.recent_engagement.RepeatedProfileVisitsSource +import com.twitter.follow_recommendations.common.candidate_sources.salsa.RecentEngagementDirectFollowSalsaExpansionSource +import com.twitter.follow_recommendations.common.candidate_sources.sims.LinearRegressionFollow2vecNearestNeighborsStore +import com.twitter.follow_recommendations.common.candidate_sources.sims_expansion.RecentEngagementSimilarUsersSource +import com.twitter.follow_recommendations.common.candidate_sources.sims_expansion.RecentFollowingSimilarUsersSource +import com.twitter.follow_recommendations.common.candidate_sources.stp.OnlineSTPSourceScorer +import com.twitter.follow_recommendations.common.candidate_sources.stp.OfflineStrongTiePredictionSource +import com.twitter.follow_recommendations.common.candidate_sources.triangular_loops.TriangularLoopsSource +import com.twitter.follow_recommendations.common.candidate_sources.user_user_graph.UserUserGraphCandidateSource +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.product_mixer.core.functional_component.candidate_source.CandidateSource +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class PostNuxMlCandidateSourceRegistry @Inject() ( + crowdSearchAccountsCandidateSource: CrowdSearchAccountsSource, + topOrganicFollowsAccountsSource: TopOrganicFollowsAccountsSource, + linearRegressionfollow2vecNearestNeighborsStore: LinearRegressionFollow2vecNearestNeighborsStore, + forwardEmailBookSource: ForwardEmailBookSource, + forwardPhoneBookSource: ForwardPhoneBookSource, + offlineStrongTiePredictionSource: OfflineStrongTiePredictionSource, + onlineSTPSource: OnlineSTPSourceScorer, + popCountrySource: PopCountrySource, + popCountryBackFillSource: PopCountryBackFillSource, + popGeohashSource: PopGeohashSource, + recentEngagementDirectFollowSimilarUsersSource: RecentEngagementSimilarUsersSource, + recentEngagementNonDirectFollowSource: RecentEngagementNonDirectFollowSource, + recentEngagementDirectFollowSalsaExpansionSource: RecentEngagementDirectFollowSalsaExpansionSource, + recentFollowingSimilarUsersSource: RecentFollowingSimilarUsersSource, + realGraphOonV2Source: RealGraphOonV2Source, + repeatedProfileVisitSource: RepeatedProfileVisitsSource, + reverseEmailBookSource: ReverseEmailBookSource, + reversePhoneBookSource: ReversePhoneBookSource, + triangularLoopsSource: TriangularLoopsSource, + userUserGraphCandidateSource: UserUserGraphCandidateSource, + ppmiLocaleFollowSource: PPMILocaleFollowSource, + popGeohashQualityFollowSource: PopGeohashQualityFollowSource, + baseStatsReceiver: StatsReceiver, +) extends CandidateSourceRegistry[PostNuxMlRequest, CandidateUser] { + import EnrichedCandidateSource._ + + override val statsReceiver = baseStatsReceiver + .scope("post_nux_ml_flow", "candidate_sources") + + // sources primarily based on social graph signals + private[this] val socialSources = Seq( + linearRegressionfollow2vecNearestNeighborsStore.mapKeys[PostNuxMlRequest]( + _.getOptionalUserId.toSeq), + forwardEmailBookSource, + forwardPhoneBookSource, + offlineStrongTiePredictionSource, + onlineSTPSource, + reverseEmailBookSource, + reversePhoneBookSource, + triangularLoopsSource, + ) + + // sources primarily based on geo signals + private[this] val geoSources = Seq( + popCountrySource, + popCountryBackFillSource, + popGeohashSource, + popGeohashQualityFollowSource, + topOrganicFollowsAccountsSource, + crowdSearchAccountsCandidateSource, + ppmiLocaleFollowSource, + ) + + // sources primarily based on recent activity signals + private[this] val activitySources = Seq( + repeatedProfileVisitSource, + recentEngagementDirectFollowSalsaExpansionSource.mapKeys[PostNuxMlRequest]( + _.getOptionalUserId.toSeq), + recentEngagementDirectFollowSimilarUsersSource, + recentEngagementNonDirectFollowSource.mapKeys[PostNuxMlRequest](_.getOptionalUserId.toSeq), + recentFollowingSimilarUsersSource, + realGraphOonV2Source, + userUserGraphCandidateSource, + ) + + override val sources: Set[CandidateSource[PostNuxMlRequest, CandidateUser]] = ( + geoSources ++ socialSources ++ activitySources + ).toSet +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/post_nux_ml/PostNuxMlCandidateSourceWeightParams.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/post_nux_ml/PostNuxMlCandidateSourceWeightParams.scala new file mode 100644 index 0000000000..9492747a40 --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/post_nux_ml/PostNuxMlCandidateSourceWeightParams.scala @@ -0,0 +1,177 @@ +package com.twitter.follow_recommendations.flows.post_nux_ml + +import com.twitter.timelines.configapi.FSBoundedParam +import com.twitter.timelines.configapi.Param + +abstract class PostNuxMlCandidateSourceWeightParams[A](default: A) extends Param[A](default) { + override val statName: String = "post_nux_ml/" + this.getClass.getSimpleName +} + +object PostNuxMlCandidateSourceWeightParams { + + case object CandidateWeightCrowdSearch + extends FSBoundedParam[Double]( + PostNuxMlFlowCandidateSourceWeightsFeatureSwitchKeys.CandidateWeightCrowdSearch, + 1.0, + 0.0, + 1000.0 + ) + + case object CandidateWeightTopOrganicFollow + extends FSBoundedParam[Double]( + PostNuxMlFlowCandidateSourceWeightsFeatureSwitchKeys.CandidateWeightTopOrganicFollow, + 1.0, + 0.0, + 1000.0 + ) + case object CandidateWeightPPMILocaleFollow + extends FSBoundedParam[Double]( + PostNuxMlFlowCandidateSourceWeightsFeatureSwitchKeys.CandidateWeightPPMILocaleFollow, + 1.0, + 0.0, + 1000.0 + ) + + case object CandidateWeightForwardEmailBook + extends FSBoundedParam[Double]( + PostNuxMlFlowCandidateSourceWeightsFeatureSwitchKeys.CandidateWeightForwardEmailBook, + 1.0, + 0.0, + 1000.0 + ) + case object CandidateWeightForwardPhoneBook + extends FSBoundedParam[Double]( + PostNuxMlFlowCandidateSourceWeightsFeatureSwitchKeys.CandidateWeightForwardPhoneBook, + 1.0, + 0.0, + 1000.0 + ) + + case object CandidateWeightOfflineStrongTiePrediction + extends FSBoundedParam[Double]( + PostNuxMlFlowCandidateSourceWeightsFeatureSwitchKeys.CandidateWeightOfflineStrongTiePrediction, + 1.0, + 0.0, + 1000.0 + ) + case object CandidateWeightOnlineStp + extends FSBoundedParam[Double]( + PostNuxMlFlowCandidateSourceWeightsFeatureSwitchKeys.CandidateWeightOnlineStp, + 1.0, + 0.0, + 1000.0 + ) + case object CandidateWeightPopCountry + extends FSBoundedParam[Double]( + PostNuxMlFlowCandidateSourceWeightsFeatureSwitchKeys.CandidateWeightPopCountry, + 1.0, + 0.0, + 1000.0 + ) + case object CandidateWeightPopGeohash + extends FSBoundedParam[Double]( + PostNuxMlFlowCandidateSourceWeightsFeatureSwitchKeys.CandidateWeightPopGeohash, + 1.0, + 0.0, + 1000.0 + ) + case object CandidateWeightPopGeohashQualityFollow + extends FSBoundedParam[Double]( + PostNuxMlFlowCandidateSourceWeightsFeatureSwitchKeys.CandidateWeightPopGeohashQualityFollow, + 1.0, + 0.0, + 1000.0 + ) + case object CandidateWeightPopGeoBackfill + extends FSBoundedParam[Double]( + PostNuxMlFlowCandidateSourceWeightsFeatureSwitchKeys.CandidateWeightPopGeoBackfill, + 1, + 0.0, + 1000.0 + ) + case object CandidateWeightRecentFollowingSimilarUsers + extends FSBoundedParam[Double]( + PostNuxMlFlowCandidateSourceWeightsFeatureSwitchKeys.CandidateWeightRecentFollowingSimilarUsers, + 1.0, + 0.0, + 1000.0 + ) + case object CandidateWeightRecentEngagementDirectFollowSalsaExpansion + extends FSBoundedParam[Double]( + PostNuxMlFlowCandidateSourceWeightsFeatureSwitchKeys.CandidateWeightRecentEngagementDirectFollowSalsaExpansion, + 1.0, + 0.0, + 1000.0 + ) + case object CandidateWeightRecentEngagementNonDirectFollow + extends FSBoundedParam[Double]( + PostNuxMlFlowCandidateSourceWeightsFeatureSwitchKeys.CandidateWeightRecentEngagementNonDirectFollow, + 1.0, + 0.0, + 1000.0 + ) + case object CandidateWeightRecentEngagementSimilarUsers + extends FSBoundedParam[Double]( + PostNuxMlFlowCandidateSourceWeightsFeatureSwitchKeys.CandidateWeightRecentEngagementSimilarUsers, + 1.0, + 0.0, + 1000.0 + ) + case object CandidateWeightRepeatedProfileVisits + extends FSBoundedParam[Double]( + PostNuxMlFlowCandidateSourceWeightsFeatureSwitchKeys.CandidateWeightRepeatedProfileVisits, + 1.0, + 0.0, + 1000.0 + ) + case object CandidateWeightFollow2vecNearestNeighbors + extends FSBoundedParam[Double]( + PostNuxMlFlowCandidateSourceWeightsFeatureSwitchKeys.CandidateWeightFollow2vecNearestNeighbors, + 1.0, + 0.0, + 1000.0 + ) + case object CandidateWeightReverseEmailBook + extends FSBoundedParam[Double]( + PostNuxMlFlowCandidateSourceWeightsFeatureSwitchKeys.CandidateWeightReverseEmailBook, + 1.0, + 0.0, + 1000.0 + ) + case object CandidateWeightReversePhoneBook + extends FSBoundedParam[Double]( + PostNuxMlFlowCandidateSourceWeightsFeatureSwitchKeys.CandidateWeightReversePhoneBook, + 1.0, + 0.0, + 1000.0 + ) + case object CandidateWeightTriangularLoops + extends FSBoundedParam[Double]( + PostNuxMlFlowCandidateSourceWeightsFeatureSwitchKeys.CandidateWeightTriangularLoops, + 1.0, + 0.0, + 1000.0 + ) + case object CandidateWeightTwoHopRandomWalk + extends FSBoundedParam[Double]( + PostNuxMlFlowCandidateSourceWeightsFeatureSwitchKeys.CandidateWeightTwoHopRandomWalk, + 1.0, + 0.0, + 1000.0 + ) + case object CandidateWeightUserUserGraph + extends FSBoundedParam[Double]( + PostNuxMlFlowCandidateSourceWeightsFeatureSwitchKeys.CandidateWeightUserUserGraph, + 1.0, + 0.0, + 1000.0 + ) + + case object CandidateWeightRealGraphOonV2 + extends FSBoundedParam[Double]( + PostNuxMlFlowCandidateSourceWeightsFeatureSwitchKeys.CandidateWeightRealGraphOonV2, + 1.0, + 0.0, + 2000.0 + ) +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/post_nux_ml/PostNuxMlCombinedRankerBuilder.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/post_nux_ml/PostNuxMlCombinedRankerBuilder.scala new file mode 100644 index 0000000000..14e982a413 --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/post_nux_ml/PostNuxMlCombinedRankerBuilder.scala @@ -0,0 +1,193 @@ +package com.twitter.follow_recommendations.flows.post_nux_ml + +import com.google.inject.Inject +import com.google.inject.Singleton +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.follow_recommendations.common.base.IdentityRanker +import com.twitter.follow_recommendations.common.base.IdentityTransform +import com.twitter.follow_recommendations.common.base.Ranker +import com.twitter.follow_recommendations.common.base.Transform +import com.twitter.follow_recommendations.common.feature_hydration.common.HasPreFetchedFeature +import com.twitter.follow_recommendations.common.models._ +import com.twitter.follow_recommendations.common.rankers.common.RankerId +import com.twitter.follow_recommendations.common.rankers.fatigue_ranker.ImpressionBasedFatigueRanker +import com.twitter.follow_recommendations.common.rankers.first_n_ranker.FirstNRanker +import com.twitter.follow_recommendations.common.rankers.first_n_ranker.FirstNRankerParams +import com.twitter.follow_recommendations.common.rankers.interleave_ranker.InterleaveRanker +import com.twitter.follow_recommendations.common.rankers.ml_ranker.ranking.HydrateFeaturesTransform +import com.twitter.follow_recommendations.common.rankers.ml_ranker.ranking.MlRanker +import com.twitter.follow_recommendations.common.rankers.ml_ranker.ranking.MlRankerParams +import com.twitter.follow_recommendations.common.rankers.weighted_candidate_source_ranker.WeightedCandidateSourceRanker +import com.twitter.follow_recommendations.configapi.candidates.HydrateCandidateParamsTransform +import com.twitter.product_mixer.core.model.common.identifier.CandidateSourceIdentifier +import com.twitter.product_mixer.core.model.marshalling.request.HasClientContext +import com.twitter.timelines.configapi.HasParams + +/** + * Used to build the combined ranker comprising 4 stages of ranking: + * - weighted sampler + * - truncating to the top N merged results for ranking + * - ML ranker + * - Interleaving ranker for producer-side experiments + * - impression-based fatigueing + */ +@Singleton +class PostNuxMlCombinedRankerBuilder[ + T <: HasParams with HasSimilarToContext with HasClientContext with HasExcludedUserIds with HasDisplayLocation with HasDebugOptions with HasPreFetchedFeature with HasDismissedUserIds with HasQualityFactor] @Inject() ( + firstNRanker: FirstNRanker[T], + hydrateFeaturesTransform: HydrateFeaturesTransform[T], + hydrateCandidateParamsTransform: HydrateCandidateParamsTransform[T], + mlRanker: MlRanker[T], + statsReceiver: StatsReceiver) { + private[this] val stats: StatsReceiver = statsReceiver.scope("post_nux_ml_ranker") + + // we construct each ranker independently and chain them together + def build( + request: T, + candidateSourceWeights: Map[CandidateSourceIdentifier, Double] + ): Ranker[T, CandidateUser] = { + val displayLocationStats = stats.scope(request.displayLocation.toString) + val weightedRankerStats: StatsReceiver = + displayLocationStats.scope("weighted_candidate_source_ranker") + val firstNRankerStats: StatsReceiver = + displayLocationStats.scope("first_n_ranker") + val hydrateCandidateParamsStats = + displayLocationStats.scope("hydrate_candidate_params") + val fatigueRankerStats = displayLocationStats.scope("fatigue_ranker") + val interleaveRankerStats = + displayLocationStats.scope("interleave_ranker") + val allRankersStats = displayLocationStats.scope("all_rankers") + + // Checking if the heavy-ranker is an experimental model. + // If it is, InterleaveRanker and candidate parameter hydration are disabled. + // *NOTE* that consumer-side experiments should at any time take a small % of traffic, less + // than 20% for instance, to leave enough room for producer experiments. Increasing bucket + // size for producer experiments lead to other issues and is not a viable option for faster + // experiments. + val requestRankerId = request.params(MlRankerParams.RequestScorerIdParam) + if (requestRankerId != RankerId.PostNuxProdRanker) { + hydrateCandidateParamsStats.counter(s"disabled_by_${requestRankerId.toString}").incr() + interleaveRankerStats.counter(s"disabled_by_${requestRankerId.toString}").incr() + } + + // weighted ranker that samples from the candidate sources + val weightedRanker = WeightedCandidateSourceRanker + .build[T]( + candidateSourceWeights, + request.params(PostNuxMlParams.CandidateShuffler).shuffle(request.getRandomizationSeed), + randomSeed = request.getRandomizationSeed + ).observe(weightedRankerStats) + + // ranker that takes the first n results (ie truncates output) while merging duplicates + val firstNRankerObs = firstNRanker.observe(firstNRankerStats) + // either ML ranker that uses deepbirdv2 to score or no ranking + val mainRanker: Ranker[T, CandidateUser] = + buildMainRanker(request, requestRankerId == RankerId.PostNuxProdRanker, displayLocationStats) + // fatigue ranker that uses wtf impressions to fatigue + val fatigueRanker = buildFatigueRanker(request, fatigueRankerStats).observe(fatigueRankerStats) + + // interleaveRanker combines rankings from several rankers and enforces candidates' ranks in + // experiment buckets according to their assigned ranker model. + val interleaveRanker = + buildInterleaveRanker( + request, + requestRankerId == RankerId.PostNuxProdRanker, + interleaveRankerStats) + .observe(interleaveRankerStats) + + weightedRanker + .andThen(firstNRankerObs) + .andThen(mainRanker) + .andThen(fatigueRanker) + .andThen(interleaveRanker) + .observe(allRankersStats) + } + + def buildMainRanker( + request: T, + isMainRankerPostNuxProd: Boolean, + displayLocationStats: StatsReceiver + ): Ranker[T, CandidateUser] = { + + // note that we may be disabling heavy ranker for users not bucketed + // (due to empty results from the new candidate source) + // need a better solution in the future + val mlRankerStats = displayLocationStats.scope("ml_ranker") + val noMlRankerStats = displayLocationStats.scope("no_ml_ranker") + val hydrateFeaturesStats = + displayLocationStats.scope("hydrate_features") + val hydrateCandidateParamsStats = + displayLocationStats.scope("hydrate_candidate_params") + val notHydrateCandidateParamsStats = + displayLocationStats.scope("not_hydrate_candidate_params") + val rankerStats = displayLocationStats.scope("ranker") + val mlRankerDisabledByExperimentsCounter = + mlRankerStats.counter("disabled_by_experiments") + val mlRankerDisabledByQualityFactorCounter = + mlRankerStats.counter("disabled_by_quality_factor") + + val disabledByQualityFactor = request.qualityFactor + .exists(_ <= request.params(PostNuxMlParams.TurnoffMLScorerQFThreshold)) + + if (disabledByQualityFactor) + mlRankerDisabledByQualityFactorCounter.incr() + + if (request.params(PostNuxMlParams.UseMlRanker) && !disabledByQualityFactor) { + + val hydrateFeatures = hydrateFeaturesTransform + .observe(hydrateFeaturesStats) + + val optionalHydratedParamsTransform: Transform[T, CandidateUser] = { + // We disable candidate parameter hydration for experimental heavy-ranker models. + if (isMainRankerPostNuxProd && + request.params(PostNuxMlParams.EnableCandidateParamHydration)) { + hydrateCandidateParamsTransform + .observe(hydrateCandidateParamsStats) + } else { + new IdentityTransform[T, CandidateUser]() + .observe(notHydrateCandidateParamsStats) + } + } + val candidateSize = request.params(FirstNRankerParams.CandidatesToRank) + Ranker + .chain( + hydrateFeatures.andThen(optionalHydratedParamsTransform), + mlRanker.observe(mlRankerStats), + ) + .within( + request.params(PostNuxMlParams.MlRankerBudget), + rankerStats.scope(s"n$candidateSize")) + } else { + new IdentityRanker[T, CandidateUser].observe(noMlRankerStats) + } + } + + def buildInterleaveRanker( + request: T, + isMainRankerPostNuxProd: Boolean, + interleaveRankerStats: StatsReceiver + ): Ranker[T, CandidateUser] = { + // InterleaveRanker is enabled only for display locations powered by the PostNux heavy-ranker. + if (request.params(PostNuxMlParams.EnableInterleaveRanker) && + // InterleaveRanker is disabled for requests with experimental heavy-rankers. + isMainRankerPostNuxProd) { + new InterleaveRanker[T](interleaveRankerStats) + } else { + new IdentityRanker[T, CandidateUser]() + } + } + + def buildFatigueRanker( + request: T, + fatigueRankerStats: StatsReceiver + ): Ranker[T, CandidateUser] = { + if (request.params(PostNuxMlParams.EnableFatigueRanker)) { + ImpressionBasedFatigueRanker + .build[T]( + fatigueRankerStats + ).within(request.params(PostNuxMlParams.FatigueRankerBudget), fatigueRankerStats) + } else { + new IdentityRanker[T, CandidateUser]() + } + } +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/post_nux_ml/PostNuxMlFlow.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/post_nux_ml/PostNuxMlFlow.scala new file mode 100644 index 0000000000..092f071001 --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/post_nux_ml/PostNuxMlFlow.scala @@ -0,0 +1,304 @@ +package com.twitter.follow_recommendations.flows.post_nux_ml + +import com.twitter.conversions.DurationOps._ +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.follow_recommendations.common.base.EnrichedCandidateSource._ +import com.twitter.follow_recommendations.common.base._ +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.follow_recommendations.common.models.FilterReason +import com.twitter.follow_recommendations.common.predicates.dismiss.DismissedCandidatePredicate +import com.twitter.follow_recommendations.common.predicates.gizmoduck.GizmoduckPredicate +import com.twitter.follow_recommendations.common.transforms.ranker_id.RandomRankerIdTransform +import com.twitter.follow_recommendations.common.predicates.sgs.InvalidTargetCandidateRelationshipTypesPredicate +import com.twitter.follow_recommendations.common.predicates.sgs.RecentFollowingPredicate +import com.twitter.follow_recommendations.common.predicates.CandidateParamPredicate +import com.twitter.follow_recommendations.common.predicates.CandidateSourceParamPredicate +import com.twitter.follow_recommendations.common.predicates.CuratedCompetitorListPredicate +import com.twitter.follow_recommendations.common.predicates.ExcludedUserIdPredicate +import com.twitter.follow_recommendations.common.predicates.InactivePredicate +import com.twitter.follow_recommendations.common.predicates.PreviouslyRecommendedUserIdsPredicate +import com.twitter.follow_recommendations.common.predicates.user_activity.NonNearZeroUserActivityPredicate +import com.twitter.follow_recommendations.common.transforms.dedup.DedupTransform +import com.twitter.follow_recommendations.common.transforms.modify_social_proof.ModifySocialProofTransform +import com.twitter.follow_recommendations.common.transforms.tracking_token.TrackingTokenTransform +import com.twitter.follow_recommendations.common.transforms.weighted_sampling.SamplingTransform +import com.twitter.follow_recommendations.configapi.candidates.CandidateUserParamsFactory +import com.twitter.follow_recommendations.configapi.params.GlobalParams +import com.twitter.follow_recommendations.configapi.params.GlobalParams.EnableGFSSocialProofTransform +import com.twitter.follow_recommendations.utils.CandidateSourceHoldbackUtil +import com.twitter.product_mixer.core.functional_component.candidate_source.CandidateSource +import com.twitter.product_mixer.core.model.common.identifier.CandidateSourceIdentifier +import com.twitter.timelines.configapi.Params +import com.twitter.util.Duration + +import javax.inject.Inject +import javax.inject.Singleton +import com.twitter.follow_recommendations.common.clients.socialgraph.SocialGraphClient +import com.twitter.follow_recommendations.common.predicates.hss.HssPredicate +import com.twitter.follow_recommendations.common.predicates.sgs.InvalidRelationshipPredicate +import com.twitter.follow_recommendations.common.transforms.modify_social_proof.RemoveAccountProofTransform +import com.twitter.follow_recommendations.logging.FrsLogger +import com.twitter.follow_recommendations.models.RecommendationFlowData +import com.twitter.follow_recommendations.utils.RecommendationFlowBaseSideEffectsUtil +import com.twitter.product_mixer.core.model.common.identifier.RecommendationPipelineIdentifier +import com.twitter.product_mixer.core.quality_factor.BoundsWithDefault +import com.twitter.product_mixer.core.quality_factor.LinearLatencyQualityFactor +import com.twitter.product_mixer.core.quality_factor.LinearLatencyQualityFactorConfig +import com.twitter.product_mixer.core.quality_factor.LinearLatencyQualityFactorObserver +import com.twitter.product_mixer.core.quality_factor.QualityFactorObserver +import com.twitter.stitch.Stitch + +/** + * We use this flow for all post-nux display locations that would use a machine-learning-based-ranker + * eg HTL, Sidebar, etc + * Note that the RankedPostNuxFlow is used primarily for scribing/data collection, and doesn't + * incorporate all of the other components in a flow (candidate source generation, predicates etc) + */ +@Singleton +class PostNuxMlFlow @Inject() ( + postNuxMlCandidateSourceRegistry: PostNuxMlCandidateSourceRegistry, + postNuxMlCombinedRankerBuilder: PostNuxMlCombinedRankerBuilder[PostNuxMlRequest], + curatedCompetitorListPredicate: CuratedCompetitorListPredicate, + gizmoduckPredicate: GizmoduckPredicate, + sgsPredicate: InvalidTargetCandidateRelationshipTypesPredicate, + hssPredicate: HssPredicate, + invalidRelationshipPredicate: InvalidRelationshipPredicate, + recentFollowingPredicate: RecentFollowingPredicate, + nonNearZeroUserActivityPredicate: NonNearZeroUserActivityPredicate, + inactivePredicate: InactivePredicate, + dismissedCandidatePredicate: DismissedCandidatePredicate, + previouslyRecommendedUserIdsPredicate: PreviouslyRecommendedUserIdsPredicate, + modifySocialProofTransform: ModifySocialProofTransform, + removeAccountProofTransform: RemoveAccountProofTransform, + trackingTokenTransform: TrackingTokenTransform, + randomRankerIdTransform: RandomRankerIdTransform, + candidateParamsFactory: CandidateUserParamsFactory[PostNuxMlRequest], + samplingTransform: SamplingTransform, + frsLogger: FrsLogger, + baseStatsReceiver: StatsReceiver) + extends RecommendationFlow[PostNuxMlRequest, CandidateUser] + with RecommendationFlowBaseSideEffectsUtil[PostNuxMlRequest, CandidateUser] + with CandidateSourceHoldbackUtil { + override protected val targetEligibility: Predicate[PostNuxMlRequest] = + new ParamPredicate[PostNuxMlRequest](PostNuxMlParams.TargetEligibility) + + override val statsReceiver: StatsReceiver = baseStatsReceiver.scope("post_nux_ml_flow") + + override val qualityFactorObserver: Option[QualityFactorObserver] = { + val config = LinearLatencyQualityFactorConfig( + qualityFactorBounds = + BoundsWithDefault(minInclusive = 0.1, maxInclusive = 1.0, default = 1.0), + initialDelay = 60.seconds, + targetLatency = 700.milliseconds, + targetLatencyPercentile = 95.0, + delta = 0.001 + ) + val qualityFactor = LinearLatencyQualityFactor(config) + val observer = LinearLatencyQualityFactorObserver(qualityFactor) + statsReceiver.provideGauge("quality_factor")(qualityFactor.currentValue.toFloat) + Some(observer) + } + + override protected def updateTarget(request: PostNuxMlRequest): Stitch[PostNuxMlRequest] = { + Stitch.value( + request.copy(qualityFactor = qualityFactorObserver.map(_.qualityFactor.currentValue)) + ) + } + + private[post_nux_ml] def getCandidateSourceIdentifiers( + params: Params + ): Set[CandidateSourceIdentifier] = { + PostNuxMlFlowCandidateSourceWeights.getWeights(params).keySet + } + + override protected def candidateSources( + request: PostNuxMlRequest + ): Seq[CandidateSource[PostNuxMlRequest, CandidateUser]] = { + val identifiers = getCandidateSourceIdentifiers(request.params) + val selected: Set[CandidateSource[PostNuxMlRequest, CandidateUser]] = + postNuxMlCandidateSourceRegistry.select(identifiers) + val budget: Duration = request.params(PostNuxMlParams.FetchCandidateSourceBudget) + filterCandidateSources( + request, + selected.map(c => c.failOpenWithin(budget, statsReceiver)).toSeq) + } + + override protected val preRankerCandidateFilter: Predicate[(PostNuxMlRequest, CandidateUser)] = { + val stats = statsReceiver.scope("pre_ranker") + + object excludeNearZeroUserPredicate + extends GatedPredicateBase[(PostNuxMlRequest, CandidateUser)]( + nonNearZeroUserActivityPredicate, + stats.scope("exclude_near_zero_predicate") + ) { + override def gate(item: (PostNuxMlRequest, CandidateUser)): Boolean = + item._1.params(PostNuxMlParams.ExcludeNearZeroCandidates) + } + + object invalidRelationshipGatedPredicate + extends GatedPredicateBase[(PostNuxMlRequest, CandidateUser)]( + invalidRelationshipPredicate, + stats.scope("invalid_relationship_predicate") + ) { + override def gate(item: (PostNuxMlRequest, CandidateUser)): Boolean = + item._1.params(PostNuxMlParams.EnableInvalidRelationshipPredicate) + } + + ExcludedUserIdPredicate + .observe(stats.scope("exclude_user_id_predicate")) + .andThen( + recentFollowingPredicate.observe(stats.scope("recent_following_predicate")) + ) + .andThen( + dismissedCandidatePredicate.observe(stats.scope("dismissed_candidate_predicate")) + ) + .andThen( + previouslyRecommendedUserIdsPredicate.observe( + stats.scope("previously_recommended_user_ids_predicate")) + ) + .andThen( + invalidRelationshipGatedPredicate.observe(stats.scope("invalid_relationship_predicate")) + ) + .andThen( + excludeNearZeroUserPredicate.observe(stats.scope("exclude_near_zero_user_state")) + ) + .observe(stats.scope("overall_pre_ranker_candidate_filter")) + } + + override protected def selectRanker( + request: PostNuxMlRequest + ): Ranker[PostNuxMlRequest, CandidateUser] = { + postNuxMlCombinedRankerBuilder.build( + request, + PostNuxMlFlowCandidateSourceWeights.getWeights(request.params)) + } + + override protected val postRankerTransform: Transform[PostNuxMlRequest, CandidateUser] = { + new DedupTransform[PostNuxMlRequest, CandidateUser] + .observe(statsReceiver.scope("dedupping")) + .andThen( + samplingTransform + .gated(PostNuxMlParams.SamplingTransformEnabled) + .observe(statsReceiver.scope("samplingtransform"))) + } + + override protected val validateCandidates: Predicate[(PostNuxMlRequest, CandidateUser)] = { + val stats = statsReceiver.scope("validate_candidates") + val competitorPredicate = + curatedCompetitorListPredicate.map[(PostNuxMlRequest, CandidateUser)](_._2) + + val producerHoldbackPredicate = new CandidateParamPredicate[CandidateUser]( + GlobalParams.KeepUserCandidate, + FilterReason.CandidateSideHoldback + ).map[(PostNuxMlRequest, CandidateUser)] { + case (request, user) => candidateParamsFactory(user, request) + } + val pymkProducerHoldbackPredicate = new CandidateSourceParamPredicate( + GlobalParams.KeepSocialUserCandidate, + FilterReason.CandidateSideHoldback, + CandidateSourceHoldbackUtil.SocialCandidateSourceIds + ).map[(PostNuxMlRequest, CandidateUser)] { + case (request, user) => candidateParamsFactory(user, request) + } + val sgsPredicateStats = stats.scope("sgs_predicate") + object sgsGatedPredicate + extends GatedPredicateBase[(PostNuxMlRequest, CandidateUser)]( + sgsPredicate.observe(sgsPredicateStats), + sgsPredicateStats + ) { + + /** + * When SGS predicate is turned off, only query SGS exists API for (user, candidate, relationship) + * when the user's number of invalid relationships exceeds the threshold during request + * building step. This is to minimize load to SGS and underlying Flock DB. + */ + override def gate(item: (PostNuxMlRequest, CandidateUser)): Boolean = + item._1.params(PostNuxMlParams.EnableSGSPredicate) || + SocialGraphClient.enablePostRankerSgsPredicate( + item._1.invalidRelationshipUserIds.getOrElse(Set.empty).size) + } + + val hssPredicateStats = stats.scope("hss_predicate") + object hssGatedPredicate + extends GatedPredicateBase[(PostNuxMlRequest, CandidateUser)]( + hssPredicate.observe(hssPredicateStats), + hssPredicateStats + ) { + override def gate(item: (PostNuxMlRequest, CandidateUser)): Boolean = + item._1.params(PostNuxMlParams.EnableHssPredicate) + } + + Predicate + .andConcurrently[(PostNuxMlRequest, CandidateUser)]( + Seq( + competitorPredicate.observe(stats.scope("curated_competitor_predicate")), + gizmoduckPredicate.observe(stats.scope("gizmoduck_predicate")), + sgsGatedPredicate, + hssGatedPredicate, + inactivePredicate.observe(stats.scope("inactive_predicate")), + ) + ) + // to avoid dilutions, we need to apply the receiver holdback predicates at the very last step + .andThen(pymkProducerHoldbackPredicate.observe(stats.scope("pymk_receiver_side_holdback"))) + .andThen(producerHoldbackPredicate.observe(stats.scope("receiver_side_holdback"))) + .observe(stats.scope("overall_validate_candidates")) + } + + override protected val transformResults: Transform[PostNuxMlRequest, CandidateUser] = { + modifySocialProofTransform + .gated(EnableGFSSocialProofTransform) + .andThen(trackingTokenTransform) + .andThen(randomRankerIdTransform.gated(PostNuxMlParams.LogRandomRankerId)) + .andThen(removeAccountProofTransform.gated(PostNuxMlParams.EnableRemoveAccountProofTransform)) + } + + override protected def resultsConfig(request: PostNuxMlRequest): RecommendationResultsConfig = { + RecommendationResultsConfig( + request.maxResults.getOrElse(request.params(PostNuxMlParams.ResultSizeParam)), + request.params(PostNuxMlParams.BatchSizeParam) + ) + } + + override def applySideEffects( + target: PostNuxMlRequest, + candidateSources: Seq[CandidateSource[PostNuxMlRequest, CandidateUser]], + candidatesFromCandidateSources: Seq[CandidateUser], + mergedCandidates: Seq[CandidateUser], + filteredCandidates: Seq[CandidateUser], + rankedCandidates: Seq[CandidateUser], + transformedCandidates: Seq[CandidateUser], + truncatedCandidates: Seq[CandidateUser], + results: Seq[CandidateUser] + ): Stitch[Unit] = { + frsLogger.logRecommendationFlowData[PostNuxMlRequest]( + target, + RecommendationFlowData[PostNuxMlRequest]( + target, + PostNuxMlFlow.identifier, + candidateSources, + candidatesFromCandidateSources, + mergedCandidates, + filteredCandidates, + rankedCandidates, + transformedCandidates, + truncatedCandidates, + results + ) + ) + super.applySideEffects( + target, + candidateSources, + candidatesFromCandidateSources, + mergedCandidates, + filteredCandidates, + rankedCandidates, + transformedCandidates, + truncatedCandidates, + results + ) + } +} + +object PostNuxMlFlow { + val identifier = RecommendationPipelineIdentifier("PostNuxMlFlow") +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/post_nux_ml/PostNuxMlFlowCandidateSourceWeights.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/post_nux_ml/PostNuxMlFlowCandidateSourceWeights.scala new file mode 100644 index 0000000000..edb447cba3 --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/post_nux_ml/PostNuxMlFlowCandidateSourceWeights.scala @@ -0,0 +1,68 @@ +package com.twitter.follow_recommendations.flows.post_nux_ml + +import com.twitter.follow_recommendations.common.candidate_sources.addressbook.ForwardEmailBookSource +import com.twitter.follow_recommendations.common.candidate_sources.addressbook.ForwardPhoneBookSource +import com.twitter.follow_recommendations.common.candidate_sources.addressbook.ReverseEmailBookSource +import com.twitter.follow_recommendations.common.candidate_sources.addressbook.ReversePhoneBookSource +import com.twitter.follow_recommendations.common.candidate_sources.crowd_search_accounts.CrowdSearchAccountsSource +import com.twitter.follow_recommendations.common.candidate_sources.geo.PopCountryBackFillSource +import com.twitter.follow_recommendations.common.candidate_sources.geo.PopCountrySource +import com.twitter.follow_recommendations.common.candidate_sources.geo.PopGeohashQualityFollowSource +import com.twitter.follow_recommendations.common.candidate_sources.geo.PopGeohashSource +import com.twitter.follow_recommendations.common.candidate_sources.ppmi_locale_follow.PPMILocaleFollowSource +import com.twitter.follow_recommendations.common.candidate_sources.real_graph.RealGraphOonV2Source +import com.twitter.follow_recommendations.common.candidate_sources.recent_engagement.RecentEngagementNonDirectFollowSource +import com.twitter.follow_recommendations.common.candidate_sources.recent_engagement.RepeatedProfileVisitsSource +import com.twitter.follow_recommendations.common.candidate_sources.salsa.RecentEngagementDirectFollowSalsaExpansionSource +import com.twitter.follow_recommendations.common.candidate_sources.sims_expansion.RecentEngagementSimilarUsersSource +import com.twitter.follow_recommendations.common.candidate_sources.sims_expansion.RecentFollowingSimilarUsersSource +import com.twitter.follow_recommendations.common.candidate_sources.sims.Follow2vecNearestNeighborsStore +import com.twitter.follow_recommendations.common.candidate_sources.stp.BaseOnlineSTPSource +import com.twitter.follow_recommendations.common.candidate_sources.stp.OfflineStrongTiePredictionSource +import com.twitter.follow_recommendations.common.candidate_sources.top_organic_follows_accounts.TopOrganicFollowsAccountsSource +import com.twitter.follow_recommendations.common.candidate_sources.triangular_loops.TriangularLoopsSource +import com.twitter.follow_recommendations.common.candidate_sources.two_hop_random_walk.TwoHopRandomWalkSource +import com.twitter.follow_recommendations.common.candidate_sources.user_user_graph.UserUserGraphCandidateSource +import com.twitter.follow_recommendations.flows.post_nux_ml.PostNuxMlCandidateSourceWeightParams._ +import com.twitter.product_mixer.core.model.common.identifier.CandidateSourceIdentifier +import com.twitter.timelines.configapi.Params + +object PostNuxMlFlowCandidateSourceWeights { + + def getWeights(params: Params): Map[CandidateSourceIdentifier, Double] = { + Map[CandidateSourceIdentifier, Double]( + // Social based + PPMILocaleFollowSource.Identifier -> params(CandidateWeightPPMILocaleFollow), + Follow2vecNearestNeighborsStore.IdentifierF2vLinearRegression -> params( + CandidateWeightFollow2vecNearestNeighbors), + RecentFollowingSimilarUsersSource.Identifier -> params( + CandidateWeightRecentFollowingSimilarUsers), + BaseOnlineSTPSource.Identifier -> params(CandidateWeightOnlineStp), + OfflineStrongTiePredictionSource.Identifier -> params( + CandidateWeightOfflineStrongTiePrediction), + ForwardEmailBookSource.Identifier -> params(CandidateWeightForwardEmailBook), + ForwardPhoneBookSource.Identifier -> params(CandidateWeightForwardPhoneBook), + ReverseEmailBookSource.Identifier -> params(CandidateWeightReverseEmailBook), + ReversePhoneBookSource.Identifier -> params(CandidateWeightReversePhoneBook), + TriangularLoopsSource.Identifier -> params(CandidateWeightTriangularLoops), + TwoHopRandomWalkSource.Identifier -> params(CandidateWeightTwoHopRandomWalk), + UserUserGraphCandidateSource.Identifier -> params(CandidateWeightUserUserGraph), + // Geo based + PopCountrySource.Identifier -> params(CandidateWeightPopCountry), + PopCountryBackFillSource.Identifier -> params(CandidateWeightPopGeoBackfill), + PopGeohashSource.Identifier -> params(CandidateWeightPopGeohash), + PopGeohashQualityFollowSource.Identifier -> params(CandidateWeightPopGeohashQualityFollow), + CrowdSearchAccountsSource.Identifier -> params(CandidateWeightCrowdSearch), + TopOrganicFollowsAccountsSource.Identifier -> params(CandidateWeightTopOrganicFollow), + // Engagement based + RealGraphOonV2Source.Identifier -> params(CandidateWeightRealGraphOonV2), + RecentEngagementNonDirectFollowSource.Identifier -> params( + CandidateWeightRecentEngagementNonDirectFollow), + RecentEngagementSimilarUsersSource.Identifier -> params( + CandidateWeightRecentEngagementSimilarUsers), + RepeatedProfileVisitsSource.Identifier -> params(CandidateWeightRepeatedProfileVisits), + RecentEngagementDirectFollowSalsaExpansionSource.Identifier -> params( + CandidateWeightRecentEngagementDirectFollowSalsaExpansion), + ) + } +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/post_nux_ml/PostNuxMlFlowCandidateSourceWeightsFeatureSwitchKeys.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/post_nux_ml/PostNuxMlFlowCandidateSourceWeightsFeatureSwitchKeys.scala new file mode 100644 index 0000000000..f329cbd13b --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/post_nux_ml/PostNuxMlFlowCandidateSourceWeightsFeatureSwitchKeys.scala @@ -0,0 +1,46 @@ +package com.twitter.follow_recommendations.flows.post_nux_ml + +object PostNuxMlFlowCandidateSourceWeightsFeatureSwitchKeys { + val CandidateWeightCrowdSearch = "post_nux_ml_flow_candidate_source_weights_user_crowd_search" + val CandidateWeightTopOrganicFollow = + "post_nux_ml_flow_candidate_source_weights_top_organic_follow" + val CandidateWeightPPMILocaleFollow = + "post_nux_ml_flow_candidate_source_weights_user_ppmi_locale_follow" + val CandidateWeightForwardEmailBook = + "post_nux_ml_flow_candidate_source_weights_user_forward_email_book" + val CandidateWeightForwardPhoneBook = + "post_nux_ml_flow_candidate_source_weights_user_forward_phone_book" + val CandidateWeightOfflineStrongTiePrediction = + "post_nux_ml_flow_candidate_source_weights_user_offline_strong_tie_prediction" + val CandidateWeightOnlineStp = "post_nux_ml_flow_candidate_source_weights_user_online_stp" + val CandidateWeightPopCountry = "post_nux_ml_flow_candidate_source_weights_user_pop_country" + val CandidateWeightPopGeohash = "post_nux_ml_flow_candidate_source_weights_user_pop_geohash" + val CandidateWeightPopGeohashQualityFollow = + "post_nux_ml_flow_candidate_source_weights_user_pop_geohash_quality_follow" + val CandidateWeightPopGeoBackfill = + "post_nux_ml_flow_candidate_source_weights_user_pop_geo_backfill" + val CandidateWeightRecentFollowingSimilarUsers = + "post_nux_ml_flow_candidate_source_weights_user_recent_following_similar_users" + val CandidateWeightRecentEngagementDirectFollowSalsaExpansion = + "post_nux_ml_flow_candidate_source_weights_user_recent_engagement_direct_follow_salsa_expansion" + val CandidateWeightRecentEngagementNonDirectFollow = + "post_nux_ml_flow_candidate_source_weights_user_recent_engagement_non_direct_follow" + val CandidateWeightRecentEngagementSimilarUsers = + "post_nux_ml_flow_candidate_source_weights_user_recent_engagement_similar_users" + val CandidateWeightRepeatedProfileVisits = + "post_nux_ml_flow_candidate_source_weights_user_repeated_profile_visits" + val CandidateWeightFollow2vecNearestNeighbors = + "post_nux_ml_flow_candidate_source_weights_user_follow2vec_nearest_neighbors" + val CandidateWeightReverseEmailBook = + "post_nux_ml_flow_candidate_source_weights_user_reverse_email_book" + val CandidateWeightReversePhoneBook = + "post_nux_ml_flow_candidate_source_weights_user_reverse_phone_book" + val CandidateWeightTriangularLoops = + "post_nux_ml_flow_candidate_source_weights_user_triangular_loops" + val CandidateWeightTwoHopRandomWalk = + "post_nux_ml_flow_candidate_source_weights_user_two_hop_random_walk" + val CandidateWeightUserUserGraph = + "post_nux_ml_flow_candidate_source_weights_user_user_user_graph" + val CandidateWeightRealGraphOonV2 = + "post_nux_ml_flow_candidate_source_weights_user_real_graph_oon_v2" +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/post_nux_ml/PostNuxMlFlowFSConfig.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/post_nux_ml/PostNuxMlFlowFSConfig.scala new file mode 100644 index 0000000000..0dd059dad7 --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/post_nux_ml/PostNuxMlFlowFSConfig.scala @@ -0,0 +1,80 @@ +package com.twitter.follow_recommendations.flows.post_nux_ml + +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.follow_recommendations.common.rankers.weighted_candidate_source_ranker.NoShuffle +import com.twitter.follow_recommendations.common.rankers.weighted_candidate_source_ranker.RandomShuffler +import com.twitter.follow_recommendations.configapi.common.FeatureSwitchConfig +import com.twitter.timelines.configapi.FSBoundedParam +import com.twitter.timelines.configapi.FSName +import com.twitter.timelines.configapi.HasDurationConversion +import com.twitter.timelines.configapi.Param +import com.twitter.util.Duration +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class PostNuxMlFlowFSConfig @Inject() () extends FeatureSwitchConfig { + override val booleanFSParams: Seq[Param[Boolean] with FSName] = Seq( + PostNuxMlParams.OnlineSTPEnabled, + PostNuxMlParams.SamplingTransformEnabled, + PostNuxMlParams.Follow2VecLinearRegressionEnabled, + PostNuxMlParams.UseMlRanker, + PostNuxMlParams.EnableCandidateParamHydration, + PostNuxMlParams.EnableInterleaveRanker, + PostNuxMlParams.EnableAdhocRanker, + PostNuxMlParams.ExcludeNearZeroCandidates, + PostNuxMlParams.IncludeRepeatedProfileVisitsCandidateSource, + PostNuxMlParams.EnableInterestsOptOutPredicate, + PostNuxMlParams.EnableSGSPredicate, + PostNuxMlParams.EnableInvalidRelationshipPredicate, + PostNuxMlParams.EnableRemoveAccountProofTransform, + PostNuxMlParams.EnablePPMILocaleFollowSourceInPostNux, + PostNuxMlParams.EnableRealGraphOonV2, + PostNuxMlParams.GetFollowersFromSgs, + PostNuxMlRequestBuilderParams.EnableInvalidRelationshipPredicate + ) + + override val doubleFSParams: Seq[FSBoundedParam[Double]] = Seq( + PostNuxMlCandidateSourceWeightParams.CandidateWeightCrowdSearch, + PostNuxMlCandidateSourceWeightParams.CandidateWeightTopOrganicFollow, + PostNuxMlCandidateSourceWeightParams.CandidateWeightPPMILocaleFollow, + PostNuxMlCandidateSourceWeightParams.CandidateWeightForwardEmailBook, + PostNuxMlCandidateSourceWeightParams.CandidateWeightForwardPhoneBook, + PostNuxMlCandidateSourceWeightParams.CandidateWeightOfflineStrongTiePrediction, + PostNuxMlCandidateSourceWeightParams.CandidateWeightOnlineStp, + PostNuxMlCandidateSourceWeightParams.CandidateWeightPopCountry, + PostNuxMlCandidateSourceWeightParams.CandidateWeightPopGeohash, + PostNuxMlCandidateSourceWeightParams.CandidateWeightPopGeohashQualityFollow, + PostNuxMlCandidateSourceWeightParams.CandidateWeightPopGeoBackfill, + PostNuxMlCandidateSourceWeightParams.CandidateWeightRecentFollowingSimilarUsers, + PostNuxMlCandidateSourceWeightParams.CandidateWeightRecentEngagementDirectFollowSalsaExpansion, + PostNuxMlCandidateSourceWeightParams.CandidateWeightRecentEngagementNonDirectFollow, + PostNuxMlCandidateSourceWeightParams.CandidateWeightRecentEngagementSimilarUsers, + PostNuxMlCandidateSourceWeightParams.CandidateWeightRepeatedProfileVisits, + PostNuxMlCandidateSourceWeightParams.CandidateWeightFollow2vecNearestNeighbors, + PostNuxMlCandidateSourceWeightParams.CandidateWeightReverseEmailBook, + PostNuxMlCandidateSourceWeightParams.CandidateWeightReversePhoneBook, + PostNuxMlCandidateSourceWeightParams.CandidateWeightTriangularLoops, + PostNuxMlCandidateSourceWeightParams.CandidateWeightTwoHopRandomWalk, + PostNuxMlCandidateSourceWeightParams.CandidateWeightUserUserGraph, + PostNuxMlCandidateSourceWeightParams.CandidateWeightRealGraphOonV2, + PostNuxMlParams.TurnoffMLScorerQFThreshold + ) + + override val durationFSParams: Seq[FSBoundedParam[Duration] with HasDurationConversion] = Seq( + PostNuxMlParams.MlRankerBudget, + PostNuxMlRequestBuilderParams.TopicIdFetchBudget, + PostNuxMlRequestBuilderParams.DismissedIdScanBudget, + PostNuxMlRequestBuilderParams.WTFImpressionsScanBudget + ) + + override val gatedOverridesMap = Map( + PostNuxMlFlowFeatureSwitchKeys.EnableRandomDataCollection -> Seq( + PostNuxMlParams.CandidateShuffler := new RandomShuffler[CandidateUser], + PostNuxMlParams.LogRandomRankerId := true + ), + PostNuxMlFlowFeatureSwitchKeys.EnableNoShuffler -> Seq( + PostNuxMlParams.CandidateShuffler := new NoShuffle[CandidateUser] + ), + ) +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/post_nux_ml/PostNuxMlFlowFeatureSwitchKeys.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/post_nux_ml/PostNuxMlFlowFeatureSwitchKeys.scala new file mode 100644 index 0000000000..6a44c4bbbb --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/post_nux_ml/PostNuxMlFlowFeatureSwitchKeys.scala @@ -0,0 +1,27 @@ +package com.twitter.follow_recommendations.flows.post_nux_ml + +object PostNuxMlFlowFeatureSwitchKeys { + val UseMlRanker = "post_nux_ml_flow_use_ml_ranker" + val EnableCandidateParamHydration = "post_nux_ml_flow_enable_candidate_param_hydration" + val OnlineSTPEnabled = "post_nux_ml_flow_online_stp_source_enabled" + val Follow2VecLinearRegressionEnabled = "post_nux_ml_flow_follow_to_vec_lr_source_enabled" + val EnableRandomDataCollection = "post_nux_ml_flow_random_data_collection_enabled" + val EnableAdhocRanker = "post_nux_ml_flow_adhoc_ranker_enabled" + val EnableFatigueRanker = "post_nux_ml_flow_fatigue_ranker_enabled" + val EnableInterleaveRanker = "post_nux_ml_flow_interleave_ranker_enabled" + val IncludeRepeatedProfileVisitsCandidateSource = + "post_nux_ml_flow_include_repeated_profile_visits_candidate_source" + val MLRankerBudget = "post_nux_ml_flow_ml_ranker_budget_millis" + val EnableNoShuffler = "post_nux_ml_flow_no_shuffler" + val SamplingTransformEnabled = "post_nux_ml_flow_sampling_transform_enabled" + val ExcludeNearZeroCandidates = "post_nux_ml_flow_exclude_near_zero_candidates" + val EnableInterestsOptOutPredicate = "post_nux_ml_flow_enable_interests_opt_out_predicate" + val EnableRemoveAccountProofTransform = "post_nux_ml_flow_enable_remove_account_proof_transform" + val EnablePPMILocaleFollowSourceInPostNux = "post_nux_ml_flow_enable_ppmilocale_follow_source" + val EnableInvalidRelationshipPredicate = "post_nux_ml_flow_enable_invalid_relationship_predicate" + val EnableRealGraphOonV2 = "post_nux_ml_flow_enable_real_graph_oon_v2" + val EnableSGSPredicate = "post_nux_ml_flow_enable_sgs_predicate" + val EnableHssPredicate = "post_nux_ml_flow_enable_hss_predicate" + val GetFollowersFromSgs = "post_nux_ml_flow_get_followers_from_sgs" + val TurnOffMLScorerQFThreshold = "post_nux_ml_flow_turn_off_ml_scorer_threhsold" +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/post_nux_ml/PostNuxMlParams.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/post_nux_ml/PostNuxMlParams.scala new file mode 100644 index 0000000000..cb5cf3648e --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/post_nux_ml/PostNuxMlParams.scala @@ -0,0 +1,133 @@ +package com.twitter.follow_recommendations.flows.post_nux_ml + +import com.twitter.conversions.DurationOps._ +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.follow_recommendations.common.rankers.weighted_candidate_source_ranker.CandidateShuffler +import com.twitter.follow_recommendations.common.rankers.weighted_candidate_source_ranker.ExponentialShuffler +import com.twitter.timelines.configapi.DurationConversion +import com.twitter.timelines.configapi.FSBoundedParam +import com.twitter.timelines.configapi.FSParam +import com.twitter.timelines.configapi.HasDurationConversion +import com.twitter.timelines.configapi.Param +import com.twitter.util.Duration + +abstract class PostNuxMlParams[A](default: A) extends Param[A](default) { + override val statName: String = "post_nux_ml/" + this.getClass.getSimpleName +} + +object PostNuxMlParams { + + // infra params: + case object FetchCandidateSourceBudget extends PostNuxMlParams[Duration](90.millisecond) + + // WTF Impression Store has very high tail latency (p9990 or p9999), but p99 latency is pretty good (~100ms) + // set the time budget for this step to be 200ms to make the performance of service more predictable + case object FatigueRankerBudget extends PostNuxMlParams[Duration](200.millisecond) + + case object MlRankerBudget + extends FSBoundedParam[Duration]( + name = PostNuxMlFlowFeatureSwitchKeys.MLRankerBudget, + default = 400.millisecond, + min = 100.millisecond, + max = 800.millisecond) + with HasDurationConversion { + override val durationConversion: DurationConversion = DurationConversion.FromMillis + } + + // product params: + case object TargetEligibility extends PostNuxMlParams[Boolean](true) + + case object ResultSizeParam extends PostNuxMlParams[Int](3) + case object BatchSizeParam extends PostNuxMlParams[Int](12) + + case object CandidateShuffler + extends PostNuxMlParams[CandidateShuffler[CandidateUser]]( + new ExponentialShuffler[CandidateUser]) + case object LogRandomRankerId extends PostNuxMlParams[Boolean](false) + + // whether or not to use the ml ranker at all (feature hydration + ranker) + case object UseMlRanker + extends FSParam[Boolean](PostNuxMlFlowFeatureSwitchKeys.UseMlRanker, false) + + // whether or not to enable candidate param hydration in postnux_ml_flow + case object EnableCandidateParamHydration + extends FSParam[Boolean](PostNuxMlFlowFeatureSwitchKeys.EnableCandidateParamHydration, false) + + // Whether or not OnlineSTP candidates are considered in the final pool of candidates. + // If set to `false`, the candidate source will be removed *after* all other considerations. + case object OnlineSTPEnabled + extends FSParam[Boolean](PostNuxMlFlowFeatureSwitchKeys.OnlineSTPEnabled, false) + + // Whether or not the candidates are sampled from a Plackett-Luce model + case object SamplingTransformEnabled + extends FSParam[Boolean](PostNuxMlFlowFeatureSwitchKeys.SamplingTransformEnabled, false) + + // Whether or not Follow2Vec candidates are considered in the final pool of candidates. + // If set to `false`, the candidate source will be removed *after* all other considerations. + case object Follow2VecLinearRegressionEnabled + extends FSParam[Boolean]( + PostNuxMlFlowFeatureSwitchKeys.Follow2VecLinearRegressionEnabled, + false) + + // Whether or not to enable AdhocRanker to allow adhoc, non-ML, score modifications. + case object EnableAdhocRanker + extends FSParam[Boolean](PostNuxMlFlowFeatureSwitchKeys.EnableAdhocRanker, false) + + // Whether the impression-based fatigue ranker is enabled or not. + case object EnableFatigueRanker + extends FSParam[Boolean](PostNuxMlFlowFeatureSwitchKeys.EnableFatigueRanker, true) + + // whether or not to enable InterleaveRanker for producer-side experiments. + case object EnableInterleaveRanker + extends FSParam[Boolean](PostNuxMlFlowFeatureSwitchKeys.EnableInterleaveRanker, false) + + // whether to exclude users in near zero user state + case object ExcludeNearZeroCandidates + extends FSParam[Boolean](PostNuxMlFlowFeatureSwitchKeys.ExcludeNearZeroCandidates, false) + + case object EnablePPMILocaleFollowSourceInPostNux + extends FSParam[Boolean]( + PostNuxMlFlowFeatureSwitchKeys.EnablePPMILocaleFollowSourceInPostNux, + false) + + case object EnableInterestsOptOutPredicate + extends FSParam[Boolean](PostNuxMlFlowFeatureSwitchKeys.EnableInterestsOptOutPredicate, false) + + case object EnableInvalidRelationshipPredicate + extends FSParam[Boolean]( + PostNuxMlFlowFeatureSwitchKeys.EnableInvalidRelationshipPredicate, + false) + + // Totally disabling SGS predicate need to disable EnableInvalidRelationshipPredicate as well + case object EnableSGSPredicate + extends FSParam[Boolean](PostNuxMlFlowFeatureSwitchKeys.EnableSGSPredicate, true) + + case object EnableHssPredicate + extends FSParam[Boolean](PostNuxMlFlowFeatureSwitchKeys.EnableHssPredicate, true) + + // Whether or not to include RepeatedProfileVisits as one of the candidate sources in the PostNuxMlFlow. If false, + // RepeatedProfileVisitsSource would not be run for the users in candidate_generation. + case object IncludeRepeatedProfileVisitsCandidateSource + extends FSParam[Boolean]( + PostNuxMlFlowFeatureSwitchKeys.IncludeRepeatedProfileVisitsCandidateSource, + false) + + case object EnableRealGraphOonV2 + extends FSParam[Boolean](PostNuxMlFlowFeatureSwitchKeys.EnableRealGraphOonV2, false) + + case object GetFollowersFromSgs + extends FSParam[Boolean](PostNuxMlFlowFeatureSwitchKeys.GetFollowersFromSgs, false) + + case object EnableRemoveAccountProofTransform + extends FSParam[Boolean]( + PostNuxMlFlowFeatureSwitchKeys.EnableRemoveAccountProofTransform, + false) + + // quality factor threshold to turn off ML ranker completely + object TurnoffMLScorerQFThreshold + extends FSBoundedParam[Double]( + name = PostNuxMlFlowFeatureSwitchKeys.TurnOffMLScorerQFThreshold, + default = 0.3, + min = 0.1, + max = 1.0) +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/post_nux_ml/PostNuxMlRequest.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/post_nux_ml/PostNuxMlRequest.scala new file mode 100644 index 0000000000..2cb1126380 --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/post_nux_ml/PostNuxMlRequest.scala @@ -0,0 +1,54 @@ +package com.twitter.follow_recommendations.flows.post_nux_ml + +import com.twitter.core_workflows.user_model.thriftscala.UserState +import com.twitter.follow_recommendations.common.feature_hydration.common.HasPreFetchedFeature +import com.twitter.follow_recommendations.common.models._ +import com.twitter.product_mixer.core.model.marshalling.request.ClientContext +import com.twitter.product_mixer.core.model.marshalling.request.HasClientContext +import com.twitter.timelines.configapi.HasParams +import com.twitter.timelines.configapi.Params + +case class PostNuxMlRequest( + override val params: Params, + override val clientContext: ClientContext, + override val similarToUserIds: Seq[Long], + inputExcludeUserIds: Seq[Long], + override val recentFollowedUserIds: Option[Seq[Long]], + override val invalidRelationshipUserIds: Option[Set[Long]], + override val recentFollowedByUserIds: Option[Seq[Long]], + override val dismissedUserIds: Option[Seq[Long]], + override val displayLocation: DisplayLocation, + maxResults: Option[Int] = None, + override val debugOptions: Option[DebugOptions] = None, + override val wtfImpressions: Option[Seq[WtfImpression]], + override val uttInterestIds: Option[Seq[Long]] = None, + override val customInterests: Option[Seq[String]] = None, + override val geohashAndCountryCode: Option[GeohashAndCountryCode] = None, + inputPreviouslyRecommendedUserIds: Option[Set[Long]] = None, + inputPreviouslyFollowedUserIds: Option[Set[Long]] = None, + override val isSoftUser: Boolean = false, + override val userState: Option[UserState] = None, + override val qualityFactor: Option[Double] = None) + extends HasParams + with HasSimilarToContext + with HasClientContext + with HasExcludedUserIds + with HasDisplayLocation + with HasDebugOptions + with HasGeohashAndCountryCode + with HasPreFetchedFeature + with HasDismissedUserIds + with HasInterestIds + with HasPreviousRecommendationsContext + with HasIsSoftUser + with HasUserState + with HasInvalidRelationshipUserIds + with HasQualityFactor { + override val excludedUserIds: Seq[Long] = { + inputExcludeUserIds ++ clientContext.userId.toSeq ++ similarToUserIds + } + override val previouslyRecommendedUserIDs: Set[Long] = + inputPreviouslyRecommendedUserIds.getOrElse(Set.empty) + override val previouslyFollowedUserIds: Set[Long] = + inputPreviouslyFollowedUserIds.getOrElse(Set.empty) +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/post_nux_ml/PostNuxMlRequestBuilder.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/post_nux_ml/PostNuxMlRequestBuilder.scala new file mode 100644 index 0000000000..aeb248b7f1 --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/post_nux_ml/PostNuxMlRequestBuilder.scala @@ -0,0 +1,173 @@ +package com.twitter.follow_recommendations.flows.post_nux_ml + +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.follow_recommendations.common.clients.dismiss_store.DismissStore +import com.twitter.follow_recommendations.common.clients.geoduck.UserLocationFetcher +import com.twitter.follow_recommendations.common.clients.impression_store.WtfImpressionStore +import com.twitter.follow_recommendations.common.clients.interests_service.InterestServiceClient +import com.twitter.follow_recommendations.common.clients.socialgraph.SocialGraphClient +import com.twitter.follow_recommendations.common.clients.user_state.UserStateClient +import com.twitter.follow_recommendations.common.predicates.dismiss.DismissedCandidatePredicateParams +import com.twitter.follow_recommendations.common.utils.RescueWithStatsUtils._ +import com.twitter.follow_recommendations.flows.post_nux_ml.PostNuxMlRequestBuilderParams.DismissedIdScanBudget +import com.twitter.follow_recommendations.flows.post_nux_ml.PostNuxMlRequestBuilderParams.TopicIdFetchBudget +import com.twitter.follow_recommendations.flows.post_nux_ml.PostNuxMlRequestBuilderParams.WTFImpressionsScanBudget +import com.twitter.follow_recommendations.products.common.ProductRequest +import com.twitter.inject.Logging +import com.twitter.stitch.Stitch +import com.twitter.util.Time +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class PostNuxMlRequestBuilder @Inject() ( + socialGraph: SocialGraphClient, + wtfImpressionStore: WtfImpressionStore, + dismissStore: DismissStore, + userLocationFetcher: UserLocationFetcher, + interestServiceClient: InterestServiceClient, + userStateClient: UserStateClient, + statsReceiver: StatsReceiver) + extends Logging { + + val stats: StatsReceiver = statsReceiver.scope("post_nux_ml_request_builder") + val invalidRelationshipUsersStats: StatsReceiver = stats.scope("invalidRelationshipUserIds") + private val invalidRelationshipUsersMaxSizeCounter = + invalidRelationshipUsersStats.counter("maxSize") + private val invalidRelationshipUsersNotMaxSizeCounter = + invalidRelationshipUsersStats.counter("notMaxSize") + + def build( + req: ProductRequest, + previouslyRecommendedUserIds: Option[Set[Long]] = None, + previouslyFollowedUserIds: Option[Set[Long]] = None + ): Stitch[PostNuxMlRequest] = { + val dl = req.recommendationRequest.displayLocation + val resultsStitch = Stitch.collect( + req.recommendationRequest.clientContext.userId + .map { userId => + val lookBackDuration = req.params(DismissedCandidatePredicateParams.LookBackDuration) + val negativeStartTs = -(Time.now - lookBackDuration).inMillis + val recentFollowedUserIdsStitch = + rescueWithStats( + socialGraph.getRecentFollowedUserIds(userId), + stats, + "recentFollowedUserIds") + val invalidRelationshipUserIdsStitch = + if (req.params(PostNuxMlParams.EnableInvalidRelationshipPredicate)) { + rescueWithStats( + socialGraph + .getInvalidRelationshipUserIds(userId) + .onSuccess(ids => + if (ids.size >= SocialGraphClient.MaxNumInvalidRelationship) { + invalidRelationshipUsersMaxSizeCounter.incr() + } else { + invalidRelationshipUsersNotMaxSizeCounter.incr() + }), + stats, + "invalidRelationshipUserIds" + ) + } else { + Stitch.value(Seq.empty) + } + // recentFollowedByUserIds are only used in experiment candidate sources + val recentFollowedByUserIdsStitch = if (req.params(PostNuxMlParams.GetFollowersFromSgs)) { + rescueWithStats( + socialGraph.getRecentFollowedByUserIdsFromCachedColumn(userId), + stats, + "recentFollowedByUserIds") + } else Stitch.value(Seq.empty) + val wtfImpressionsStitch = + rescueWithStatsWithin( + wtfImpressionStore.get(userId, dl), + stats, + "wtfImpressions", + req.params(WTFImpressionsScanBudget)) + val dismissedUserIdsStitch = + rescueWithStatsWithin( + dismissStore.get(userId, negativeStartTs, None), + stats, + "dismissedUserIds", + req.params(DismissedIdScanBudget)) + val locationStitch = + rescueOptionalWithStats( + userLocationFetcher.getGeohashAndCountryCode( + Some(userId), + req.recommendationRequest.clientContext.ipAddress), + stats, + "userLocation" + ) + val topicIdsStitch = + rescueWithStatsWithin( + interestServiceClient.fetchUttInterestIds(userId), + stats, + "topicIds", + req.params(TopicIdFetchBudget)) + val userStateStitch = + rescueOptionalWithStats(userStateClient.getUserState(userId), stats, "userState") + Stitch.join( + recentFollowedUserIdsStitch, + invalidRelationshipUserIdsStitch, + recentFollowedByUserIdsStitch, + dismissedUserIdsStitch, + wtfImpressionsStitch, + locationStitch, + topicIdsStitch, + userStateStitch + ) + }) + + resultsStitch.map { + case Some( + ( + recentFollowedUserIds, + invalidRelationshipUserIds, + recentFollowedByUserIds, + dismissedUserIds, + wtfImpressions, + locationInfo, + topicIds, + userState)) => + PostNuxMlRequest( + params = req.params, + clientContext = req.recommendationRequest.clientContext, + similarToUserIds = Nil, + inputExcludeUserIds = req.recommendationRequest.excludedIds.getOrElse(Nil), + recentFollowedUserIds = Some(recentFollowedUserIds), + invalidRelationshipUserIds = Some(invalidRelationshipUserIds.toSet), + recentFollowedByUserIds = Some(recentFollowedByUserIds), + dismissedUserIds = Some(dismissedUserIds), + displayLocation = dl, + maxResults = req.recommendationRequest.maxResults, + debugOptions = req.recommendationRequest.debugParams.flatMap(_.debugOptions), + wtfImpressions = Some(wtfImpressions), + geohashAndCountryCode = locationInfo, + uttInterestIds = Some(topicIds), + inputPreviouslyRecommendedUserIds = previouslyRecommendedUserIds, + inputPreviouslyFollowedUserIds = previouslyFollowedUserIds, + isSoftUser = req.recommendationRequest.isSoftUser, + userState = userState + ) + case _ => + PostNuxMlRequest( + params = req.params, + clientContext = req.recommendationRequest.clientContext, + similarToUserIds = Nil, + inputExcludeUserIds = req.recommendationRequest.excludedIds.getOrElse(Nil), + recentFollowedUserIds = None, + invalidRelationshipUserIds = None, + recentFollowedByUserIds = None, + dismissedUserIds = None, + displayLocation = dl, + maxResults = req.recommendationRequest.maxResults, + debugOptions = req.recommendationRequest.debugParams.flatMap(_.debugOptions), + wtfImpressions = None, + geohashAndCountryCode = None, + inputPreviouslyRecommendedUserIds = previouslyRecommendedUserIds, + inputPreviouslyFollowedUserIds = previouslyFollowedUserIds, + isSoftUser = req.recommendationRequest.isSoftUser, + userState = None + ) + } + } +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/post_nux_ml/PostNuxMlRequestBuilderParams.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/post_nux_ml/PostNuxMlRequestBuilderParams.scala new file mode 100644 index 0000000000..da60f0382d --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/post_nux_ml/PostNuxMlRequestBuilderParams.scala @@ -0,0 +1,45 @@ +package com.twitter.follow_recommendations.flows.post_nux_ml + +import com.twitter.timelines.configapi.FSBoundedParam +import com.twitter.util.Duration +import com.twitter.conversions.DurationOps._ +import com.twitter.timelines.configapi.DurationConversion +import com.twitter.timelines.configapi.FSParam +import com.twitter.timelines.configapi.HasDurationConversion + +object PostNuxMlRequestBuilderParams { + case object TopicIdFetchBudget + extends FSBoundedParam[Duration]( + name = "post_nux_ml_request_builder_topic_id_fetch_budget_millis", + default = 200.millisecond, + min = 80.millisecond, + max = 400.millisecond) + with HasDurationConversion { + override val durationConversion: DurationConversion = DurationConversion.FromMillis + } + + case object DismissedIdScanBudget + extends FSBoundedParam[Duration]( + name = "post_nux_ml_request_builder_dismissed_id_scan_budget_millis", + default = 200.millisecond, + min = 80.millisecond, + max = 400.millisecond) + with HasDurationConversion { + override val durationConversion: DurationConversion = DurationConversion.FromMillis + } + + case object WTFImpressionsScanBudget + extends FSBoundedParam[Duration]( + name = "post_nux_ml_request_builder_wtf_impressions_scan_budget_millis", + default = 200.millisecond, + min = 80.millisecond, + max = 400.millisecond) + with HasDurationConversion { + override val durationConversion: DurationConversion = DurationConversion.FromMillis + } + + case object EnableInvalidRelationshipPredicate + extends FSParam[Boolean]( + name = "post_nux_ml_request_builder_enable_invalid_relationship_predicate", + false) +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/logging/BUILD b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/logging/BUILD new file mode 100644 index 0000000000..a35992e93a --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/logging/BUILD @@ -0,0 +1,18 @@ +scala_library( + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/com/google/inject:guice", + "3rdparty/jvm/com/google/inject/extensions:guice-assistedinject", + "3rdparty/jvm/net/codingwell:scala-guice", + "3rdparty/jvm/org/slf4j:slf4j-api", + "finatra/inject/inject-core/src/main/scala", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/constants", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/params", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/models", + "follow-recommendations-service/thrift/src/main/thrift:thrift-scala", + "scribelib/marshallers/src/main/scala/com/twitter/scribelib/marshallers", + "util/util-slf4j-api/src/main/scala", + ], +) diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/logging/FrsLogger.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/logging/FrsLogger.scala new file mode 100644 index 0000000000..8b920c5562 --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/logging/FrsLogger.scala @@ -0,0 +1,164 @@ +package com.twitter.follow_recommendations.logging + +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.follow_recommendations.common.constants.GuiceNamedConstants +import com.twitter.follow_recommendations.common.models.HasIsSoftUser +import com.twitter.follow_recommendations.configapi.params.GlobalParams +import com.twitter.follow_recommendations.logging.thriftscala.RecommendationLog +import com.twitter.follow_recommendations.models.DebugParams +import com.twitter.follow_recommendations.models.RecommendationFlowData +import com.twitter.follow_recommendations.models.RecommendationRequest +import com.twitter.follow_recommendations.models.RecommendationResponse +import com.twitter.follow_recommendations.models.ScoringUserRequest +import com.twitter.follow_recommendations.models.ScoringUserResponse +import com.twitter.inject.annotations.Flag +import com.twitter.logging.LoggerFactory +import com.twitter.product_mixer.core.model.marshalling.request.ClientContext +import com.twitter.product_mixer.core.model.marshalling.request.HasClientContext +import com.twitter.scribelib.marshallers.ClientDataProvider +import com.twitter.scribelib.marshallers.ExternalRefererDataProvider +import com.twitter.scribelib.marshallers.ScribeSerialization +import com.twitter.timelines.configapi.HasParams +import com.twitter.util.Time +import javax.inject.Inject +import javax.inject.Named +import javax.inject.Singleton + +/** + * This is the standard logging class we use to log data into: + * 1) logs.follow_recommendations_logs + * + * This logger logs data for 2 endpoints: getRecommendations, scoreUserCandidates + * All data scribed via this logger have to be converted into the same thrift type: RecommendationLog + * + * 2) logs.frs_recommendation_flow_logs + * + * This logger logs recommendation flow data for getRecommendations requests + * All data scribed via this logger have to be converted into the same thrift type: FrsRecommendationFlowLog + */ +@Singleton +class FrsLogger @Inject() ( + @Named(GuiceNamedConstants.REQUEST_LOGGER) loggerFactory: LoggerFactory, + @Named(GuiceNamedConstants.FLOW_LOGGER) flowLoggerFactory: LoggerFactory, + stats: StatsReceiver, + @Flag("log_results") serviceShouldLogResults: Boolean) + extends ScribeSerialization { + private val logger = loggerFactory.apply() + private val flowLogger = flowLoggerFactory.apply() + private val logRecommendationCounter = stats.counter("scribe_recommendation") + private val logScoringCounter = stats.counter("scribe_scoring") + private val logRecommendationFlowCounter = stats.counter("scribe_recommendation_flow") + + def logRecommendationResult( + request: RecommendationRequest, + response: RecommendationResponse + ): Unit = { + if (!request.isSoftUser) { + val log = + RecommendationLog(request.toOfflineThrift, response.toOfflineThrift, Time.now.inMillis) + logRecommendationCounter.incr() + logger.info( + serializeThrift( + log, + FrsLogger.LogCategory, + FrsLogger.mkProvider(request.clientContext) + )) + } + } + + def logScoringResult(request: ScoringUserRequest, response: ScoringUserResponse): Unit = { + if (!request.isSoftUser) { + val log = + RecommendationLog( + request.toRecommendationRequest.toOfflineThrift, + response.toRecommendationResponse.toOfflineThrift, + Time.now.inMillis) + logScoringCounter.incr() + logger.info( + serializeThrift( + log, + FrsLogger.LogCategory, + FrsLogger.mkProvider(request.toRecommendationRequest.clientContext) + )) + } + } + + def logRecommendationFlowData[Target <: HasClientContext with HasIsSoftUser with HasParams]( + request: Target, + flowData: RecommendationFlowData[Target] + ): Unit = { + if (!request.isSoftUser && request.params(GlobalParams.EnableRecommendationFlowLogs)) { + val log = flowData.toRecommendationFlowLogOfflineThrift + logRecommendationFlowCounter.incr() + flowLogger.info( + serializeThrift( + log, + FrsLogger.FlowLogCategory, + FrsLogger.mkProvider(request.clientContext) + )) + } + } + + // We prefer the settings given in the user request, and if none provided we default to the + // aurora service configuration. + def shouldLog(debugParamsOpt: Option[DebugParams]): Boolean = + debugParamsOpt match { + case Some(debugParams) => + debugParams.debugOptions match { + case Some(debugOptions) => + !debugOptions.doNotLog + case None => + serviceShouldLogResults + } + case None => + serviceShouldLogResults + } + +} + +object FrsLogger { + val LogCategory = "follow_recommendations_logs" + val FlowLogCategory = "frs_recommendation_flow_logs" + + def mkProvider(clientContext: ClientContext) = new ClientDataProvider { + + /** The id of the current user. When the user is logged out, this method should return None. */ + override val userId: Option[Long] = clientContext.userId + + /** The id of the guest, which is present in logged-in or loged-out states */ + override val guestId: Option[Long] = clientContext.guestId + + /** The personalization id (pid) of the user, used to personalize Twitter services */ + override val personalizationId: Option[String] = None + + /** The id of the individual device the user is currently using. This id will be unique for different users' devices. */ + override val deviceId: Option[String] = clientContext.deviceId + + /** The OAuth application id of the application the user is currently using */ + override val clientApplicationId: Option[Long] = clientContext.appId + + /** The OAuth parent application id of the application the user is currently using */ + override val parentApplicationId: Option[Long] = None + + /** The two-letter, upper-case country code used to designate the country from which the scribe event occurred */ + override val countryCode: Option[String] = clientContext.countryCode + + /** The two-letter, lower-case language code used to designate the probably language spoken by the scribe event initiator */ + override val languageCode: Option[String] = clientContext.languageCode + + /** The user-agent header used to identify the client browser or device that the user is currently active on */ + override val userAgent: Option[String] = clientContext.userAgent + + /** Whether the user is accessing Twitter via a secured connection */ + override val isSsl: Option[Boolean] = Some(true) + + /** The referring URL to the current page for web-based clients, if applicable */ + override val referer: Option[String] = None + + /** + * The external site, partner, or email that lead to the current Twitter application. Returned value consists of a + * tuple including the encrypted referral data and the type of referral + */ + override val externalReferer: Option[ExternalRefererDataProvider] = None + } +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/models/BUILD b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/models/BUILD new file mode 100644 index 0000000000..597ab76c40 --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/models/BUILD @@ -0,0 +1,13 @@ +scala_library( + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "configapi/configapi-core/src/main/scala/com/twitter/timelines/configapi", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/common", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/assembler/models", + "follow-recommendations-service/thrift/src/main/thrift:thrift-scala", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/request", + ], +) diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/models/CandidateSourceType.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/models/CandidateSourceType.scala new file mode 100644 index 0000000000..38215c44b2 --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/models/CandidateSourceType.scala @@ -0,0 +1,9 @@ +package com.twitter.follow_recommendations.models + +object CandidateSourceType extends Enumeration { + type CandidateSourceType = Value + val Social = Value("social") + val GeoAndInterests = Value("geo_and_interests") + val ActivityContextual = Value("activity_contextual") + val None = Value("none") +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/models/CandidateUserDebugParams.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/models/CandidateUserDebugParams.scala new file mode 100644 index 0000000000..a5702b2b3f --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/models/CandidateUserDebugParams.scala @@ -0,0 +1,5 @@ +package com.twitter.follow_recommendations.models + +import com.twitter.timelines.configapi.Params + +case class CandidateUserDebugParams(paramsMap: Map[Long, Params]) diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/models/DebugParams.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/models/DebugParams.scala new file mode 100644 index 0000000000..dee7f9b6a6 --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/models/DebugParams.scala @@ -0,0 +1,28 @@ +package com.twitter.follow_recommendations.models + +import com.twitter.follow_recommendations.common.models.DebugOptions +import com.twitter.follow_recommendations.common.models.DebugOptions.fromDebugParamsThrift +import com.twitter.follow_recommendations.logging.{thriftscala => offline} +import com.twitter.follow_recommendations.{thriftscala => t} +import com.twitter.timelines.configapi.{FeatureValue => ConfigApiFeatureValue} + +case class DebugParams( + featureOverrides: Option[Map[String, ConfigApiFeatureValue]], + debugOptions: Option[DebugOptions]) + +object DebugParams { + def fromThrift(thrift: t.DebugParams): DebugParams = DebugParams( + featureOverrides = thrift.featureOverrides.map { map => + map.mapValues(FeatureValue.fromThrift).toMap + }, + debugOptions = Some( + fromDebugParamsThrift(thrift) + ) + ) + def toOfflineThrift(model: DebugParams): offline.OfflineDebugParams = + offline.OfflineDebugParams(randomizationSeed = model.debugOptions.flatMap(_.randomizationSeed)) +} + +trait HasFrsDebugParams { + def frsDebugParams: Option[DebugParams] +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/models/DisplayContext.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/models/DisplayContext.scala new file mode 100644 index 0000000000..59f0adfd7f --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/models/DisplayContext.scala @@ -0,0 +1,113 @@ +package com.twitter.follow_recommendations.models + +import com.twitter.follow_recommendations.common.models.FlowContext +import com.twitter.follow_recommendations.common.models.RecentlyEngagedUserId +import com.twitter.follow_recommendations.logging.thriftscala.OfflineDisplayContext +import com.twitter.follow_recommendations.logging.{thriftscala => offline} +import com.twitter.follow_recommendations.{thriftscala => t} +import scala.reflect.ClassTag +import scala.reflect.classTag + +trait DisplayContext { + def toOfflineThrift: offline.OfflineDisplayContext +} + +object DisplayContext { + case class Profile(profileId: Long) extends DisplayContext { + override val toOfflineThrift: OfflineDisplayContext = + offline.OfflineDisplayContext.Profile(offline.OfflineProfile(profileId)) + } + case class Search(searchQuery: String) extends DisplayContext { + override val toOfflineThrift: OfflineDisplayContext = + offline.OfflineDisplayContext.Search(offline.OfflineSearch(searchQuery)) + } + case class Rux(focalAuthorId: Long) extends DisplayContext { + override val toOfflineThrift: OfflineDisplayContext = + offline.OfflineDisplayContext.Rux(offline.OfflineRux(focalAuthorId)) + } + + case class Topic(topicId: Long) extends DisplayContext { + override val toOfflineThrift: OfflineDisplayContext = + offline.OfflineDisplayContext.Topic(offline.OfflineTopic(topicId)) + } + + case class ReactiveFollow(followedUserIds: Seq[Long]) extends DisplayContext { + override val toOfflineThrift: OfflineDisplayContext = + offline.OfflineDisplayContext.ReactiveFollow(offline.OfflineReactiveFollow(followedUserIds)) + } + + case class NuxInterests(flowContext: Option[FlowContext], uttInterestIds: Option[Seq[Long]]) + extends DisplayContext { + override val toOfflineThrift: OfflineDisplayContext = + offline.OfflineDisplayContext.NuxInterests( + offline.OfflineNuxInterests(flowContext.map(_.toOfflineThrift))) + } + + case class PostNuxFollowTask(flowContext: Option[FlowContext]) extends DisplayContext { + override val toOfflineThrift: OfflineDisplayContext = + offline.OfflineDisplayContext.PostNuxFollowTask( + offline.OfflinePostNuxFollowTask(flowContext.map(_.toOfflineThrift))) + } + + case class AdCampaignTarget(similarToUserIds: Seq[Long]) extends DisplayContext { + override val toOfflineThrift: OfflineDisplayContext = + offline.OfflineDisplayContext.AdCampaignTarget( + offline.OfflineAdCampaignTarget(similarToUserIds)) + } + + case class ConnectTab( + byfSeedUserIds: Seq[Long], + similarToUserIds: Seq[Long], + engagedUserIds: Seq[RecentlyEngagedUserId]) + extends DisplayContext { + override val toOfflineThrift: OfflineDisplayContext = + offline.OfflineDisplayContext.ConnectTab( + offline.OfflineConnectTab( + byfSeedUserIds, + similarToUserIds, + engagedUserIds.map(user => user.toOfflineThrift))) + } + + case class SimilarToUser(similarToUserId: Long) extends DisplayContext { + override val toOfflineThrift: OfflineDisplayContext = + offline.OfflineDisplayContext.SimilarToUser(offline.OfflineSimilarToUser(similarToUserId)) + } + + def fromThrift(tDisplayContext: t.DisplayContext): DisplayContext = tDisplayContext match { + case t.DisplayContext.Profile(p) => Profile(p.profileId) + case t.DisplayContext.Search(s) => Search(s.searchQuery) + case t.DisplayContext.Rux(r) => Rux(r.focalAuthorId) + case t.DisplayContext.Topic(t) => Topic(t.topicId) + case t.DisplayContext.ReactiveFollow(f) => ReactiveFollow(f.followedUserIds) + case t.DisplayContext.NuxInterests(n) => + NuxInterests(n.flowContext.map(FlowContext.fromThrift), n.uttInterestIds) + case t.DisplayContext.AdCampaignTarget(a) => + AdCampaignTarget(a.similarToUserIds) + case t.DisplayContext.ConnectTab(connect) => + ConnectTab( + connect.byfSeedUserIds, + connect.similarToUserIds, + connect.recentlyEngagedUserIds.map(RecentlyEngagedUserId.fromThrift)) + case t.DisplayContext.SimilarToUser(r) => + SimilarToUser(r.similarToUserId) + case t.DisplayContext.PostNuxFollowTask(p) => + PostNuxFollowTask(p.flowContext.map(FlowContext.fromThrift)) + case t.DisplayContext.UnknownUnionField(t) => + throw new UnknownDisplayContextException(t.field.name) + } + + def getDisplayContextAs[T <: DisplayContext: ClassTag](displayContext: DisplayContext): T = + displayContext match { + case context: T => context + case _ => + throw new UnexpectedDisplayContextTypeException( + displayContext, + classTag[T].getClass.getSimpleName) + } +} + +class UnknownDisplayContextException(name: String) + extends Exception(s"Unknown DisplayContext in Thrift: ${name}") + +class UnexpectedDisplayContextTypeException(displayContext: DisplayContext, expectedType: String) + extends Exception(s"DisplayContext ${displayContext} not of expected type ${expectedType}") diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/models/FeatureValue.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/models/FeatureValue.scala new file mode 100644 index 0000000000..66f0afafad --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/models/FeatureValue.scala @@ -0,0 +1,24 @@ +package com.twitter.follow_recommendations.models + +import com.twitter.follow_recommendations.{thriftscala => t} +import com.twitter.timelines.configapi._ + +object FeatureValue { + def fromThrift(thriftFeatureValue: t.FeatureValue): FeatureValue = thriftFeatureValue match { + case t.FeatureValue.PrimitiveValue(t.PrimitiveFeatureValue.BoolValue(bool)) => + BooleanFeatureValue(bool) + case t.FeatureValue.PrimitiveValue(t.PrimitiveFeatureValue.StrValue(string)) => + StringFeatureValue(string) + case t.FeatureValue.PrimitiveValue(t.PrimitiveFeatureValue.IntValue(int)) => + NumberFeatureValue(int) + case t.FeatureValue.PrimitiveValue(t.PrimitiveFeatureValue.LongValue(long)) => + NumberFeatureValue(long) + case t.FeatureValue.PrimitiveValue(t.PrimitiveFeatureValue.UnknownUnionField(field)) => + throw new UnknownFeatureValueException(s"Primitive: ${field.field.name}") + case t.FeatureValue.UnknownUnionField(field) => + throw new UnknownFeatureValueException(field.field.name) + } +} + +class UnknownFeatureValueException(fieldName: String) + extends Exception(s"Unknown FeatureValue name in thrift: ${fieldName}") diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/models/RecommendationFlowData.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/models/RecommendationFlowData.scala new file mode 100644 index 0000000000..06b19ac46c --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/models/RecommendationFlowData.scala @@ -0,0 +1,104 @@ +package com.twitter.follow_recommendations.models + +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.follow_recommendations.common.models.ClientContextConverter +import com.twitter.follow_recommendations.common.models.HasUserState +import com.twitter.follow_recommendations.common.utils.UserSignupUtil +import com.twitter.follow_recommendations.logging.{thriftscala => offline} +import com.twitter.product_mixer.core.functional_component.candidate_source.CandidateSource +import com.twitter.product_mixer.core.model.common.identifier.CandidateSourceIdentifier +import com.twitter.product_mixer.core.model.common.identifier.RecommendationPipelineIdentifier +import com.twitter.product_mixer.core.model.marshalling.HasMarshalling +import com.twitter.product_mixer.core.model.marshalling.request.HasClientContext +import com.twitter.util.Time + +case class RecommendationFlowData[Target <: HasClientContext]( + request: Target, + recommendationFlowIdentifier: RecommendationPipelineIdentifier, + candidateSources: Seq[CandidateSource[Target, CandidateUser]], + candidatesFromCandidateSources: Seq[CandidateUser], + mergedCandidates: Seq[CandidateUser], + filteredCandidates: Seq[CandidateUser], + rankedCandidates: Seq[CandidateUser], + transformedCandidates: Seq[CandidateUser], + truncatedCandidates: Seq[CandidateUser], + results: Seq[CandidateUser]) + extends HasMarshalling { + + import RecommendationFlowData._ + + lazy val toRecommendationFlowLogOfflineThrift: offline.RecommendationFlowLog = { + val userMetadata = userToOfflineRecommendationFlowUserMetadata(request) + val signals = userToOfflineRecommendationFlowSignals(request) + val filteredCandidateSourceCandidates = + candidatesToOfflineRecommendationFlowCandidateSourceCandidates( + candidateSources, + filteredCandidates + ) + val rankedCandidateSourceCandidates = + candidatesToOfflineRecommendationFlowCandidateSourceCandidates( + candidateSources, + rankedCandidates + ) + val truncatedCandidateSourceCandidates = + candidatesToOfflineRecommendationFlowCandidateSourceCandidates( + candidateSources, + truncatedCandidates + ) + + offline.RecommendationFlowLog( + ClientContextConverter.toFRSOfflineClientContextThrift(request.clientContext), + userMetadata, + signals, + Time.now.inMillis, + recommendationFlowIdentifier.name, + Some(filteredCandidateSourceCandidates), + Some(rankedCandidateSourceCandidates), + Some(truncatedCandidateSourceCandidates) + ) + } +} + +object RecommendationFlowData { + def userToOfflineRecommendationFlowUserMetadata[Target <: HasClientContext]( + request: Target + ): Option[offline.OfflineRecommendationFlowUserMetadata] = { + val userSignupAge = UserSignupUtil.userSignupAge(request).map(_.inDays) + val userState = request match { + case req: HasUserState => req.userState.map(_.name) + case _ => None + } + Some(offline.OfflineRecommendationFlowUserMetadata(userSignupAge, userState)) + } + + def userToOfflineRecommendationFlowSignals[Target <: HasClientContext]( + request: Target + ): Option[offline.OfflineRecommendationFlowSignals] = { + val countryCode = request.getCountryCode + Some(offline.OfflineRecommendationFlowSignals(countryCode)) + } + + def candidatesToOfflineRecommendationFlowCandidateSourceCandidates[Target <: HasClientContext]( + candidateSources: Seq[CandidateSource[Target, CandidateUser]], + candidates: Seq[CandidateUser], + ): Seq[offline.OfflineRecommendationFlowCandidateSourceCandidates] = { + val candidatesGroupedByCandidateSources = + candidates.groupBy( + _.getPrimaryCandidateSource.getOrElse(CandidateSourceIdentifier("NoCandidateSource"))) + + candidateSources.map(candidateSource => { + val candidates = + candidatesGroupedByCandidateSources.get(candidateSource.identifier).toSeq.flatten + val candidateUserIds = candidates.map(_.id) + val candidateUserScores = candidates.map(_.score).exists(_.nonEmpty) match { + case true => Some(candidates.map(_.score.getOrElse(-1.0))) + case false => None + } + offline.OfflineRecommendationFlowCandidateSourceCandidates( + candidateSource.identifier.name, + candidateUserIds, + candidateUserScores + ) + }) + } +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/models/RecommendationRequest.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/models/RecommendationRequest.scala new file mode 100644 index 0000000000..fa768b5368 --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/models/RecommendationRequest.scala @@ -0,0 +1,29 @@ +package com.twitter.follow_recommendations.models + +import com.twitter.follow_recommendations.common.models.ClientContextConverter +import com.twitter.follow_recommendations.common.models.DisplayLocation +import com.twitter.follow_recommendations.logging.{thriftscala => offline} +import com.twitter.product_mixer.core.model.marshalling.request.ClientContext + +case class RecommendationRequest( + clientContext: ClientContext, + displayLocation: DisplayLocation, + displayContext: Option[DisplayContext], + maxResults: Option[Int], + cursor: Option[String], + excludedIds: Option[Seq[Long]], + fetchPromotedContent: Option[Boolean], + debugParams: Option[DebugParams] = None, + userLocationState: Option[String] = None, + isSoftUser: Boolean = false) { + def toOfflineThrift: offline.OfflineRecommendationRequest = offline.OfflineRecommendationRequest( + ClientContextConverter.toFRSOfflineClientContextThrift(clientContext), + displayLocation.toOfflineThrift, + displayContext.map(_.toOfflineThrift), + maxResults, + cursor, + excludedIds, + fetchPromotedContent, + debugParams.map(DebugParams.toOfflineThrift) + ) +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/models/RecommendationResponse.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/models/RecommendationResponse.scala new file mode 100644 index 0000000000..fadff377b2 --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/models/RecommendationResponse.scala @@ -0,0 +1,14 @@ +package com.twitter.follow_recommendations.models + +import com.twitter.follow_recommendations.{thriftscala => t} +import com.twitter.follow_recommendations.logging.{thriftscala => offline} +import com.twitter.follow_recommendations.common.models.Recommendation +import com.twitter.product_mixer.core.model.marshalling.HasMarshalling + +case class RecommendationResponse(recommendations: Seq[Recommendation]) extends HasMarshalling { + lazy val toThrift: t.RecommendationResponse = + t.RecommendationResponse(recommendations.map(_.toThrift)) + + lazy val toOfflineThrift: offline.OfflineRecommendationResponse = + offline.OfflineRecommendationResponse(recommendations.map(_.toOfflineThrift)) +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/models/Request.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/models/Request.scala new file mode 100644 index 0000000000..a8798bda2a --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/models/Request.scala @@ -0,0 +1,22 @@ +package com.twitter.follow_recommendations.models + +import com.twitter.follow_recommendations.common.models.DisplayLocation +import com.twitter.product_mixer.core.model.marshalling.request +import com.twitter.product_mixer.core.model.marshalling.request.ClientContext +import com.twitter.product_mixer.core.model.marshalling.request.ProductContext +import com.twitter.product_mixer.core.model.marshalling.request.{Request => ProductMixerRequest} + +case class Request( + override val maxResults: Option[Int], + override val debugParams: Option[request.DebugParams], + override val productContext: Option[ProductContext], + override val product: request.Product, + override val clientContext: ClientContext, + override val serializedRequestCursor: Option[String], + override val frsDebugParams: Option[DebugParams], + displayLocation: DisplayLocation, + excludedIds: Option[Seq[Long]], + fetchPromotedContent: Option[Boolean], + userLocationState: Option[String] = None) + extends ProductMixerRequest + with HasFrsDebugParams {} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/models/ScoringUserRequest.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/models/ScoringUserRequest.scala new file mode 100644 index 0000000000..84d9d3ee33 --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/models/ScoringUserRequest.scala @@ -0,0 +1,45 @@ +package com.twitter.follow_recommendations.models + +import com.twitter.follow_recommendations.common.feature_hydration.common.HasPreFetchedFeature +import com.twitter.follow_recommendations.common.models._ +import com.twitter.follow_recommendations.logging.{thriftscala => offline} +import com.twitter.product_mixer.core.model.marshalling.request.ClientContext +import com.twitter.product_mixer.core.model.marshalling.request.HasClientContext +import com.twitter.timelines.configapi.HasParams +import com.twitter.timelines.configapi.Params + +case class ScoringUserRequest( + override val clientContext: ClientContext, + override val displayLocation: DisplayLocation, + override val params: Params, + override val debugOptions: Option[DebugOptions] = None, + override val recentFollowedUserIds: Option[Seq[Long]], + override val recentFollowedByUserIds: Option[Seq[Long]], + override val wtfImpressions: Option[Seq[WtfImpression]], + override val similarToUserIds: Seq[Long], + candidates: Seq[CandidateUser], + debugParams: Option[DebugParams] = None, + isSoftUser: Boolean = false) + extends HasClientContext + with HasDisplayLocation + with HasParams + with HasDebugOptions + with HasPreFetchedFeature + with HasSimilarToContext { + def toOfflineThrift: offline.OfflineScoringUserRequest = offline.OfflineScoringUserRequest( + ClientContextConverter.toFRSOfflineClientContextThrift(clientContext), + displayLocation.toOfflineThrift, + candidates.map(_.toOfflineUserThrift) + ) + def toRecommendationRequest: RecommendationRequest = RecommendationRequest( + clientContext = clientContext, + displayLocation = displayLocation, + displayContext = None, + maxResults = None, + cursor = None, + excludedIds = None, + fetchPromotedContent = None, + debugParams = debugParams, + isSoftUser = isSoftUser + ) +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/models/ScoringUserResponse.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/models/ScoringUserResponse.scala new file mode 100644 index 0000000000..4611386d3f --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/models/ScoringUserResponse.scala @@ -0,0 +1,15 @@ +package com.twitter.follow_recommendations.models + +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.follow_recommendations.logging.{thriftscala => offline} +import com.twitter.follow_recommendations.{thriftscala => t} + +case class ScoringUserResponse(candidates: Seq[CandidateUser]) { + lazy val toThrift: t.ScoringUserResponse = + t.ScoringUserResponse(candidates.map(_.toUserThrift)) + + lazy val toRecommendationResponse: RecommendationResponse = RecommendationResponse(candidates) + + lazy val toOfflineThrift: offline.OfflineScoringUserResponse = + offline.OfflineScoringUserResponse(candidates.map(_.toOfflineUserThrift)) +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/models/failures/BUILD b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/models/failures/BUILD new file mode 100644 index 0000000000..4874d636ec --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/models/failures/BUILD @@ -0,0 +1,8 @@ +scala_library( + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/pipeline_failure", + ], +) diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/models/failures/TimeoutPipelineFailure.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/models/failures/TimeoutPipelineFailure.scala new file mode 100644 index 0000000000..023b4c63ea --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/models/failures/TimeoutPipelineFailure.scala @@ -0,0 +1,12 @@ +package com.twitter.follow_recommendations.models.failures + +import com.twitter.product_mixer.core.pipeline.pipeline_failure.CandidateSourceTimeout +import com.twitter.product_mixer.core.pipeline.pipeline_failure.PipelineFailure + +object TimeoutPipelineFailure { + def apply(candidateSourceName: String): PipelineFailure = { + PipelineFailure( + CandidateSourceTimeout, + s"Candidate Source $candidateSourceName timed out before returning candidates") + } +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/modules/ABDeciderModule.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/modules/ABDeciderModule.scala new file mode 100644 index 0000000000..b75b6753e3 --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/modules/ABDeciderModule.scala @@ -0,0 +1,31 @@ +package com.twitter.follow_recommendations.modules + +import com.google.inject.Provides +import com.google.inject.name.Named +import com.twitter.abdecider.ABDeciderFactory +import com.twitter.abdecider.LoggingABDecider +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.follow_recommendations.common.constants.GuiceNamedConstants +import com.twitter.inject.TwitterModule +import com.twitter.logging.LoggerFactory +import javax.inject.Singleton + +object ABDeciderModule extends TwitterModule { + @Provides + @Singleton + def provideABDecider( + stats: StatsReceiver, + @Named(GuiceNamedConstants.CLIENT_EVENT_LOGGER) factory: LoggerFactory + ): LoggingABDecider = { + + val ymlPath = "/usr/local/config/abdecider/abdecider.yml" + + val abDeciderFactory = ABDeciderFactory( + abDeciderYmlPath = ymlPath, + scribeLogger = Some(factory()), + environment = Some("production") + ) + + abDeciderFactory.buildWithLogging() + } +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/modules/BUILD b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/modules/BUILD new file mode 100644 index 0000000000..e7fb683805 --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/modules/BUILD @@ -0,0 +1,24 @@ +scala_library( + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/com/google/inject:guice", + "3rdparty/jvm/com/google/inject/extensions:guice-assistedinject", + "3rdparty/jvm/javax/inject:javax.inject", + "3rdparty/jvm/net/codingwell:scala-guice", + "3rdparty/jvm/org/slf4j:slf4j-api", + "finatra-internal/mtls-thriftmux/src/main/scala", + "finatra/inject/inject-core/src/main/scala", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/constants", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/transforms/modify_social_proof", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products", + "follow-recommendations-service/thrift/src/main/thrift:thrift-scala", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/product/guice", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/product/registry", + "twml/runtime/src/main/scala/com/twitter/deepbird/runtime/prediction_engine", + "util/util-slf4j-api/src/main/scala", + ], +) diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/modules/ConfigApiModule.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/modules/ConfigApiModule.scala new file mode 100644 index 0000000000..ef3865bf28 --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/modules/ConfigApiModule.scala @@ -0,0 +1,20 @@ +package com.twitter.follow_recommendations.modules + +import com.google.inject.Provides +import com.twitter.decider.Decider +import com.twitter.follow_recommendations.configapi.ConfigBuilder +import com.twitter.inject.TwitterModule +import com.twitter.servo.decider.DeciderGateBuilder +import com.twitter.timelines.configapi.Config +import javax.inject.Singleton + +object ConfigApiModule extends TwitterModule { + @Provides + @Singleton + def providesDeciderGateBuilder(decider: Decider): DeciderGateBuilder = + new DeciderGateBuilder(decider) + + @Provides + @Singleton + def providesConfig(configBuilder: ConfigBuilder): Config = configBuilder.build() +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/modules/DiffyModule.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/modules/DiffyModule.scala new file mode 100644 index 0000000000..4ab0e4eba2 --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/modules/DiffyModule.scala @@ -0,0 +1,71 @@ +package com.twitter.follow_recommendations.modules + +import com.google.inject.Provides +import com.google.inject.Singleton +import com.twitter.inject.annotations.Flag +import com.twitter.decider.RandomRecipient +import com.twitter.finagle.ThriftMux +import com.twitter.finagle.mtls.authentication.ServiceIdentifier +import com.twitter.finagle.mtls.client.MtlsStackClient.MtlsThriftMuxClientSyntax +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.finagle.thrift.ClientId +import com.twitter.finatra.annotations.DarkTrafficService +import com.twitter.follow_recommendations.configapi.deciders.DeciderKey +import com.twitter.follow_recommendations.thriftscala.FollowRecommendationsThriftService +import com.twitter.inject.TwitterModule +import com.twitter.inject.thrift.filters.DarkTrafficFilter +import com.twitter.servo.decider.DeciderGateBuilder + +object DiffyModule extends TwitterModule { + // diffy.dest is defined in the Follow Recommendations Service aurora file + // and points to the Dark Traffic Proxy server + private val destFlag = + flag[String]("diffy.dest", "/$/nil", "Resolvable name of diffy-service or proxy") + + @Provides + @Singleton + @DarkTrafficService + def provideDarkTrafficService( + serviceIdentifier: ServiceIdentifier + ): FollowRecommendationsThriftService.ReqRepServicePerEndpoint = { + ThriftMux.client + .withClientId(ClientId("follow_recos_service_darktraffic_proxy_client")) + .withMutualTls(serviceIdentifier) + .servicePerEndpoint[FollowRecommendationsThriftService.ReqRepServicePerEndpoint]( + dest = destFlag(), + label = "darktrafficproxy" + ) + } + + @Provides + @Singleton + def provideDarkTrafficFilter( + @DarkTrafficService darkService: FollowRecommendationsThriftService.ReqRepServicePerEndpoint, + deciderGateBuilder: DeciderGateBuilder, + statsReceiver: StatsReceiver, + @Flag("environment") env: String + ): DarkTrafficFilter[FollowRecommendationsThriftService.ReqRepServicePerEndpoint] = { + // sampleFunction is used to determine which requests should get replicated + // to the dark traffic proxy server + val sampleFunction: Any => Boolean = { _ => + // check whether the current FRS instance is deployed in production + env match { + case "prod" => + statsReceiver.scope("provideDarkTrafficFilter").counter("prod").incr() + destFlag.isDefined && deciderGateBuilder + .keyToFeature(DeciderKey.EnableTrafficDarkReading).isAvailable(RandomRecipient) + case _ => + statsReceiver.scope("provideDarkTrafficFilter").counter("devel").incr() + // replicate zero requests if in non-production environment + false + } + } + new DarkTrafficFilter[FollowRecommendationsThriftService.ReqRepServicePerEndpoint]( + darkService, + sampleFunction, + forwardAfterService = true, + statsReceiver.scope("DarkTrafficFilter"), + lookupByMethod = true + ) + } +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/modules/FeatureSwitchesModule.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/modules/FeatureSwitchesModule.scala new file mode 100644 index 0000000000..1600344b62 --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/modules/FeatureSwitchesModule.scala @@ -0,0 +1,85 @@ +package com.twitter.follow_recommendations.modules + +import com.google.inject.Provides +import com.twitter.abdecider.LoggingABDecider +import com.twitter.featureswitches.v2.Feature +import com.twitter.featureswitches.v2.FeatureFilter +import com.twitter.featureswitches.v2.FeatureSwitches +import com.twitter.featureswitches.v2.builder.FeatureSwitchesBuilder +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.follow_recommendations.common.constants.GuiceNamedConstants.PRODUCER_SIDE_FEATURE_SWITCHES +import com.twitter.inject.TwitterModule +import javax.inject.Named +import javax.inject.Singleton + +object FeaturesSwitchesModule extends TwitterModule { + private val DefaultConfigRepoPath = "/usr/local/config" + private val FeaturesPath = "/features/onboarding/follow-recommendations-service/main" + val isLocal = flag("configrepo.local", false, "Is the server running locally or in a DC") + val localConfigRepoPath = flag( + "local.configrepo", + System.getProperty("user.home") + "/workspace/config", + "Path to your local config repo" + ) + + @Provides + @Singleton + def providesFeatureSwitches( + abDecider: LoggingABDecider, + statsReceiver: StatsReceiver + ): FeatureSwitches = { + val configRepoPath = if (isLocal()) { + localConfigRepoPath() + } else { + DefaultConfigRepoPath + } + + FeatureSwitchesBuilder + .createDefault(FeaturesPath, abDecider, Some(statsReceiver)) + .configRepoAbsPath(configRepoPath) + .serviceDetailsFromAurora() + .build() + } + + @Provides + @Singleton + @Named(PRODUCER_SIDE_FEATURE_SWITCHES) + def providesProducerFeatureSwitches( + abDecider: LoggingABDecider, + statsReceiver: StatsReceiver + ): FeatureSwitches = { + val configRepoPath = if (isLocal()) { + localConfigRepoPath() + } else { + DefaultConfigRepoPath + } + + /** + * Feature Switches evaluate all tied FS Keys on Params construction time, which is very inefficient + * for producer/candidate side holdbacks because we have 100s of candidates, and 100s of FS which result + * in 10,000 FS evaluations when we want 1 per candidate (100 total), so we create a new FS Client + * which has a [[ProducerFeatureFilter]] set for feature filter to reduce the FS Keys we evaluate. + */ + FeatureSwitchesBuilder + .createDefault(FeaturesPath, abDecider, Some(statsReceiver.scope("producer_side_fs"))) + .configRepoAbsPath(configRepoPath) + .serviceDetailsFromAurora() + .addFeatureFilter(ProducerFeatureFilter) + .build() + } +} + +case object ProducerFeatureFilter extends FeatureFilter { + private val AllowedKeys = Set( + "post_nux_ml_flow_candidate_user_scorer_id", + "frs_receiver_holdback_keep_social_user_candidate", + "frs_receiver_holdback_keep_user_candidate") + + override def filter(feature: Feature): Option[Feature] = { + if (AllowedKeys.exists(feature.parameters.contains)) { + Some(feature) + } else { + None + } + } +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/modules/FlagsModule.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/modules/FlagsModule.scala new file mode 100644 index 0000000000..f8ff5ae94e --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/modules/FlagsModule.scala @@ -0,0 +1,18 @@ +package com.twitter.follow_recommendations.modules +import com.twitter.inject.TwitterModule + +object FlagsModule extends TwitterModule { + flag[Boolean]( + name = "fetch_prod_promoted_accounts", + help = "Whether or not to fetch production promoted accounts (true / false)" + ) + flag[Boolean]( + name = "interests_opt_out_prod_enabled", + help = "Whether to fetch intersts opt out data from the prod strato column or not" + ) + flag[Boolean]( + name = "log_results", + default = false, + help = "Whether to log results such that we use them for scoring or metrics" + ) +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/modules/ProductRegistryModule.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/modules/ProductRegistryModule.scala new file mode 100644 index 0000000000..218f3b9735 --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/modules/ProductRegistryModule.scala @@ -0,0 +1,12 @@ +package com.twitter.follow_recommendations.modules + +import com.twitter.follow_recommendations.products.ProdProductRegistry +import com.twitter.follow_recommendations.products.common.ProductRegistry +import com.twitter.inject.TwitterModule +import javax.inject.Singleton + +object ProductRegistryModule extends TwitterModule { + override protected def configure(): Unit = { + bind[ProductRegistry].to[ProdProductRegistry].in[Singleton] + } +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/modules/ScorerModule.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/modules/ScorerModule.scala new file mode 100644 index 0000000000..035cc04bf2 --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/modules/ScorerModule.scala @@ -0,0 +1,40 @@ +package com.twitter.follow_recommendations.modules + +import com.google.inject.Provides +import com.google.inject.Singleton +import com.twitter.inject.TwitterModule +import com.twitter.relevance.ep_model.common.CommonConstants +import com.twitter.relevance.ep_model.scorer.EPScorer +import com.twitter.relevance.ep_model.scorer.EPScorerBuilder +import java.io.File +import java.io.FileOutputStream +import scala.language.postfixOps + +object ScorerModule extends TwitterModule { + private val STPScorerPath = "/quality/stp_models/20141223" + + private def fileFromResource(resource: String): File = { + val inputStream = getClass.getResourceAsStream(resource) + val file = File.createTempFile(resource, "temp") + val fos = new FileOutputStream(file) + Iterator + .continually(inputStream.read) + .takeWhile(-1 !=) + .foreach(fos.write) + file + } + + @Provides + @Singleton + def provideEpScorer: EPScorer = { + val modelPath = + fileFromResource(STPScorerPath + "/" + CommonConstants.EP_MODEL_FILE_NAME).getAbsolutePath + val trainingConfigPath = + fileFromResource(STPScorerPath + "/" + CommonConstants.TRAINING_CONFIG).getAbsolutePath + val epScorer = new EPScorerBuilder + epScorer + .withModelPath(modelPath) + .withTrainingConfig(trainingConfigPath) + .build() + } +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/modules/ScribeModule.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/modules/ScribeModule.scala new file mode 100644 index 0000000000..35af77c1aa --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/modules/ScribeModule.scala @@ -0,0 +1,95 @@ +package com.twitter.follow_recommendations.modules + +import com.google.inject.Provides +import com.google.inject.Singleton +import com.google.inject.name.Named +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.follow_recommendations.common.constants.GuiceNamedConstants +import com.twitter.inject.TwitterModule +import com.twitter.logging.BareFormatter +import com.twitter.logging.HandlerFactory +import com.twitter.logging.Level +import com.twitter.logging.LoggerFactory +import com.twitter.logging.NullHandler +import com.twitter.logging.QueueingHandler +import com.twitter.logging.ScribeHandler + +object ScribeModule extends TwitterModule { + val useProdLogger = flag( + name = "scribe.use_prod_loggers", + default = false, + help = "whether to use production logging for service" + ) + + @Provides + @Singleton + @Named(GuiceNamedConstants.CLIENT_EVENT_LOGGER) + def provideClientEventsLoggerFactory(stats: StatsReceiver): LoggerFactory = { + val loggerCategory = "client_event" + val clientEventsHandler: HandlerFactory = if (useProdLogger()) { + QueueingHandler( + maxQueueSize = 10000, + handler = ScribeHandler( + category = loggerCategory, + formatter = BareFormatter, + level = Some(Level.INFO), + statsReceiver = stats.scope("client_event_scribe") + ) + ) + } else { () => NullHandler } + LoggerFactory( + node = "abdecider", + level = Some(Level.INFO), + useParents = false, + handlers = clientEventsHandler :: Nil + ) + } + + @Provides + @Singleton + @Named(GuiceNamedConstants.REQUEST_LOGGER) + def provideFollowRecommendationsLoggerFactory(stats: StatsReceiver): LoggerFactory = { + val loggerCategory = "follow_recommendations_logs" + val handlerFactory: HandlerFactory = if (useProdLogger()) { + QueueingHandler( + maxQueueSize = 10000, + handler = ScribeHandler( + category = loggerCategory, + formatter = BareFormatter, + level = Some(Level.INFO), + statsReceiver = stats.scope("follow_recommendations_logs_scribe") + ) + ) + } else { () => NullHandler } + LoggerFactory( + node = loggerCategory, + level = Some(Level.INFO), + useParents = false, + handlers = handlerFactory :: Nil + ) + } + + @Provides + @Singleton + @Named(GuiceNamedConstants.FLOW_LOGGER) + def provideFrsRecommendationFlowLoggerFactory(stats: StatsReceiver): LoggerFactory = { + val loggerCategory = "frs_recommendation_flow_logs" + val handlerFactory: HandlerFactory = if (useProdLogger()) { + QueueingHandler( + maxQueueSize = 10000, + handler = ScribeHandler( + category = loggerCategory, + formatter = BareFormatter, + level = Some(Level.INFO), + statsReceiver = stats.scope("frs_recommendation_flow_logs_scribe") + ) + ) + } else { () => NullHandler } + LoggerFactory( + node = loggerCategory, + level = Some(Level.INFO), + useParents = false, + handlers = handlerFactory :: Nil + ) + } +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/modules/TimerModule.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/modules/TimerModule.scala new file mode 100644 index 0000000000..0572e43bfa --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/modules/TimerModule.scala @@ -0,0 +1,13 @@ +package com.twitter.follow_recommendations.modules + +import com.google.inject.Provides +import com.google.inject.Singleton +import com.twitter.finagle.memcached.ZookeeperStateMonitor.DefaultTimer +import com.twitter.inject.TwitterModule +import com.twitter.util.Timer + +object TimerModule extends TwitterModule { + @Provides + @Singleton + def providesTimer: Timer = DefaultTimer +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/BUILD b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/BUILD new file mode 100644 index 0000000000..5840c0f2ff --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/BUILD @@ -0,0 +1,16 @@ +scala_library( + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/org/slf4j:slf4j-api", + "finatra/inject/inject-app/src/main/scala", + "finatra/inject/inject-core/src/main/scala", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/common", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/explore_tab", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/home_timeline", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/home_timeline_tweet_recs", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/sidebar", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/product/guice", + ], +) diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/ProdProductRegistry.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/ProdProductRegistry.scala new file mode 100644 index 0000000000..9a0dbb9957 --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/ProdProductRegistry.scala @@ -0,0 +1,44 @@ +package com.twitter.follow_recommendations.products + +import com.twitter.follow_recommendations.common.models.DisplayLocation +import com.twitter.follow_recommendations.products.common.ProductRegistry +import com.twitter.follow_recommendations.products.explore_tab.ExploreTabProduct +import com.twitter.follow_recommendations.products.home_timeline.HomeTimelineProduct +import com.twitter.follow_recommendations.products.home_timeline_tweet_recs.HomeTimelineTweetRecsProduct +import com.twitter.follow_recommendations.products.sidebar.SidebarProduct + +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class ProdProductRegistry @Inject() ( + exploreTabProduct: ExploreTabProduct, + homeTimelineProduct: HomeTimelineProduct, + homeTimelineTweetRecsProduct: HomeTimelineTweetRecsProduct, + sidebarProduct: SidebarProduct, +) extends ProductRegistry { + + override val products: Seq[common.Product] = + Seq( + exploreTabProduct, + homeTimelineProduct, + homeTimelineTweetRecsProduct, + sidebarProduct + ) + + override val displayLocationProductMap: Map[DisplayLocation, common.Product] = + products.groupBy(_.displayLocation).flatMap { + case (loc, products) => + assert(products.size == 1, s"Found more than 1 Product for ${loc}") + products.headOption.map { product => loc -> product } + } + + override def getProductByDisplayLocation(displayLocation: DisplayLocation): common.Product = { + displayLocationProductMap.getOrElse( + displayLocation, + throw new MissingProductException(displayLocation)) + } +} + +class MissingProductException(displayLocation: DisplayLocation) + extends Exception(s"No Product found for ${displayLocation}") diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/common/BUILD b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/common/BUILD new file mode 100644 index 0000000000..4b32816e4a --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/common/BUILD @@ -0,0 +1,12 @@ +scala_library( + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "configapi/configapi-core", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/models", + "follow-recommendations-service/thrift/src/main/thrift:thrift-scala", + ], +) diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/common/Exceptions.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/common/Exceptions.scala new file mode 100644 index 0000000000..c00d8c4074 --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/common/Exceptions.scala @@ -0,0 +1,7 @@ +package com.twitter.follow_recommendations.products.common + +abstract class ProductException(message: String) extends Exception(message) + +class MissingFieldException(productRequest: ProductRequest, fieldName: String) + extends ProductException( + s"Missing ${fieldName} field for ${productRequest.recommendationRequest.displayLocation} request") diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/common/Product.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/common/Product.scala new file mode 100644 index 0000000000..28c348204a --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/common/Product.scala @@ -0,0 +1,56 @@ +package com.twitter.follow_recommendations.products.common + +import com.twitter.follow_recommendations.assembler.models.Layout +import com.twitter.follow_recommendations.common.base.BaseRecommendationFlow +import com.twitter.follow_recommendations.common.base.Transform +import com.twitter.follow_recommendations.common.models.DisplayLocation +import com.twitter.follow_recommendations.common.models.Recommendation +import com.twitter.follow_recommendations.models.RecommendationRequest +import com.twitter.product_mixer.core.model.marshalling.request.{Product => ProductMixerProduct} +import com.twitter.stitch.Stitch +import com.twitter.timelines.configapi.Params + +trait Product { + + /** Each product also requires a human-readable name. + * You can change this at any time + */ + def name: String + + /** + * Every product needs a machine-friendly identifier for internal use. + * You should use the same name as the product package name. + * Except dashes are better than underscore + * + * Avoid changing this once it's in production. + */ + def identifier: String + + def displayLocation: DisplayLocation + + def selectWorkflows( + request: ProductRequest + ): Stitch[Seq[BaseRecommendationFlow[ProductRequest, _ <: Recommendation]]] + + /** + * Blender is responsible for blending together the candidates generated by different flows used + * in a product. For example, if a product uses two flows, it is blender's responsibility to + * interleave their generated candidates together and make a unified sequence of candidates. + */ + def blender: Transform[ProductRequest, Recommendation] + + /** + * It is resultsTransformer job to do any final transformations needed on the final list of + * candidates generated by a product. For example, if a final quality check on candidates needed, + * resultsTransformer will handle it. + */ + def resultsTransformer(request: ProductRequest): Stitch[Transform[ProductRequest, Recommendation]] + + def enabled(request: ProductRequest): Stitch[Boolean] + + def layout: Option[Layout] = None + + def productMixerProduct: Option[ProductMixerProduct] = None +} + +case class ProductRequest(recommendationRequest: RecommendationRequest, params: Params) diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/common/ProductRegistry.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/common/ProductRegistry.scala new file mode 100644 index 0000000000..fbe4865360 --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/common/ProductRegistry.scala @@ -0,0 +1,9 @@ +package com.twitter.follow_recommendations.products.common + +import com.twitter.follow_recommendations.common.models.DisplayLocation + +trait ProductRegistry { + def products: Seq[Product] + def displayLocationProductMap: Map[DisplayLocation, Product] + def getProductByDisplayLocation(displayLocation: DisplayLocation): Product +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/explore_tab/BUILD b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/explore_tab/BUILD new file mode 100644 index 0000000000..2f94126126 --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/explore_tab/BUILD @@ -0,0 +1,14 @@ +scala_library( + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "finatra/inject/inject-app/src/main/scala", + "finatra/inject/inject-core/src/main/scala", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/blenders", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/ads", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/post_nux_ml", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/common", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/explore_tab/configapi", + ], +) diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/explore_tab/ExploreTabProduct.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/explore_tab/ExploreTabProduct.scala new file mode 100644 index 0000000000..a49fccb450 --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/explore_tab/ExploreTabProduct.scala @@ -0,0 +1,50 @@ +package com.twitter.follow_recommendations.products.explore_tab + +import com.twitter.follow_recommendations.common.base.BaseRecommendationFlow +import com.twitter.follow_recommendations.common.base.IdentityTransform +import com.twitter.follow_recommendations.common.base.Transform +import com.twitter.follow_recommendations.common.models.DisplayLocation +import com.twitter.follow_recommendations.common.models.Recommendation +import com.twitter.follow_recommendations.flows.post_nux_ml.PostNuxMlFlow +import com.twitter.follow_recommendations.flows.post_nux_ml.PostNuxMlRequestBuilder +import com.twitter.follow_recommendations.products.common.Product +import com.twitter.follow_recommendations.products.common.ProductRequest +import com.twitter.follow_recommendations.products.explore_tab.configapi.ExploreTabParams +import com.twitter.stitch.Stitch +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class ExploreTabProduct @Inject() ( + postNuxMlFlow: PostNuxMlFlow, + postNuxMlRequestBuilder: PostNuxMlRequestBuilder) + extends Product { + override val name: String = "Explore Tab" + + override val identifier: String = "explore-tab" + + override val displayLocation: DisplayLocation = DisplayLocation.ExploreTab + + override def selectWorkflows( + request: ProductRequest + ): Stitch[Seq[BaseRecommendationFlow[ProductRequest, _ <: Recommendation]]] = { + postNuxMlRequestBuilder.build(request).map { postNuxMlRequest => + Seq(postNuxMlFlow.mapKey({ _: ProductRequest => postNuxMlRequest })) + } + } + + override val blender: Transform[ProductRequest, Recommendation] = + new IdentityTransform[ProductRequest, Recommendation] + + override def resultsTransformer( + request: ProductRequest + ): Stitch[Transform[ProductRequest, Recommendation]] = + Stitch.value(new IdentityTransform[ProductRequest, Recommendation]) + + override def enabled(request: ProductRequest): Stitch[Boolean] = { + // Ideally we should hook up is_soft_user as custom FS field and disable the product through FS + val enabledForUserType = !request.recommendationRequest.isSoftUser || request.params( + ExploreTabParams.EnableProductForSoftUser) + Stitch.value(request.params(ExploreTabParams.EnableProduct) && enabledForUserType) + } +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/explore_tab/configapi/BUILD b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/explore_tab/configapi/BUILD new file mode 100644 index 0000000000..3bb732e35b --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/explore_tab/configapi/BUILD @@ -0,0 +1,9 @@ +scala_library( + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "configapi/configapi-core", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/common", + ], +) diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/explore_tab/configapi/ExploreTabFSConfig.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/explore_tab/configapi/ExploreTabFSConfig.scala new file mode 100644 index 0000000000..092252aca9 --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/explore_tab/configapi/ExploreTabFSConfig.scala @@ -0,0 +1,14 @@ +package com.twitter.follow_recommendations.products.explore_tab.configapi + +import com.twitter.follow_recommendations.configapi.common.FeatureSwitchConfig +import com.twitter.follow_recommendations.products.explore_tab.configapi.ExploreTabParams._ +import com.twitter.timelines.configapi.FSName +import com.twitter.timelines.configapi.Param +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class ExploreTabFSConfig @Inject() () extends FeatureSwitchConfig { + override val booleanFSParams: Seq[Param[Boolean] with FSName] = + Seq(EnableProductForSoftUser) +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/explore_tab/configapi/ExploreTabParams.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/explore_tab/configapi/ExploreTabParams.scala new file mode 100644 index 0000000000..b9d9d3b872 --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/explore_tab/configapi/ExploreTabParams.scala @@ -0,0 +1,10 @@ +package com.twitter.follow_recommendations.products.explore_tab.configapi + +import com.twitter.timelines.configapi.Param +import com.twitter.timelines.configapi.FSParam + +object ExploreTabParams { + object EnableProduct extends Param[Boolean](false) + object EnableProductForSoftUser + extends FSParam[Boolean]("explore_tab_enable_product_for_soft_user", false) +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/home_timeline/BUILD b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/home_timeline/BUILD new file mode 100644 index 0000000000..4b0586ff7a --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/home_timeline/BUILD @@ -0,0 +1,14 @@ +scala_library( + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "finatra/inject/inject-app/src/main/scala", + "finatra/inject/inject-core/src/main/scala", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/blenders", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/ads", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/post_nux_ml", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/common", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/home_timeline/configapi", + ], +) diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/home_timeline/HTLProductMixer.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/home_timeline/HTLProductMixer.scala new file mode 100644 index 0000000000..a2051c150a --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/home_timeline/HTLProductMixer.scala @@ -0,0 +1,9 @@ +package com.twitter.follow_recommendations.products.home_timeline + +import com.twitter.product_mixer.core.model.common.identifier.ProductIdentifier +import com.twitter.product_mixer.core.model.marshalling.request.Product + +case object HTLProductMixer extends Product { + override val identifier: ProductIdentifier = ProductIdentifier("HomeTimeline") + override val stringCenterProject: Option[String] = Some("people-discovery") +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/home_timeline/HomeTimelineProduct.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/home_timeline/HomeTimelineProduct.scala new file mode 100644 index 0000000000..590aab182f --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/home_timeline/HomeTimelineProduct.scala @@ -0,0 +1,114 @@ +package com.twitter.follow_recommendations.products.home_timeline + +import com.twitter.follow_recommendations.assembler.models.ActionConfig +import com.twitter.follow_recommendations.assembler.models.FollowedByUsersProof +import com.twitter.follow_recommendations.assembler.models.FooterConfig +import com.twitter.follow_recommendations.assembler.models.GeoContextProof +import com.twitter.follow_recommendations.assembler.models.HeaderConfig +import com.twitter.follow_recommendations.assembler.models.Layout +import com.twitter.follow_recommendations.assembler.models.TitleConfig +import com.twitter.follow_recommendations.assembler.models.UserListLayout +import com.twitter.follow_recommendations.assembler.models.UserListOptions +import com.twitter.follow_recommendations.common.base.BaseRecommendationFlow +import com.twitter.follow_recommendations.common.base.IdentityTransform +import com.twitter.follow_recommendations.common.base.Transform +import com.twitter.follow_recommendations.flows.ads.PromotedAccountsFlow +import com.twitter.follow_recommendations.flows.ads.PromotedAccountsFlowRequest +import com.twitter.follow_recommendations.blenders.PromotedAccountsBlender +import com.twitter.follow_recommendations.common.models.DisplayLocation +import com.twitter.follow_recommendations.common.models.Recommendation +import com.twitter.follow_recommendations.flows.post_nux_ml.PostNuxMlFlow +import com.twitter.follow_recommendations.flows.post_nux_ml.PostNuxMlRequestBuilder +import com.twitter.follow_recommendations.products.common.Product +import com.twitter.follow_recommendations.products.common.ProductRequest +import com.twitter.follow_recommendations.products.home_timeline.configapi.HomeTimelineParams._ +import com.twitter.inject.Injector +import com.twitter.product_mixer.core.model.marshalling.request +import com.twitter.product_mixer.core.product.guice.ProductScope +import com.twitter.stitch.Stitch +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class HomeTimelineProduct @Inject() ( + postNuxMlFlow: PostNuxMlFlow, + postNuxMlRequestBuilder: PostNuxMlRequestBuilder, + promotedAccountsFlow: PromotedAccountsFlow, + promotedAccountsBlender: PromotedAccountsBlender, + productScope: ProductScope, + injector: Injector, +) extends Product { + + override val name: String = "Home Timeline" + + override val identifier: String = "home-timeline" + + override val displayLocation: DisplayLocation = DisplayLocation.HomeTimeline + + override def selectWorkflows( + request: ProductRequest + ): Stitch[Seq[BaseRecommendationFlow[ProductRequest, _ <: Recommendation]]] = { + postNuxMlRequestBuilder.build(request).map { postNuxMlRequest => + Seq( + postNuxMlFlow.mapKey({ request: ProductRequest => postNuxMlRequest }), + promotedAccountsFlow.mapKey(mkPromotedAccountsRequest)) + } + } + + override val blender: Transform[ProductRequest, Recommendation] = { + promotedAccountsBlender.mapTarget[ProductRequest](getMaxResults) + } + + private val identityTransform = new IdentityTransform[ProductRequest, Recommendation] + + override def resultsTransformer( + request: ProductRequest + ): Stitch[Transform[ProductRequest, Recommendation]] = Stitch.value(identityTransform) + + override def enabled(request: ProductRequest): Stitch[Boolean] = + Stitch.value(request.params(EnableProduct)) + + override def layout: Option[Layout] = { + productMixerProduct.map { product => + val homeTimelineStrings = productScope.let(product) { + injector.instance[HomeTimelineStrings] + } + UserListLayout( + header = Some(HeaderConfig(TitleConfig(homeTimelineStrings.whoToFollowModuleTitle))), + userListOptions = UserListOptions(userBioEnabled = true, userBioTruncated = true, None), + socialProofs = Some( + Seq( + FollowedByUsersProof( + homeTimelineStrings.whoToFollowFollowedByManyUserSingleString, + homeTimelineStrings.whoToFollowFollowedByManyUserDoubleString, + homeTimelineStrings.whoToFollowFollowedByManyUserMultipleString + ), + GeoContextProof(homeTimelineStrings.whoToFollowPopularInCountryKey) + )), + footer = Some( + FooterConfig( + Some(ActionConfig(homeTimelineStrings.whoToFollowModuleFooter, "http://twitter.com")))) + ) + } + } + + override def productMixerProduct: Option[request.Product] = Some(HTLProductMixer) + + private[home_timeline] def mkPromotedAccountsRequest( + req: ProductRequest + ): PromotedAccountsFlowRequest = { + PromotedAccountsFlowRequest( + req.recommendationRequest.clientContext, + req.params, + req.recommendationRequest.displayLocation, + None, + req.recommendationRequest.excludedIds.getOrElse(Nil) + ) + } + + private[home_timeline] def getMaxResults(req: ProductRequest): Int = { + req.recommendationRequest.maxResults.getOrElse( + req.params(DefaultMaxResults) + ) + } +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/home_timeline/HomeTimelineStrings.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/home_timeline/HomeTimelineStrings.scala new file mode 100644 index 0000000000..75819555ed --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/home_timeline/HomeTimelineStrings.scala @@ -0,0 +1,26 @@ +package com.twitter.follow_recommendations.products.home_timeline + +import com.twitter.product_mixer.core.product.guice.scope.ProductScoped +import com.twitter.stringcenter.client.ExternalStringRegistry +import com.twitter.stringcenter.client.core.ExternalString +import javax.inject.Inject +import javax.inject.Provider +import javax.inject.Singleton + +@Singleton +class HomeTimelineStrings @Inject() ( + @ProductScoped externalStringRegistryProvider: Provider[ExternalStringRegistry]) { + private val externalStringRegistry = externalStringRegistryProvider.get() + val whoToFollowFollowedByManyUserSingleString: ExternalString = + externalStringRegistry.createProdString("WtfRecommendationContext.followedByManyUserSingle") + val whoToFollowFollowedByManyUserDoubleString: ExternalString = + externalStringRegistry.createProdString("WtfRecommendationContext.followedByManyUserDouble") + val whoToFollowFollowedByManyUserMultipleString: ExternalString = + externalStringRegistry.createProdString("WtfRecommendationContext.followedByManyUserMultiple") + val whoToFollowPopularInCountryKey: ExternalString = + externalStringRegistry.createProdString("WtfRecommendationContext.popularInCountry") + val whoToFollowModuleTitle: ExternalString = + externalStringRegistry.createProdString("WhoToFollowModule.title") + val whoToFollowModuleFooter: ExternalString = + externalStringRegistry.createProdString("WhoToFollowModule.pivot") +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/home_timeline/configapi/BUILD b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/home_timeline/configapi/BUILD new file mode 100644 index 0000000000..3bb732e35b --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/home_timeline/configapi/BUILD @@ -0,0 +1,9 @@ +scala_library( + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "configapi/configapi-core", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/common", + ], +) diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/home_timeline/configapi/HomeTimelineFSConfig.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/home_timeline/configapi/HomeTimelineFSConfig.scala new file mode 100644 index 0000000000..15e97b3a5a --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/home_timeline/configapi/HomeTimelineFSConfig.scala @@ -0,0 +1,22 @@ +package com.twitter.follow_recommendations.products.home_timeline.configapi + +import com.twitter.follow_recommendations.configapi.common.FeatureSwitchConfig +import com.twitter.follow_recommendations.products.home_timeline.configapi.HomeTimelineParams._ +import com.twitter.timelines.configapi.FSBoundedParam +import com.twitter.timelines.configapi.FSName +import com.twitter.timelines.configapi.HasDurationConversion +import com.twitter.timelines.configapi.Param +import com.twitter.util.Duration +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class HomeTimelineFSConfig @Inject() () extends FeatureSwitchConfig { + override val booleanFSParams: Seq[Param[Boolean] with FSName] = + Seq(EnableWritingServingHistory) + + override val durationFSParams: Seq[FSBoundedParam[Duration] with HasDurationConversion] = Seq( + DurationGuardrailToForceSuggest, + SuggestBasedFatigueDuration + ) +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/home_timeline/configapi/HomeTimelineParams.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/home_timeline/configapi/HomeTimelineParams.scala new file mode 100644 index 0000000000..65ab5ae231 --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/home_timeline/configapi/HomeTimelineParams.scala @@ -0,0 +1,38 @@ +package com.twitter.follow_recommendations.products.home_timeline.configapi + +import com.twitter.conversions.DurationOps._ +import com.twitter.timelines.configapi.DurationConversion +import com.twitter.timelines.configapi.FSBoundedParam +import com.twitter.timelines.configapi.FSParam +import com.twitter.timelines.configapi.HasDurationConversion +import com.twitter.timelines.configapi.Param +import com.twitter.util.Duration + +object HomeTimelineParams { + object EnableProduct extends Param[Boolean](false) + + object DefaultMaxResults extends Param[Int](20) + + object EnableWritingServingHistory + extends FSParam[Boolean]("home_timeline_enable_writing_serving_history", false) + + object DurationGuardrailToForceSuggest + extends FSBoundedParam[Duration]( + name = "home_timeline_duration_guardrail_to_force_suggest_in_hours", + default = 0.hours, + min = 0.hours, + max = 1000.hours) + with HasDurationConversion { + override val durationConversion: DurationConversion = DurationConversion.FromHours + } + + object SuggestBasedFatigueDuration + extends FSBoundedParam[Duration]( + name = "home_timeline_suggest_based_fatigue_duration_in_hours", + default = 0.hours, + min = 0.hours, + max = 1000.hours) + with HasDurationConversion { + override val durationConversion: DurationConversion = DurationConversion.FromHours + } +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/home_timeline_tweet_recs/BUILD b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/home_timeline_tweet_recs/BUILD new file mode 100644 index 0000000000..140cc928da --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/home_timeline_tweet_recs/BUILD @@ -0,0 +1,13 @@ +scala_library( + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "finatra/inject/inject-app/src/main/scala", + "finatra/inject/inject-core/src/main/scala", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/blenders", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/content_recommender_flow", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/common", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/home_timeline_tweet_recs/configapi", + ], +) diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/home_timeline_tweet_recs/HomeTimelineTweetRecsProduct.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/home_timeline_tweet_recs/HomeTimelineTweetRecsProduct.scala new file mode 100644 index 0000000000..a5586f296f --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/home_timeline_tweet_recs/HomeTimelineTweetRecsProduct.scala @@ -0,0 +1,50 @@ +package com.twitter.follow_recommendations.products.home_timeline_tweet_recs + +import com.twitter.follow_recommendations.common.base.BaseRecommendationFlow +import com.twitter.follow_recommendations.common.base.IdentityTransform +import com.twitter.follow_recommendations.common.base.Transform +import com.twitter.follow_recommendations.common.models.DisplayLocation +import com.twitter.follow_recommendations.common.models.Recommendation +import com.twitter.follow_recommendations.flows.content_recommender_flow.ContentRecommenderFlow +import com.twitter.follow_recommendations.flows.content_recommender_flow.ContentRecommenderRequestBuilder +import com.twitter.follow_recommendations.products.common.Product +import com.twitter.follow_recommendations.products.common.ProductRequest +import com.twitter.follow_recommendations.products.home_timeline_tweet_recs.configapi.HomeTimelineTweetRecsParams._ +import com.twitter.stitch.Stitch +import javax.inject.Inject +import javax.inject.Singleton + +/* + * This "DisplayLocation" is used to generate user recommendations using the ContentRecommenderFlow. These recommendations are later used downstream + * to generate recommended tweets on Home Timeline. + */ +@Singleton +class HomeTimelineTweetRecsProduct @Inject() ( + contentRecommenderFlow: ContentRecommenderFlow, + contentRecommenderRequestBuilder: ContentRecommenderRequestBuilder) + extends Product { + override val name: String = "Home Timeline Tweet Recs" + + override val identifier: String = "home-timeline-tweet-recs" + + override val displayLocation: DisplayLocation = DisplayLocation.HomeTimelineTweetRecs + + override def selectWorkflows( + request: ProductRequest + ): Stitch[Seq[BaseRecommendationFlow[ProductRequest, _ <: Recommendation]]] = { + contentRecommenderRequestBuilder.build(request).map { contentRecommenderRequest => + Seq(contentRecommenderFlow.mapKey({ request: ProductRequest => contentRecommenderRequest })) + } + } + + override val blender: Transform[ProductRequest, Recommendation] = + new IdentityTransform[ProductRequest, Recommendation] + + override def resultsTransformer( + request: ProductRequest + ): Stitch[Transform[ProductRequest, Recommendation]] = + Stitch.value(new IdentityTransform[ProductRequest, Recommendation]) + + override def enabled(request: ProductRequest): Stitch[Boolean] = + Stitch.value(request.params(EnableProduct)) +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/home_timeline_tweet_recs/configapi/BUILD b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/home_timeline_tweet_recs/configapi/BUILD new file mode 100644 index 0000000000..9be8d96473 --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/home_timeline_tweet_recs/configapi/BUILD @@ -0,0 +1,10 @@ +scala_library( + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "configapi/configapi-core", + "configapi/configapi-decider", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/common", + ], +) diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/home_timeline_tweet_recs/configapi/HomeTimelineTweetRecsParams.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/home_timeline_tweet_recs/configapi/HomeTimelineTweetRecsParams.scala new file mode 100644 index 0000000000..319a51847f --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/home_timeline_tweet_recs/configapi/HomeTimelineTweetRecsParams.scala @@ -0,0 +1,7 @@ +package com.twitter.follow_recommendations.products.home_timeline_tweet_recs.configapi + +import com.twitter.timelines.configapi.Param + +object HomeTimelineTweetRecsParams { + object EnableProduct extends Param[Boolean](false) +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/sidebar/BUILD b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/sidebar/BUILD new file mode 100644 index 0000000000..f469a47484 --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/sidebar/BUILD @@ -0,0 +1,14 @@ +scala_library( + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "finatra/inject/inject-app/src/main/scala", + "finatra/inject/inject-core/src/main/scala", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/blenders", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/ads", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/flows/post_nux_ml", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/common", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/sidebar/configapi", + ], +) diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/sidebar/SidebarProduct.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/sidebar/SidebarProduct.scala new file mode 100644 index 0000000000..29f7880118 --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/sidebar/SidebarProduct.scala @@ -0,0 +1,73 @@ +package com.twitter.follow_recommendations.products.sidebar + +import com.twitter.follow_recommendations.common.base.BaseRecommendationFlow +import com.twitter.follow_recommendations.common.base.IdentityTransform +import com.twitter.follow_recommendations.common.base.Transform +import com.twitter.follow_recommendations.flows.ads.PromotedAccountsFlow +import com.twitter.follow_recommendations.flows.ads.PromotedAccountsFlowRequest +import com.twitter.follow_recommendations.blenders.PromotedAccountsBlender +import com.twitter.follow_recommendations.common.models.DisplayLocation +import com.twitter.follow_recommendations.common.models.Recommendation +import com.twitter.follow_recommendations.flows.post_nux_ml.PostNuxMlFlow +import com.twitter.follow_recommendations.flows.post_nux_ml.PostNuxMlRequestBuilder +import com.twitter.follow_recommendations.products.common.Product +import com.twitter.follow_recommendations.products.common.ProductRequest +import com.twitter.follow_recommendations.products.sidebar.configapi.SidebarParams +import com.twitter.stitch.Stitch +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class SidebarProduct @Inject() ( + postNuxMlFlow: PostNuxMlFlow, + postNuxMlRequestBuilder: PostNuxMlRequestBuilder, + promotedAccountsFlow: PromotedAccountsFlow, + promotedAccountsBlender: PromotedAccountsBlender) + extends Product { + override val name: String = "Sidebar" + + override val identifier: String = "sidebar" + + override val displayLocation: DisplayLocation = DisplayLocation.Sidebar + + override def selectWorkflows( + request: ProductRequest + ): Stitch[Seq[BaseRecommendationFlow[ProductRequest, _ <: Recommendation]]] = { + postNuxMlRequestBuilder.build(request).map { postNuxMlRequest => + Seq( + postNuxMlFlow.mapKey({ _: ProductRequest => postNuxMlRequest }), + promotedAccountsFlow.mapKey(mkPromotedAccountsRequest) + ) + } + } + + override val blender: Transform[ProductRequest, Recommendation] = { + promotedAccountsBlender.mapTarget[ProductRequest](getMaxResults) + } + + private[sidebar] def mkPromotedAccountsRequest( + req: ProductRequest + ): PromotedAccountsFlowRequest = { + PromotedAccountsFlowRequest( + req.recommendationRequest.clientContext, + req.params, + req.recommendationRequest.displayLocation, + None, + req.recommendationRequest.excludedIds.getOrElse(Nil) + ) + } + + private[sidebar] def getMaxResults(req: ProductRequest): Int = { + req.recommendationRequest.maxResults.getOrElse( + req.params(SidebarParams.DefaultMaxResults) + ) + } + + override def resultsTransformer( + request: ProductRequest + ): Stitch[Transform[ProductRequest, Recommendation]] = + Stitch.value(new IdentityTransform[ProductRequest, Recommendation]) + + override def enabled(request: ProductRequest): Stitch[Boolean] = + Stitch.value(request.params(SidebarParams.EnableProduct)) +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/sidebar/configapi/BUILD b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/sidebar/configapi/BUILD new file mode 100644 index 0000000000..6fee24f895 --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/sidebar/configapi/BUILD @@ -0,0 +1,8 @@ +scala_library( + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "configapi/configapi-core", + ], +) diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/sidebar/configapi/SidebarParams.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/sidebar/configapi/SidebarParams.scala new file mode 100644 index 0000000000..bbd0264957 --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products/sidebar/configapi/SidebarParams.scala @@ -0,0 +1,9 @@ +package com.twitter.follow_recommendations.products.sidebar.configapi + +import com.twitter.timelines.configapi.Param + +object SidebarParams { + object EnableProduct extends Param[Boolean](false) + + object DefaultMaxResults extends Param[Int](20) +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/services/BUILD b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/services/BUILD new file mode 100644 index 0000000000..fe29c22dea --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/services/BUILD @@ -0,0 +1,34 @@ +scala_library( + compiler_option_sets = ["fatal_warnings"], + tags = ["bazel-compatible"], + dependencies = [ + "finatra/inject/inject-app/src/main/scala", + "finatra/inject/inject-core/src/main/scala", + "finatra/inject/inject-server/src/main/scala", + "finatra/inject/inject-thrift-client", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/crowd_search_accounts", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/real_graph", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/recent_engagement", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/top_organic_follows_accounts", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/clients/impression_store", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/sources", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/predicates/sgs", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/ml_ranker/ranking", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/rankers/ml_ranker/scoring", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/utils", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/assembler/models", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/deciders", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/logging", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/products", + "follow-recommendations-service/thrift/src/main/thrift:thrift-scala", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/product", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/product/registry", + "twitter-server/server/src/main/scala", + "util/util-app/src/main/scala", + "util/util-core:scala", + "util/util-slf4j-api/src/main/scala", + ], +) diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/services/FollowRecommendationsServiceWarmupHandler.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/services/FollowRecommendationsServiceWarmupHandler.scala new file mode 100644 index 0000000000..7567fe9cee --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/services/FollowRecommendationsServiceWarmupHandler.scala @@ -0,0 +1,101 @@ +package com.twitter.follow_recommendations.services + +import com.twitter.finagle.thrift.ClientId +import com.twitter.finatra.thrift.routing.ThriftWarmup +import com.twitter.follow_recommendations.thriftscala.FollowRecommendationsThriftService.GetRecommendations +import com.twitter.follow_recommendations.thriftscala.ClientContext +import com.twitter.follow_recommendations.thriftscala.DebugParams +import com.twitter.follow_recommendations.thriftscala.DisplayContext +import com.twitter.follow_recommendations.thriftscala.DisplayLocation +import com.twitter.follow_recommendations.thriftscala.Profile +import com.twitter.follow_recommendations.thriftscala.RecommendationRequest +import com.twitter.inject.Logging +import com.twitter.inject.utils.Handler +import com.twitter.scrooge.Request +import com.twitter.scrooge.Response +import com.twitter.util.Return +import com.twitter.util.Throw +import com.twitter.util.Try +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class FollowRecommendationsServiceWarmupHandler @Inject() (warmup: ThriftWarmup) + extends Handler + with Logging { + + private val clientId = ClientId("thrift-warmup-client") + + override def handle(): Unit = { + val testIds = Seq(1L) + def warmupQuery(userId: Long, displayLocation: DisplayLocation): RecommendationRequest = { + val clientContext = ClientContext( + userId = Some(userId), + guestId = None, + appId = Some(258901L), + ipAddress = Some("0.0.0.0"), + userAgent = Some("FAKE_USER_AGENT_FOR_WARMUPS"), + countryCode = Some("US"), + languageCode = Some("en"), + isTwoffice = None, + userRoles = None, + deviceId = Some("FAKE_DEVICE_ID_FOR_WARMUPS") + ) + RecommendationRequest( + clientContext = clientContext, + displayLocation = displayLocation, + displayContext = None, + maxResults = Some(3), + fetchPromotedContent = Some(false), + debugParams = Some(DebugParams(doNotLog = Some(true))) + ) + } + + // Add FRS display locations here if they should be targeted for warm-up + // when FRS is starting from a fresh state after a deploy + val displayLocationsToWarmUp: Seq[DisplayLocation] = Seq( + DisplayLocation.HomeTimeline, + DisplayLocation.HomeTimelineReverseChron, + DisplayLocation.ProfileSidebar, + DisplayLocation.NuxInterests, + DisplayLocation.NuxPymk + ) + + try { + clientId.asCurrent { + // Iterate over each user ID created for testing + testIds foreach { id => + // Iterate over each display location targeted for warm-up + displayLocationsToWarmUp foreach { displayLocation => + val warmupReq = warmupQuery(id, displayLocation) + info(s"Sending warm-up request to service with query: $warmupReq") + warmup.sendRequest( + method = GetRecommendations, + req = Request(GetRecommendations.Args(warmupReq)))(assertWarmupResponse) + // send the request one more time so that it goes through cache hits + warmup.sendRequest( + method = GetRecommendations, + req = Request(GetRecommendations.Args(warmupReq)))(assertWarmupResponse) + } + } + } + } catch { + case e: Throwable => + // we don't want a warmup failure to prevent start-up + error(e.getMessage, e) + } + info("Warm-up done.") + } + + /* Private */ + + private def assertWarmupResponse(result: Try[Response[GetRecommendations.SuccessType]]): Unit = { + // we collect and log any exceptions from the result. + result match { + case Return(_) => // ok + case Throw(exception) => + warn() + error(s"Error performing warm-up request: ${exception.getMessage}", exception) + } + } +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/services/ProductMixerRecommendationService.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/services/ProductMixerRecommendationService.scala new file mode 100644 index 0000000000..daff9040dc --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/services/ProductMixerRecommendationService.scala @@ -0,0 +1,72 @@ +package com.twitter.follow_recommendations.services + +import com.twitter.finagle.stats.StatsReceiver +import javax.inject.Inject +import javax.inject.Singleton +import com.twitter.timelines.configapi.Params +import com.twitter.follow_recommendations.common.utils.DisplayLocationProductConverterUtil +import com.twitter.follow_recommendations.configapi.deciders.DeciderParams +import com.twitter.follow_recommendations.logging.FrsLogger +import com.twitter.follow_recommendations.models.{DebugParams => FrsDebugParams} +import com.twitter.follow_recommendations.models.RecommendationRequest +import com.twitter.follow_recommendations.models.RecommendationResponse +import com.twitter.follow_recommendations.models.Request +import com.twitter.product_mixer.core.model.marshalling.request.{ + DebugParams => ProductMixerDebugParams +} +import com.twitter.product_mixer.core.product.registry.ProductPipelineRegistry +import com.twitter.product_mixer.core.pipeline.product.ProductPipelineRequest +import com.twitter.stitch.Stitch + +@Singleton +class ProductMixerRecommendationService @Inject() ( + productPipelineRegistry: ProductPipelineRegistry, + resultLogger: FrsLogger, + baseStats: StatsReceiver) { + + private val stats = baseStats.scope("product_mixer_recos_service_stats") + private val loggingStats = stats.scope("logged") + + def get(request: RecommendationRequest, params: Params): Stitch[RecommendationResponse] = { + if (params(DeciderParams.EnableRecommendations)) { + val productMixerRequest = convertToProductMixerRequest(request) + + productPipelineRegistry + .getProductPipeline[Request, RecommendationResponse](productMixerRequest.product) + .process(ProductPipelineRequest(productMixerRequest, params)).onSuccess { response => + if (resultLogger.shouldLog(request.debugParams)) { + loggingStats.counter().incr() + resultLogger.logRecommendationResult(request, response) + } + } + } else { + Stitch.value(RecommendationResponse(Nil)) + } + + } + + def convertToProductMixerRequest(frsRequest: RecommendationRequest): Request = { + Request( + maxResults = frsRequest.maxResults, + debugParams = convertToProductMixerDebugParams(frsRequest.debugParams), + productContext = None, + product = + DisplayLocationProductConverterUtil.displayLocationToProduct(frsRequest.displayLocation), + clientContext = frsRequest.clientContext, + serializedRequestCursor = frsRequest.cursor, + frsDebugParams = frsRequest.debugParams, + displayLocation = frsRequest.displayLocation, + excludedIds = frsRequest.excludedIds, + fetchPromotedContent = frsRequest.fetchPromotedContent, + userLocationState = frsRequest.userLocationState + ) + } + + private def convertToProductMixerDebugParams( + frsDebugParams: Option[FrsDebugParams] + ): Option[ProductMixerDebugParams] = { + frsDebugParams.map { debugParams => + ProductMixerDebugParams(debugParams.featureOverrides, None) + } + } +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/services/ProductPipelineSelector.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/services/ProductPipelineSelector.scala new file mode 100644 index 0000000000..1c949f03d4 --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/services/ProductPipelineSelector.scala @@ -0,0 +1,188 @@ +package com.twitter.follow_recommendations.services + +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.follow_recommendations.common.base.StatsUtil +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.follow_recommendations.common.models.DebugOptions +import com.twitter.follow_recommendations.models.DebugParams +import com.twitter.follow_recommendations.models.RecommendationRequest +import com.twitter.follow_recommendations.models.RecommendationResponse +import com.twitter.stitch.Stitch +import com.twitter.timelines.configapi.Params +import javax.inject.Inject +import javax.inject.Singleton +import scala.util.Random + +@Singleton +class ProductPipelineSelector @Inject() ( + recommendationsService: RecommendationsService, + productMixerRecommendationService: ProductMixerRecommendationService, + productPipelineSelectorConfig: ProductPipelineSelectorConfig, + baseStats: StatsReceiver) { + + private val frsStats = baseStats.scope("follow_recommendations_service") + private val stats = frsStats.scope("product_pipeline_selector_parity") + + private val readFromProductMixerCounter = stats.counter("select_product_mixer") + private val readFromOldFRSCounter = stats.counter("select_old_frs") + + def selectPipeline( + request: RecommendationRequest, + params: Params + ): Stitch[RecommendationResponse] = { + productPipelineSelectorConfig + .getDarkReadAndExpParams(request.displayLocation).map { darkReadAndExpParam => + if (params(darkReadAndExpParam.expParam)) { + readFromProductMixerPipeline(request, params) + } else if (params(darkReadAndExpParam.darkReadParam)) { + darkReadAndReturnResult(request, params) + } else { + readFromOldFrsPipeline(request, params) + } + }.getOrElse(readFromOldFrsPipeline(request, params)) + } + + private def readFromProductMixerPipeline( + request: RecommendationRequest, + params: Params + ): Stitch[RecommendationResponse] = { + readFromProductMixerCounter.incr() + productMixerRecommendationService.get(request, params) + } + + private def readFromOldFrsPipeline( + request: RecommendationRequest, + params: Params + ): Stitch[RecommendationResponse] = { + readFromOldFRSCounter.incr() + recommendationsService.get(request, params) + } + + private def darkReadAndReturnResult( + request: RecommendationRequest, + params: Params + ): Stitch[RecommendationResponse] = { + val darkReadStats = stats.scope("select_dark_read", request.displayLocation.toFsName) + darkReadStats.counter("count").incr() + + // If no seed is set, create a random one that both requests will use to remove differences + // in randomness for the WeightedCandidateSourceRanker + val randomizationSeed = new Random().nextLong() + + val oldFRSPiplelineRequest = request.copy( + debugParams = Some( + request.debugParams.getOrElse( + DebugParams(None, Some(DebugOptions(randomizationSeed = Some(randomizationSeed)))))) + ) + val productMixerPipelineRequest = request.copy( + debugParams = Some( + request.debugParams.getOrElse( + DebugParams( + None, + Some(DebugOptions(doNotLog = true, randomizationSeed = Some(randomizationSeed)))))) + ) + + StatsUtil + .profileStitch( + readFromOldFrsPipeline(oldFRSPiplelineRequest, params), + darkReadStats.scope("frs_timing")).applyEffect { frsOldPipelineResponse => + Stitch.async( + StatsUtil + .profileStitch( + readFromProductMixerPipeline(productMixerPipelineRequest, params), + darkReadStats.scope("product_mixer_timing")).liftToOption().map { + case Some(frsProductMixerResponse) => + darkReadStats.counter("product_mixer_pipeline_success").incr() + compare(request, frsOldPipelineResponse, frsProductMixerResponse) + case None => + darkReadStats.counter("product_mixer_pipeline_failure").incr() + } + ) + } + } + + def compare( + request: RecommendationRequest, + frsOldPipelineResponse: RecommendationResponse, + frsProductMixerResponse: RecommendationResponse + ): Unit = { + val compareStats = stats.scope("pipeline_comparison", request.displayLocation.toFsName) + compareStats.counter("total-comparisons").incr() + + val oldFrsMap = frsOldPipelineResponse.recommendations.map { user => user.id -> user }.toMap + val productMixerMap = frsProductMixerResponse.recommendations.map { user => + user.id -> user + }.toMap + + compareTopNResults(3, frsOldPipelineResponse, frsProductMixerResponse, compareStats) + compareTopNResults(5, frsOldPipelineResponse, frsProductMixerResponse, compareStats) + compareTopNResults(25, frsOldPipelineResponse, frsProductMixerResponse, compareStats) + compareTopNResults(50, frsOldPipelineResponse, frsProductMixerResponse, compareStats) + compareTopNResults(75, frsOldPipelineResponse, frsProductMixerResponse, compareStats) + + // Compare individual matching candidates + oldFrsMap.keys.foreach(userId => { + if (productMixerMap.contains(userId)) { + (oldFrsMap(userId), productMixerMap(userId)) match { + case (oldFrsUser: CandidateUser, productMixerUser: CandidateUser) => + compareStats.counter("matching-user-count").incr() + compareUser(oldFrsUser, productMixerUser, compareStats) + case _ => + compareStats.counter("unknown-user-type-count").incr() + } + } else { + compareStats.counter("missing-user-count").incr() + } + }) + } + + private def compareTopNResults( + n: Int, + frsOldPipelineResponse: RecommendationResponse, + frsProductMixerResponse: RecommendationResponse, + compareStats: StatsReceiver + ): Unit = { + if (frsOldPipelineResponse.recommendations.size >= n && frsProductMixerResponse.recommendations.size >= n) { + val oldFrsPipelineFirstN = frsOldPipelineResponse.recommendations.take(n).map(_.id) + val productMixerPipelineFirstN = frsProductMixerResponse.recommendations.take(n).map(_.id) + + if (oldFrsPipelineFirstN.sorted == productMixerPipelineFirstN.sorted) + compareStats.counter(s"first-$n-sorted-equal-ids").incr() + if (oldFrsPipelineFirstN == productMixerPipelineFirstN) + compareStats.counter(s"first-$n-unsorted-ids-equal").incr() + else + compareStats.counter(s"first-$n-unsorted-ids-unequal").incr() + } + } + + private def compareUser( + oldFrsUser: CandidateUser, + productMixerUser: CandidateUser, + stats: StatsReceiver + ): Unit = { + val userStats = stats.scope("matching-user") + + if (oldFrsUser.score != productMixerUser.score) + userStats.counter("mismatch-score").incr() + if (oldFrsUser.reason != productMixerUser.reason) + userStats.counter("mismatch-reason").incr() + if (oldFrsUser.userCandidateSourceDetails != productMixerUser.userCandidateSourceDetails) + userStats.counter("mismatch-userCandidateSourceDetails").incr() + if (oldFrsUser.adMetadata != productMixerUser.adMetadata) + userStats.counter("mismatch-adMetadata").incr() + if (oldFrsUser.trackingToken != productMixerUser.trackingToken) + userStats.counter("mismatch-trackingToken").incr() + if (oldFrsUser.dataRecord != productMixerUser.dataRecord) + userStats.counter("mismatch-dataRecord").incr() + if (oldFrsUser.scores != productMixerUser.scores) + userStats.counter("mismatch-scores").incr() + if (oldFrsUser.infoPerRankingStage != productMixerUser.infoPerRankingStage) + userStats.counter("mismatch-infoPerRankingStage").incr() + if (oldFrsUser.params != productMixerUser.params) + userStats.counter("mismatch-params").incr() + if (oldFrsUser.engagements != productMixerUser.engagements) + userStats.counter("mismatch-engagements").incr() + if (oldFrsUser.recommendationFlowIdentifier != productMixerUser.recommendationFlowIdentifier) + userStats.counter("mismatch-recommendationFlowIdentifier").incr() + } +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/services/ProductPipelineSelectorConfig.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/services/ProductPipelineSelectorConfig.scala new file mode 100644 index 0000000000..a1cac3316f --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/services/ProductPipelineSelectorConfig.scala @@ -0,0 +1,19 @@ +package com.twitter.follow_recommendations.services + +import com.twitter.follow_recommendations.common.models.DisplayLocation +import com.twitter.timelines.configapi.FSParam +import com.twitter.timelines.configapi.Param +import javax.inject.Singleton + +@Singleton +class ProductPipelineSelectorConfig { + private val paramsMap: Map[DisplayLocation, DarkReadAndExpParams] = Map.empty + + def getDarkReadAndExpParams( + displayLocation: DisplayLocation + ): Option[DarkReadAndExpParams] = { + paramsMap.get(displayLocation) + } +} + +case class DarkReadAndExpParams(darkReadParam: Param[Boolean], expParam: FSParam[Boolean]) diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/services/ProductRecommenderService.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/services/ProductRecommenderService.scala new file mode 100644 index 0000000000..967790a080 --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/services/ProductRecommenderService.scala @@ -0,0 +1,72 @@ +package com.twitter.follow_recommendations.services + +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.follow_recommendations.common.base.StatsUtil +import com.twitter.follow_recommendations.common.models.Recommendation +import com.twitter.follow_recommendations.models.RecommendationRequest +import com.twitter.follow_recommendations.products.common.ProductRegistry +import com.twitter.follow_recommendations.products.common.ProductRequest +import com.twitter.stitch.Stitch +import com.twitter.follow_recommendations.configapi.params.GlobalParams.EnableWhoToFollowProducts +import com.twitter.timelines.configapi.Params +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class ProductRecommenderService @Inject() ( + productRegistry: ProductRegistry, + statsReceiver: StatsReceiver) { + + private val stats = statsReceiver.scope("ProductRecommenderService") + + def getRecommendations( + request: RecommendationRequest, + params: Params + ): Stitch[Seq[Recommendation]] = { + val displayLocation = request.displayLocation + val displayLocationStatName = displayLocation.toString + val locationStats = stats.scope(displayLocationStatName) + val loggedInOrOutStats = if (request.clientContext.userId.isDefined) { + stats.scope("logged_in").scope(displayLocationStatName) + } else { + stats.scope("logged_out").scope(displayLocationStatName) + } + + loggedInOrOutStats.counter("requests").incr() + val product = productRegistry.getProductByDisplayLocation(displayLocation) + val productRequest = ProductRequest(request, params) + val productEnabledStitch = + StatsUtil.profileStitch(product.enabled(productRequest), locationStats.scope("enabled")) + productEnabledStitch.flatMap { productEnabled => + if (productEnabled && params(EnableWhoToFollowProducts)) { + loggedInOrOutStats.counter("enabled").incr() + val stitch = for { + workflows <- StatsUtil.profileStitch( + product.selectWorkflows(productRequest), + locationStats.scope("select_workflows")) + workflowRecos <- StatsUtil.profileStitch( + Stitch.collect( + workflows.map(_.process(productRequest).map(_.result.getOrElse(Seq.empty)))), + locationStats.scope("execute_workflows") + ) + blendedCandidates <- StatsUtil.profileStitch( + product.blender.transform(productRequest, workflowRecos.flatten), + locationStats.scope("blend_results")) + resultsTransformer <- StatsUtil.profileStitch( + product.resultsTransformer(productRequest), + locationStats.scope("results_transformer")) + transformedCandidates <- StatsUtil.profileStitch( + resultsTransformer.transform(productRequest, blendedCandidates), + locationStats.scope("execute_results_transformer")) + } yield { + transformedCandidates + } + StatsUtil.profileStitchResults[Seq[Recommendation]](stitch, locationStats, _.size) + } else { + loggedInOrOutStats.counter("disabled").incr() + locationStats.counter("disabled_product").incr() + Stitch.Nil + } + } + } +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/services/RecommendationsService.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/services/RecommendationsService.scala new file mode 100644 index 0000000000..e4bc1e3c0d --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/services/RecommendationsService.scala @@ -0,0 +1,28 @@ +package com.twitter.follow_recommendations.services + +import com.twitter.follow_recommendations.configapi.deciders.DeciderParams +import com.twitter.follow_recommendations.logging.FrsLogger +import com.twitter.follow_recommendations.models.RecommendationRequest +import com.twitter.follow_recommendations.models.RecommendationResponse +import com.twitter.stitch.Stitch +import com.twitter.timelines.configapi.Params +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class RecommendationsService @Inject() ( + productRecommenderService: ProductRecommenderService, + resultLogger: FrsLogger) { + def get(request: RecommendationRequest, params: Params): Stitch[RecommendationResponse] = { + if (params(DeciderParams.EnableRecommendations)) { + productRecommenderService + .getRecommendations(request, params).map(RecommendationResponse).onSuccess { response => + if (resultLogger.shouldLog(request.debugParams)) { + resultLogger.logRecommendationResult(request, response) + } + } + } else { + Stitch.value(RecommendationResponse(Nil)) + } + } +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/services/UserScoringService.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/services/UserScoringService.scala new file mode 100644 index 0000000000..b3a8c66649 --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/services/UserScoringService.scala @@ -0,0 +1,84 @@ +package com.twitter.follow_recommendations.services + +import com.twitter.finagle.stats.Counter +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.follow_recommendations.common.base.StatsUtil.profileStitchSeqResults +import com.twitter.follow_recommendations.common.clients.impression_store.WtfImpressionStore +import com.twitter.follow_recommendations.common.clients.socialgraph.SocialGraphClient +import com.twitter.follow_recommendations.common.rankers.ml_ranker.ranking.HydrateFeaturesTransform +import com.twitter.follow_recommendations.common.rankers.ml_ranker.ranking.MlRanker +import com.twitter.follow_recommendations.common.utils.RescueWithStatsUtils.rescueWithStats +import com.twitter.follow_recommendations.configapi.deciders.DeciderParams +import com.twitter.follow_recommendations.logging.FrsLogger +import com.twitter.follow_recommendations.models.ScoringUserRequest +import com.twitter.follow_recommendations.models.ScoringUserResponse +import com.twitter.stitch.Stitch +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class UserScoringService @Inject() ( + socialGraph: SocialGraphClient, + wtfImpressionStore: WtfImpressionStore, + hydrateFeaturesTransform: HydrateFeaturesTransform[ScoringUserRequest], + mlRanker: MlRanker[ScoringUserRequest], + resultLogger: FrsLogger, + stats: StatsReceiver) { + + private val scopedStats: StatsReceiver = stats.scope(this.getClass.getSimpleName) + private val disabledCounter: Counter = scopedStats.counter("disabled") + + def get(request: ScoringUserRequest): Stitch[ScoringUserResponse] = { + if (request.params(DeciderParams.EnableScoreUserCandidates)) { + val hydratedRequest = hydrate(request) + val candidatesStitch = hydratedRequest.flatMap { req => + hydrateFeaturesTransform.transform(req, request.candidates).flatMap { + candidateWithFeatures => + mlRanker.rank(req, candidateWithFeatures) + } + } + profileStitchSeqResults(candidatesStitch, scopedStats) + .map(ScoringUserResponse) + .onSuccess { response => + if (resultLogger.shouldLog(request.debugParams)) { + resultLogger.logScoringResult(request, response) + } + } + } else { + disabledCounter.incr() + Stitch.value(ScoringUserResponse(Nil)) + } + } + + private def hydrate(request: ScoringUserRequest): Stitch[ScoringUserRequest] = { + val allStitches = Stitch.collect(request.clientContext.userId.map { userId => + val recentFollowedUserIdsStitch = + rescueWithStats( + socialGraph.getRecentFollowedUserIds(userId), + stats, + "recentFollowedUserIds") + val recentFollowedByUserIdsStitch = + rescueWithStats( + socialGraph.getRecentFollowedByUserIds(userId), + stats, + "recentFollowedByUserIds") + val wtfImpressionsStitch = + rescueWithStats( + wtfImpressionStore.get(userId, request.displayLocation), + stats, + "wtfImpressions") + Stitch.join(recentFollowedUserIdsStitch, recentFollowedByUserIdsStitch, wtfImpressionsStitch) + }) + allStitches.map { + case Some((recentFollowedUserIds, recentFollowedByUserIds, wtfImpressions)) => + request.copy( + recentFollowedUserIds = + if (recentFollowedUserIds.isEmpty) None else Some(recentFollowedUserIds), + recentFollowedByUserIds = + if (recentFollowedByUserIds.isEmpty) None else Some(recentFollowedByUserIds), + wtfImpressions = if (wtfImpressions.isEmpty) None else Some(wtfImpressions) + ) + case _ => request + } + } +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/services/exceptions/BUILD b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/services/exceptions/BUILD new file mode 100644 index 0000000000..8b2c2d0412 --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/services/exceptions/BUILD @@ -0,0 +1,14 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + tags = ["bazel-compatible"], + dependencies = [ + "finatra/thrift/src/main/scala/com/twitter/finatra/thrift", + "finatra/thrift/src/main/scala/com/twitter/finatra/thrift:controller", + "finatra/thrift/src/main/scala/com/twitter/finatra/thrift/exceptions", + "finatra/thrift/src/main/scala/com/twitter/finatra/thrift/filters", + "finatra/thrift/src/main/scala/com/twitter/finatra/thrift/modules", + "finatra/thrift/src/main/scala/com/twitter/finatra/thrift/response", + "finatra/thrift/src/main/scala/com/twitter/finatra/thrift/routing", + ], +) diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/services/exceptions/UnknownExceptionMapper.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/services/exceptions/UnknownExceptionMapper.scala new file mode 100644 index 0000000000..f3a09a6d7d --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/services/exceptions/UnknownExceptionMapper.scala @@ -0,0 +1,18 @@ +package com.twitter.follow_recommendations.service.exceptions + +import com.twitter.finatra.thrift.exceptions.ExceptionMapper +import com.twitter.inject.Logging +import com.twitter.util.Future +import javax.inject.Singleton + +@Singleton +class UnknownLoggingExceptionMapper extends ExceptionMapper[Exception, Throwable] with Logging { + def handleException(throwable: Exception): Future[Throwable] = { + error( + s"Unmapped Exception: ${throwable.getMessage} - ${throwable.getStackTrace.mkString(", \n\t")}", + throwable + ) + + Future.exception(throwable) + } +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/utils/BUILD b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/utils/BUILD new file mode 100644 index 0000000000..e92976ad87 --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/utils/BUILD @@ -0,0 +1,29 @@ +scala_library( + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "finatra/inject/inject-app/src/main/scala", + "finatra/inject/inject-core/src/main/scala", + "finatra/inject/inject-server/src/main/scala", + "finatra/inject/inject-thrift-client", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/base", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/addressbook", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/geo", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/ppmi_locale_follow", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/recent_engagement", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/sims_expansion", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/socialgraph", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/stp", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/triangular_loops", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/candidate_sources/two_hop_random_walk", + "follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/feature_hydration/sources", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/configapi/params", + "follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/models", + "twitter-server/server/src/main/scala", + "util/util-app/src/main/scala", + "util/util-core:scala", + "util/util-slf4j-api/src/main/scala", + ], +) diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/utils/CandidateSourceHoldbackUtil.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/utils/CandidateSourceHoldbackUtil.scala new file mode 100644 index 0000000000..60a28f1a8e --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/utils/CandidateSourceHoldbackUtil.scala @@ -0,0 +1,82 @@ +package com.twitter.follow_recommendations.utils + +import com.twitter.follow_recommendations.common.candidate_sources.addressbook._ +import com.twitter.follow_recommendations.common.candidate_sources.geo.PopCountrySource +import com.twitter.follow_recommendations.common.candidate_sources.geo.PopCountryBackFillSource +import com.twitter.follow_recommendations.common.candidate_sources.geo.PopGeoSource +import com.twitter.follow_recommendations.common.candidate_sources.geo.PopGeohashSource +import com.twitter.follow_recommendations.common.candidate_sources.ppmi_locale_follow.PPMILocaleFollowSource +import com.twitter.follow_recommendations.common.candidate_sources.recent_engagement.RecentEngagementNonDirectFollowSource +import com.twitter.follow_recommendations.common.candidate_sources.sims.SwitchingSimsSource +import com.twitter.follow_recommendations.common.candidate_sources.sims_expansion.RecentEngagementSimilarUsersSource +import com.twitter.follow_recommendations.common.candidate_sources.sims_expansion.RecentFollowingSimilarUsersSource +import com.twitter.follow_recommendations.common.candidate_sources.sims_expansion.RecentStrongEngagementDirectFollowSimilarUsersSource +import com.twitter.follow_recommendations.common.candidate_sources.socialgraph.RecentFollowingRecentFollowingExpansionSource +import com.twitter.follow_recommendations.common.candidate_sources.stp.MutualFollowStrongTiePredictionSource +import com.twitter.follow_recommendations.common.candidate_sources.stp.OfflineStrongTiePredictionSource +import com.twitter.follow_recommendations.common.candidate_sources.stp.BaseOnlineSTPSource +import com.twitter.follow_recommendations.common.candidate_sources.stp.SocialProofEnforcedOfflineStrongTiePredictionSource +import com.twitter.follow_recommendations.common.candidate_sources.triangular_loops.TriangularLoopsSource +import com.twitter.follow_recommendations.common.candidate_sources.two_hop_random_walk.TwoHopRandomWalkSource +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.follow_recommendations.configapi.params.GlobalParams +import com.twitter.follow_recommendations.models.CandidateSourceType +import com.twitter.product_mixer.core.functional_component.candidate_source.CandidateSource +import com.twitter.product_mixer.core.model.common.identifier.CandidateSourceIdentifier +import com.twitter.timelines.configapi.HasParams + +trait CandidateSourceHoldbackUtil { + import CandidateSourceHoldbackUtil._ + def filterCandidateSources[T <: HasParams]( + request: T, + sources: Seq[CandidateSource[T, CandidateUser]] + ): Seq[CandidateSource[T, CandidateUser]] = { + val typeToFilter = request.params(GlobalParams.CandidateSourcesToFilter) + val sourcesToFilter = CandidateSourceTypeToMap.get(typeToFilter).getOrElse(Set.empty) + sources.filterNot { source => sourcesToFilter.contains(source.identifier) } + } +} + +object CandidateSourceHoldbackUtil { + final val ContextualActivityCandidateSourceIds: Set[CandidateSourceIdentifier] = + Set( + RecentFollowingSimilarUsersSource.Identifier, + RecentEngagementNonDirectFollowSource.Identifier, + RecentEngagementSimilarUsersSource.Identifier, + RecentStrongEngagementDirectFollowSimilarUsersSource.Identifier, + SwitchingSimsSource.Identifier, + ) + + final val SocialCandidateSourceIds: Set[CandidateSourceIdentifier] = + Set( + ForwardEmailBookSource.Identifier, + ForwardPhoneBookSource.Identifier, + ReverseEmailBookSource.Identifier, + ReversePhoneBookSource.Identifier, + RecentFollowingRecentFollowingExpansionSource.Identifier, + BaseOnlineSTPSource.Identifier, + MutualFollowStrongTiePredictionSource.Identifier, + OfflineStrongTiePredictionSource.Identifier, + SocialProofEnforcedOfflineStrongTiePredictionSource.Identifier, + TriangularLoopsSource.Identifier, + TwoHopRandomWalkSource.Identifier + ) + + final val GeoCandidateSourceIds: Set[CandidateSourceIdentifier] = + Set( + PPMILocaleFollowSource.Identifier, + PopCountrySource.Identifier, + PopGeohashSource.Identifier, + PopCountryBackFillSource.Identifier, + PopGeoSource.Identifier, + ) + + final val CandidateSourceTypeToMap: Map[CandidateSourceType.Value, Set[ + CandidateSourceIdentifier + ]] = + Map( + CandidateSourceType.Social -> SocialCandidateSourceIds, + CandidateSourceType.ActivityContextual -> ContextualActivityCandidateSourceIds, + CandidateSourceType.GeoAndInterests -> GeoCandidateSourceIds + ) +} diff --git a/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/utils/RecommendationFlowBaseSideEffectsUtil.scala b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/utils/RecommendationFlowBaseSideEffectsUtil.scala new file mode 100644 index 0000000000..9304fb398b --- /dev/null +++ b/follow-recommendations-service/server/src/main/scala/com/twitter/follow_recommendations/utils/RecommendationFlowBaseSideEffectsUtil.scala @@ -0,0 +1,121 @@ +package com.twitter.follow_recommendations.utils + +import com.twitter.follow_recommendations.common.base.RecommendationFlow +import com.twitter.follow_recommendations.common.base.SideEffectsUtil +import com.twitter.follow_recommendations.common.models.CandidateUser +import com.twitter.product_mixer.core.functional_component.candidate_source.CandidateSource +import com.twitter.product_mixer.core.model.common.identifier.CandidateSourceIdentifier +import com.twitter.product_mixer.core.model.marshalling.request.HasClientContext +import com.twitter.snowflake.id.SnowflakeId +import com.twitter.stitch.Stitch + +trait RecommendationFlowBaseSideEffectsUtil[Target <: HasClientContext, Candidate <: CandidateUser] + extends SideEffectsUtil[Target, Candidate] { + recommendationFlow: RecommendationFlow[Target, Candidate] => + + override def applySideEffects( + target: Target, + candidateSources: Seq[CandidateSource[Target, Candidate]], + candidatesFromCandidateSources: Seq[Candidate], + mergedCandidates: Seq[Candidate], + filteredCandidates: Seq[Candidate], + rankedCandidates: Seq[Candidate], + transformedCandidates: Seq[Candidate], + truncatedCandidates: Seq[Candidate], + results: Seq[Candidate] + ): Stitch[Unit] = { + Stitch.async( + Stitch.collect( + Seq( + applySideEffectsCandidateSourceCandidates( + target, + candidateSources, + candidatesFromCandidateSources), + applySideEffectsMergedCandidates(target, mergedCandidates), + applySideEffectsFilteredCandidates(target, filteredCandidates), + applySideEffectsRankedCandidates(target, rankedCandidates), + applySideEffectsTransformedCandidates(target, transformedCandidates), + applySideEffectsTruncatedCandidates(target, truncatedCandidates), + applySideEffectsResults(target, results) + ) + )) + } + + /* + In subclasses, override functions below to apply custom side effects at each step in pipeline. + Call super.applySideEffectsXYZ to scribe basic scribes implemented in this parent class + */ + def applySideEffectsCandidateSourceCandidates( + target: Target, + candidateSources: Seq[CandidateSource[Target, Candidate]], + candidatesFromCandidateSources: Seq[Candidate] + ): Stitch[Unit] = { + val candidatesGroupedByCandidateSources = + candidatesFromCandidateSources.groupBy( + _.getPrimaryCandidateSource.getOrElse(CandidateSourceIdentifier("NoCandidateSource"))) + + target.getOptionalUserId match { + case Some(userId) => + val userAgeOpt = SnowflakeId.timeFromIdOpt(userId).map(_.untilNow.inDays) + userAgeOpt match { + case Some(userAge) if userAge <= 30 => + candidateSources.map { candidateSource => + { + val candidateSourceStats = statsReceiver.scope(candidateSource.identifier.name) + + val isEmpty = + !candidatesGroupedByCandidateSources.keySet.contains(candidateSource.identifier) + + if (userAge <= 1) + candidateSourceStats + .scope("user_age", "1", "empty").counter(isEmpty.toString).incr() + if (userAge <= 7) + candidateSourceStats + .scope("user_age", "7", "empty").counter(isEmpty.toString).incr() + if (userAge <= 30) + candidateSourceStats + .scope("user_age", "30", "empty").counter(isEmpty.toString).incr() + } + } + case _ => Nil + } + case None => Nil + } + Stitch.Unit + } + + def applySideEffectsBaseCandidates( + target: Target, + candidates: Seq[Candidate] + ): Stitch[Unit] = Stitch.Unit + + def applySideEffectsMergedCandidates( + target: Target, + candidates: Seq[Candidate] + ): Stitch[Unit] = applySideEffectsBaseCandidates(target, candidates) + + def applySideEffectsFilteredCandidates( + target: Target, + candidates: Seq[Candidate] + ): Stitch[Unit] = applySideEffectsBaseCandidates(target, candidates) + + def applySideEffectsRankedCandidates( + target: Target, + candidates: Seq[Candidate] + ): Stitch[Unit] = applySideEffectsBaseCandidates(target, candidates) + + def applySideEffectsTransformedCandidates( + target: Target, + candidates: Seq[Candidate] + ): Stitch[Unit] = applySideEffectsBaseCandidates(target, candidates) + + def applySideEffectsTruncatedCandidates( + target: Target, + candidates: Seq[Candidate] + ): Stitch[Unit] = applySideEffectsBaseCandidates(target, candidates) + + def applySideEffectsResults( + target: Target, + candidates: Seq[Candidate] + ): Stitch[Unit] = applySideEffectsBaseCandidates(target, candidates) +} diff --git a/follow-recommendations-service/thrift/src/main/thrift/BUILD b/follow-recommendations-service/thrift/src/main/thrift/BUILD new file mode 100644 index 0000000000..e5cbd19cf7 --- /dev/null +++ b/follow-recommendations-service/thrift/src/main/thrift/BUILD @@ -0,0 +1,21 @@ +create_thrift_libraries( + base_name = "thrift", + sources = ["*.thrift"], + platform = "java8", + tags = ["bazel-compatible"], + dependency_roots = [ + "finatra-internal/thrift/src/main/thrift", + "follow-recommendations-service/thrift/src/main/thrift/logging:thrift", + "product-mixer/core/src/main/thrift/com/twitter/product_mixer/core:thrift", + "src/thrift/com/twitter/ads/adserver:adserver_common", + "src/thrift/com/twitter/ml/api:data", + "src/thrift/com/twitter/suggests/controller_data", + ], + generate_languages = [ + "java", + "scala", + "strato", + ], + provides_java_name = "follow-recommendations-java", + provides_scala_name = "follow-recommendations-scala", +) diff --git a/follow-recommendations-service/thrift/src/main/thrift/assembler.thrift b/follow-recommendations-service/thrift/src/main/thrift/assembler.thrift new file mode 100644 index 0000000000..eb782d0fe2 --- /dev/null +++ b/follow-recommendations-service/thrift/src/main/thrift/assembler.thrift @@ -0,0 +1,42 @@ +namespace java com.twitter.follow_recommendations.thriftjava +#@namespace scala com.twitter.follow_recommendations.thriftscala +#@namespace strato com.twitter.follow_recommendations + +struct Header { + 1: required Title title +} + +struct Title { + 1: required string text +} + +struct Footer { + 1: optional Action action +} + +struct Action { + 1: required string text + 2: required string actionURL +} + +struct UserList { + 1: required bool userBioEnabled + 2: required bool userBioTruncated + 3: optional i64 userBioMaxLines + 4: optional FeedbackAction feedbackAction +} + +struct Carousel { + 1: optional FeedbackAction feedbackAction +} + +union WTFPresentation { + 1: UserList userBioList + 2: Carousel carousel +} + +struct DismissUserId {} + +union FeedbackAction { + 1: DismissUserId dismissUserId +} diff --git a/follow-recommendations-service/thrift/src/main/thrift/client_context.thrift b/follow-recommendations-service/thrift/src/main/thrift/client_context.thrift new file mode 100644 index 0000000000..adbdc407a0 --- /dev/null +++ b/follow-recommendations-service/thrift/src/main/thrift/client_context.thrift @@ -0,0 +1,19 @@ +namespace java com.twitter.follow_recommendations.thriftjava +#@namespace scala com.twitter.follow_recommendations.thriftscala +#@namespace strato com.twitter.follow_recommendations + +// Caller/Client level specific context (e.g, user id/guest id/app id). +struct ClientContext { + 1: optional i64 userId(personalDataType='UserId') + 2: optional i64 guestId(personalDataType='GuestId') + 3: optional i64 appId(personalDataType='AppId') + 4: optional string ipAddress(personalDataType='IpAddress') + 5: optional string userAgent(personalDataType='UserAgent') + 6: optional string countryCode(personalDataType='InferredCountry') + 7: optional string languageCode(personalDataType='InferredLanguage') + 9: optional bool isTwoffice(personalDataType='InferredLocation') + 10: optional set userRoles + 11: optional string deviceId(personalDataType='DeviceId') + 12: optional i64 guestIdAds(personalDataType='GuestId') + 13: optional i64 guestIdMarketing(personalDataType='GuestId') +}(hasPersonalData='true') diff --git a/follow-recommendations-service/thrift/src/main/thrift/debug.thrift b/follow-recommendations-service/thrift/src/main/thrift/debug.thrift new file mode 100644 index 0000000000..a41c59114d --- /dev/null +++ b/follow-recommendations-service/thrift/src/main/thrift/debug.thrift @@ -0,0 +1,73 @@ +namespace java com.twitter.follow_recommendations.thriftjava +#@namespace scala com.twitter.follow_recommendations.thriftscala +#@namespace strato com.twitter.follow_recommendation + +// These are broken into their own union +// because we can have features that are +// complex flavors of these (such as Seq) +union PrimitiveFeatureValue { + 1: i32 intValue + 2: i64 longValue + 3: string strValue + 4: bool boolValue +} + +union FeatureValue { + 1: PrimitiveFeatureValue primitiveValue +} + +struct DebugParams { + 1: optional map featureOverrides + 2: optional i64 randomizationSeed + 3: optional bool includeDebugInfoInResults + 4: optional bool doNotLog +} + +enum DebugCandidateSourceIdentifier { + UTT_INTERESTS_RELATED_USERS_SOURCE = 0 + UTT_PRODUCER_EXPANSION_SOURCE = 1 + UTT_SEED_ACCOUNT_SOURCE = 2 + BYF_USER_FOLLOW_CLUSTER_SIMS_SOURCE = 3 + BYF_USER_FOLLOW_CLUSTER_SOURCE = 4 + USER_FOLLOW_CLUSTER_SOURCE = 5 + RECENT_SEARCH_BASED_SOURCE = 6 + PEOPLE_ACTIVITY_RECENT_ENGAGEMENT_SOURCE = 7 + PEOPLE_ACTIVITY_RECENT_ENGAGEMENT_SIMS_SOURCE = 8, + REVERSE_PHONE_BOOK_SOURCE = 9, + REVERSE_EMAIL_BOOK_SOURCE = 10, + SIMS_DEBUG_STORE = 11, + UTT_PRODUCER_ONLINE_MBCG_SOURCE = 12, + BONUS_FOLLOW_CONDITIONAL_ENGAGEMENT_STORE = 13, + // 14 (BONUS_FOLLOW_PMI_STORE) was deleted as it's not used anymore + FOLLOW2VEC_NEAREST_NEIGHBORS_STORE = 15, + OFFLINE_STP = 16, + OFFLINE_STP_BIG = 17, + OFFLINE_MUTUAL_FOLLOW_EXPANSION = 18, + REPEATED_PROFILE_VISITS = 19, + TIME_DECAY_FOLLOW2VEC_NEAREST_NEIGHBORS_STORE = 20, + LINEAR_REGRESSION_FOLLOW2VEC_NEAREST_NEIGHBORS_STORE = 21, + REAL_GRAPH_EXPANSION_SOURCE = 22, + RELATABLE_ACCOUNTS_BY_INTEREST = 23, + EMAIL_TWEET_CLICK = 24, + GOOD_TWEET_CLICK_ENGAGEMENTS = 25, + ENGAGED_FOLLOWER_RATIO = 26, + TWEET_SHARE_ENGAGEMENTS = 27, + BULK_FRIEND_FOLLOWS = 28, + REAL_GRAPH_OON_V2_SOURCE = 30, + CROWD_SEARCH_ACCOUNTS = 31, + POP_GEOHASH = 32, + POP_COUNTRY = 33, + POP_COUNTRY_BACKFILL = 34, + TWEET_SHARER_TO_SHARE_RECIPIENT_ENGAGEMENTS = 35, + TWEET_AUTHOR_TO_SHARE_RECIPIENT_ENGAGEMENTS = 36, + BULK_FRIEND_FOLLOWS_NEW_USER = 37, + ONLINE_STP_EPSCORER = 38, + ORGANIC_FOLLOW_ACCOUNTS = 39, + NUX_LO_HISTORY = 40, + TRAFFIC_ATTRIBUTION_ACCOUNTS = 41, + ONLINE_STP_RAW_ADDRESS_BOOK = 42, + POP_GEOHASH_QUALITY_FOLLOW = 43, + NOTIFICATION_ENGAGEMENT = 44, + EFR_BY_WORLDWIDE_PICTURE_PRODUCER = 45, + POP_GEOHASH_REAL_GRAPH = 46, +} diff --git a/follow-recommendations-service/thrift/src/main/thrift/display_context.thrift b/follow-recommendations-service/thrift/src/main/thrift/display_context.thrift new file mode 100644 index 0000000000..cfd613b717 --- /dev/null +++ b/follow-recommendations-service/thrift/src/main/thrift/display_context.thrift @@ -0,0 +1,62 @@ +include "flows.thrift" +include "recently_engaged_user_id.thrift" + +namespace java com.twitter.follow_recommendations.thriftjava +#@namespace scala com.twitter.follow_recommendations.thriftscala +#@namespace strato com.twitter.follow_recommendations + +struct Profile { + 1: required i64 profileId(personalDataType='UserId') +}(hasPersonalData='true') + +struct Search { + 1: required string searchQuery(personalDataType='SearchQuery') +}(hasPersonalData='true') + +struct Rux { + 1: required i64 focalAuthorId(personalDataType='UserId') +}(hasPersonalData='true') + +struct Topic { + 1: required i64 topicId(personalDataType = 'TopicFollow') +}(hasPersonalData='true') + +struct ReactiveFollow { + 1: required list followedUserIds(personalDataType='UserId') +}(hasPersonalData='true') + +struct NuxInterests { + 1: optional flows.FlowContext flowContext // set for recommendation inside an interactive flow + 2: optional list uttInterestIds // if provided, we use these interestIds for generating candidates instead of for example fetching user selected interests +}(hasPersonalData='true') + +struct AdCampaignTarget { + 1: required list similarToUserIds(personalDataType='UserId') +}(hasPersonalData='true') + +struct ConnectTab { + 1: required list byfSeedUserIds(personalDataType='UserId') + 2: required list similarToUserIds(personalDataType='UserId') + 3: required list recentlyEngagedUserIds +}(hasPersonalData='true') + +struct SimilarToUser { + 1: required i64 similarToUserId(personalDataType='UserId') +}(hasPersonalData='true') + +struct PostNuxFollowTask { + 1: optional flows.FlowContext flowContext // set for recommendation inside an interactive flow +}(hasPersonalData='true') + +union DisplayContext { + 1: Profile profile + 2: Search search + 3: Rux rux + 4: Topic topic + 5: ReactiveFollow reactiveFollow + 6: NuxInterests nuxInterests + 7: AdCampaignTarget adCampaignTarget + 8: ConnectTab connectTab + 9: SimilarToUser similarToUser + 10: PostNuxFollowTask postNuxFollowTask +}(hasPersonalData='true') diff --git a/follow-recommendations-service/thrift/src/main/thrift/display_location.thrift b/follow-recommendations-service/thrift/src/main/thrift/display_location.thrift new file mode 100644 index 0000000000..d94b9842e4 --- /dev/null +++ b/follow-recommendations-service/thrift/src/main/thrift/display_location.thrift @@ -0,0 +1,55 @@ +namespace java com.twitter.follow_recommendations.thriftjava +#@namespace scala com.twitter.follow_recommendations.thriftscala +#@namespace strato com.twitter.follow_recommendations + +enum DisplayLocation { + SIDEBAR = 0 + PROFILE_SIDEBAR = 2 + CLUSTER_FOLLOW = 7 + NEW_USER_SARUS_BACKFILL = 12 + PROFILE_DEVICE_FOLLOW = 23 + RECOS_BACKFILL = 32 + HOME_TIMELINE = 39 # HOME_TIMELINE_WTF in Hermit + PROFILE_TOP_FOLLOWING = 42 + PROFILE_TOP_FOLLOWERS = 43 + PEOPLE_PLUS_PLUS = 47 + EXPLORE_TAB = 57 + MagicRecs = 59 # Account recommendation in notification + AB_UPLOAD_INJECTION = 60 + /** + * To prevent setting 2 display locations with the same index in FRS. + * + * The display location should be added to the following files: + * - follow-recommendations-service/thrift/src/main/thrift/display_location.thrift + * - follow-recommendations-service/thrift/src/main/thrift/logging/display_location.thrift + * - follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/DisplayLocation.scala + */ + CAMPAIGN_FORM = 61 + RUX_LANDING_PAGE = 62 + PROFILE_BONUS_FOLLOW = 63 + ELECTION_EXPLORE_WTF = 64 + HTL_BONUS_FOLLOW = 65 + TOPIC_LANDING_PAGE_HEADER = 66 + NUX_PYMK = 67 + NUX_INTERESTS = 68 + REACTIVE_FOLLOW = 69 + RUX_PYMK = 70 + INDIA_COVID19_CURATED_ACCOUNTS_WTF = 71 + NUX_TOPIC_BONUS_FOLLOW = 72 + TWEET_NOTIFICATION_RECS = 73 + HTL_SPACE_HOSTS = 74 + POST_NUX_FOLLOW_TASK = 75 + TOPIC_LANDING_PAGE = 76 + USER_TYPEAHEAD_PREFETCH = 77 + HOME_TIMELINE_RELATABLE_ACCOUNTS = 78 + NUX_GEO_CATEGORY = 79 + NUX_INTERESTS_CATEGORY = 80 + NUX_PYMK_CATEGORY = 81 + TOP_ARTICLES = 82 + HOME_TIMELINE_TWEET_RECS = 83 + HTL_BULK_FRIEND_FOLLOWS = 84 + NUX_AUTO_FOLLOW = 85 + SEARCH_BONUS_FOLLOW = 86 + CONTENT_RECOMMENDER = 87 + HOME_TIMELINE_REVERSE_CHRON = 88 +} diff --git a/follow-recommendations-service/thrift/src/main/thrift/engagementType.thrift b/follow-recommendations-service/thrift/src/main/thrift/engagementType.thrift new file mode 100644 index 0000000000..ef028d008f --- /dev/null +++ b/follow-recommendations-service/thrift/src/main/thrift/engagementType.thrift @@ -0,0 +1,11 @@ +namespace java com.twitter.follow_recommendations.thriftjava +#@namespace scala com.twitter.follow_recommendations.thriftscala +#@namespace strato com.twitter.follow_recommendations + +enum EngagementType { + Click = 0 + Like = 1 + Mention = 2 + Retweet = 3 + ProfileView = 4 +} diff --git a/follow-recommendations-service/thrift/src/main/thrift/flows.thrift b/follow-recommendations-service/thrift/src/main/thrift/flows.thrift new file mode 100644 index 0000000000..894ebf81ef --- /dev/null +++ b/follow-recommendations-service/thrift/src/main/thrift/flows.thrift @@ -0,0 +1,20 @@ +/* + * This file defines additional thrift objects that should be specified in FRS request for context of recommendation, specifically the previous recommendations / new interactions in an interactive flow (series of follow steps). These typically are sent from OCF + */ + +namespace java com.twitter.follow_recommendations.thriftjava +#@namespace scala com.twitter.follow_recommendations.thriftscala +#@namespace strato com.twitter.follow_recommendations + +struct FlowRecommendation { + 1: required i64 userId(personalDataType='UserId') +}(hasPersonalData='true') + +struct RecommendationStep { + 1: required list recommendations + 2: required set followedUserIds(personalDataType='UserId') +}(hasPersonalData='true') + +struct FlowContext { + 1: required list steps +}(hasPersonalData='true') diff --git a/follow-recommendations-service/thrift/src/main/thrift/follow-recommendations-service.thrift b/follow-recommendations-service/thrift/src/main/thrift/follow-recommendations-service.thrift new file mode 100644 index 0000000000..40aadc0b65 --- /dev/null +++ b/follow-recommendations-service/thrift/src/main/thrift/follow-recommendations-service.thrift @@ -0,0 +1,100 @@ +namespace java com.twitter.follow_recommendations.thriftjava +#@namespace scala com.twitter.follow_recommendations.thriftscala +#@namespace strato com.twitter.follow_recommendations + +include "assembler.thrift" +include "client_context.thrift" +include "debug.thrift" +include "display_context.thrift" +include "display_location.thrift" +include "recommendations.thrift" +include "recently_engaged_user_id.thrift" + +include "finatra-thrift/finatra_thrift_exceptions.thrift" +include "com/twitter/product_mixer/core/pipeline_execution_result.thrift" + +struct RecommendationRequest { + 1: required client_context.ClientContext clientContext + 2: required display_location.DisplayLocation displayLocation + 3: optional display_context.DisplayContext displayContext + // Max results to return + 4: optional i32 maxResults + // Cursor to continue returning results if any + 5: optional string cursor + // IDs of Content to exclude from recommendations + 6: optional list excludedIds(personalDataType='UserId') + // Whether to also get promoted content + 7: optional bool fetchPromotedContent + 8: optional debug.DebugParams debugParams + 9: optional string userLocationState(personalDataType='InferredLocation') +}(hasPersonalData='true') + + +struct RecommendationResponse { + 1: required list recommendations +}(hasPersonalData='true') + +// for scoring a list of candidates, while logging hydrated features +struct ScoringUserRequest { + 1: required client_context.ClientContext clientContext + 2: required display_location.DisplayLocation displayLocation + 3: required list candidates + 4: optional debug.DebugParams debugParams +}(hasPersonalData='true') + +struct ScoringUserResponse { + 1: required list candidates // empty for now +}(hasPersonalData='true') + +// for getting the list of candidates generated by a single candidate source +struct DebugCandidateSourceRequest { + 1: required client_context.ClientContext clientContext + 2: required debug.DebugCandidateSourceIdentifier candidateSource + 3: optional list uttInterestIds + 4: optional debug.DebugParams debugParams + 5: optional list recentlyFollowedUserIds + 6: optional list recentlyEngagedUserIds + 7: optional list byfSeedUserIds + 8: optional list similarToUserIds + 9: required bool applySgsPredicate + 10: optional i32 maxResults +}(hasPersonalData='true') + +service FollowRecommendationsThriftService { + RecommendationResponse getRecommendations(1: RecommendationRequest request) throws ( + 1: finatra_thrift_exceptions.ServerError serverError, + 2: finatra_thrift_exceptions.UnknownClientIdError unknownClientIdError, + 3: finatra_thrift_exceptions.NoClientIdError noClientIdError + ) + RecommendationDisplayResponse getRecommendationDisplayResponse(1: RecommendationRequest request) throws ( + 1: finatra_thrift_exceptions.ServerError serverError, + 2: finatra_thrift_exceptions.UnknownClientIdError unknownClientIdError, + 3: finatra_thrift_exceptions.NoClientIdError noClientIdError + ) + // temporary endpoint for feature hydration and logging for data collection. + ScoringUserResponse scoreUserCandidates(1: ScoringUserRequest request) throws ( + 1: finatra_thrift_exceptions.ServerError serverError, + 2: finatra_thrift_exceptions.UnknownClientIdError unknownClientIdError, + 3: finatra_thrift_exceptions.NoClientIdError noClientIdError + ) + // Debug endpoint for getting recommendations of a single candidate source. We can remove this endpoint when ProMix provide this functionality and we integrate with it. + RecommendationResponse debugCandidateSource(1: DebugCandidateSourceRequest request) throws ( + 1: finatra_thrift_exceptions.ServerError serverError, + 2: finatra_thrift_exceptions.UnknownClientIdError unknownClientIdError, + 3: finatra_thrift_exceptions.NoClientIdError noClientIdError + ) + + // Get the full execution log for a pipeline (used by our debugging tools) + pipeline_execution_result.PipelineExecutionResult executePipeline(1: RecommendationRequest request) throws ( + 1: finatra_thrift_exceptions.ServerError serverError, + 2: finatra_thrift_exceptions.UnknownClientIdError unknownClientIdError, + 3: finatra_thrift_exceptions.NoClientIdError noClientIdError + ) +} + +struct RecommendationDisplayResponse { + 1: required list hydratedRecommendation + 2: optional assembler.Header header + 3: optional assembler.Footer footer + 4: optional assembler.WTFPresentation wtfPresentation +}(hasPersonalData='true') diff --git a/follow-recommendations-service/thrift/src/main/thrift/follow_recommendations_serving_history.thrift b/follow-recommendations-service/thrift/src/main/thrift/follow_recommendations_serving_history.thrift new file mode 100644 index 0000000000..404b0ae29b --- /dev/null +++ b/follow-recommendations-service/thrift/src/main/thrift/follow_recommendations_serving_history.thrift @@ -0,0 +1,9 @@ +namespace java com.twitter.follow_recommendations.thriftjava +#@namespace scala com.twitter.follow_recommendations.thriftscala +#@namespace strato com.twitter.follow_recommendations + +// struct used for storing the history of computing and serving of recommendations to a user +struct FollowRecommendationsServingHistory { + 1: required i64 lastComputationTimeMs (personalDataType = 'PrivateTimestamp') + 2: required i64 lastServingTimeMs (personalDataType = 'PrivateTimestamp') +}(persisted='true', hasPersonalData='true') diff --git a/follow-recommendations-service/thrift/src/main/thrift/logging/BUILD b/follow-recommendations-service/thrift/src/main/thrift/logging/BUILD new file mode 100644 index 0000000000..2f769d4982 --- /dev/null +++ b/follow-recommendations-service/thrift/src/main/thrift/logging/BUILD @@ -0,0 +1,18 @@ +create_thrift_libraries( + base_name = "thrift", + sources = ["*.thrift"], + platform = "java8", + tags = ["bazel-compatible"], + dependency_roots = [ + "src/thrift/com/twitter/ads/adserver:adserver_common", + "src/thrift/com/twitter/ml/api:data", + "src/thrift/com/twitter/suggests/controller_data", + ], + generate_languages = [ + "java", + "scala", + "strato", + ], + provides_java_name = "follow-recommendations-logging-java", + provides_scala_name = "follow-recommendations-logging-scala", +) diff --git a/follow-recommendations-service/thrift/src/main/thrift/logging/client_context.thrift b/follow-recommendations-service/thrift/src/main/thrift/logging/client_context.thrift new file mode 100644 index 0000000000..2b6e454b22 --- /dev/null +++ b/follow-recommendations-service/thrift/src/main/thrift/logging/client_context.thrift @@ -0,0 +1,14 @@ +namespace java com.twitter.follow_recommendations.logging.thriftjava +#@namespace scala com.twitter.follow_recommendations.logging.thriftscala +#@namespace strato com.twitter.follow_recommendations.logging + +// Offline equal of ClientContext +struct OfflineClientContext { + 1: optional i64 userId(personalDataType='UserId') + 2: optional i64 guestId(personalDataType='GuestId') + 3: optional i64 appId(personalDataType='AppId') + 4: optional string countryCode(personalDataType='InferredCountry') + 5: optional string languageCode(personalDataType='InferredLanguage') + 6: optional i64 guestIdAds(personalDataType='GuestId') + 7: optional i64 guestIdMarketing(personalDataType='GuestId') +}(persisted='true', hasPersonalData='true') diff --git a/follow-recommendations-service/thrift/src/main/thrift/logging/debug.thrift b/follow-recommendations-service/thrift/src/main/thrift/logging/debug.thrift new file mode 100644 index 0000000000..882dca0055 --- /dev/null +++ b/follow-recommendations-service/thrift/src/main/thrift/logging/debug.thrift @@ -0,0 +1,8 @@ +namespace java com.twitter.follow_recommendations.logging.thriftjava +#@namespace scala com.twitter.follow_recommendations.logging.thriftscala +#@namespace strato com.twitter.follow_recommendation.logging + +// subset of DebugParams +struct OfflineDebugParams { + 1: optional i64 randomizationSeed // track if the request was randomly ranked or not +}(persisted='true', hasPersonalData='false') diff --git a/follow-recommendations-service/thrift/src/main/thrift/logging/display_context.thrift b/follow-recommendations-service/thrift/src/main/thrift/logging/display_context.thrift new file mode 100644 index 0000000000..c388500119 --- /dev/null +++ b/follow-recommendations-service/thrift/src/main/thrift/logging/display_context.thrift @@ -0,0 +1,66 @@ +include "logging/flows.thrift" +include "logging/recently_engaged_user_id.thrift" + +namespace java com.twitter.follow_recommendations.logging.thriftjava +#@namespace scala com.twitter.follow_recommendations.logging.thriftscala +#@namespace strato com.twitter.follow_recommendations.logging + +// Offline equal of Profile DisplayContext +struct OfflineProfile { + 1: required i64 profileId(personalDataType='UserId') +}(persisted='true', hasPersonalData='true') + +// Offline equal of Search DisplayContext +struct OfflineSearch { + 1: required string searchQuery(personalDataType='SearchQuery') +}(persisted='true', hasPersonalData='true') + +// Offline equal of Rux Landing Page DisplayContext +struct OfflineRux { + 1: required i64 focalAuthorId(personalDataType="UserId") +}(persisted='true', hasPersonalData='true') + +// Offline equal of Topic DisplayContext +struct OfflineTopic { + 1: required i64 topicId(personalDataType = 'TopicFollow') +}(persisted='true', hasPersonalData='true') + +struct OfflineReactiveFollow { + 1: required list followedUserIds(personalDataType='UserId') +}(persisted='true', hasPersonalData='true') + +struct OfflineNuxInterests { + 1: optional flows.OfflineFlowContext flowContext // set for recommendation inside an interactive flow +}(persisted='true', hasPersonalData='true') + +struct OfflineAdCampaignTarget { + 1: required list similarToUserIds(personalDataType='UserId') +}(persisted='true', hasPersonalData='true') + +struct OfflineConnectTab { + 1: required list byfSeedUserIds(personalDataType='UserId') + 2: required list similarToUserIds(personalDataType='UserId') + 3: required list recentlyEngagedUserIds +}(persisted='true', hasPersonalData='true') + +struct OfflineSimilarToUser { + 1: required i64 similarToUserId(personalDataType='UserId') +}(persisted='true', hasPersonalData='true') + +struct OfflinePostNuxFollowTask { + 1: optional flows.OfflineFlowContext flowContext // set for recommendation inside an interactive flow +}(persisted='true', hasPersonalData='true') + +// Offline equal of DisplayContext +union OfflineDisplayContext { + 1: OfflineProfile profile + 2: OfflineSearch search + 3: OfflineRux rux + 4: OfflineTopic topic + 5: OfflineReactiveFollow reactiveFollow + 6: OfflineNuxInterests nuxInterests + 7: OfflineAdCampaignTarget adCampaignTarget + 8: OfflineConnectTab connectTab + 9: OfflineSimilarToUser similarToUser + 10: OfflinePostNuxFollowTask postNuxFollowTask +}(persisted='true', hasPersonalData='true') diff --git a/follow-recommendations-service/thrift/src/main/thrift/logging/display_location.thrift b/follow-recommendations-service/thrift/src/main/thrift/logging/display_location.thrift new file mode 100644 index 0000000000..a4dbbecd46 --- /dev/null +++ b/follow-recommendations-service/thrift/src/main/thrift/logging/display_location.thrift @@ -0,0 +1,55 @@ +namespace java com.twitter.follow_recommendations.logging.thriftjava +#@namespace scala com.twitter.follow_recommendations.logging.thriftscala +#@namespace strato com.twitter.follow_recommendations.logging + +/** + * Make sure you add the new DL to the following files and redeploy our attribution jobs + * - follow-recommendations-service/thrift/src/main/thrift/display_location.thrift + * - follow-recommendations-service/thrift/src/main/thrift/logging/display_location.thrift + * - follow-recommendations-service/common/src/main/scala/com/twitter/follow_recommendations/common/models/DisplayLocation.scala + */ + +// Offline equal of DisplayLocation +enum OfflineDisplayLocation { + SIDEBAR = 0 + PROFILE_SIDEBAR = 2 + CLUSTER_FOLLOW = 7 + NEW_USER_SARUS_BACKFILL = 12 + PROFILE_DEVICE_FOLLOW = 23 + RECOS_BACKFILL = 32 + HOME_TIMELINE = 39 + PROFILE_TOP_FOLLOWING = 42 + PROFILE_TOP_FOLLOWERS = 43 + PEOPLE_PLUS_PLUS = 47 + EXPLORE_TAB = 57 + MagicRecs = 59 + AB_UPLOAD_INJECTION = 60 + CAMPAIGN_FORM = 61 + RUX_LANDING_PAGE = 62 + PROFILE_BONUS_FOLLOW = 63 + ELECTION_EXPLORE_WTF = 64 + HTL_BONUS_FOLLOW = 65 + TOPIC_LANDING_PAGE_HEADER = 66 + NUX_PYMK = 67 + NUX_INTERESTS = 68 + REACTIVE_FOLLOW = 69 + RUX_PYMK = 70 + INDIA_COVID19_CURATED_ACCOUNTS_WTF=71 + NUX_TOPIC_BONUS_FOLLOW = 72 + TWEET_NOTIFICATION_RECS = 73 + HTL_SPACE_HOSTS = 74 + POST_NUX_FOLLOW_TASK = 75 + TOPIC_LANDING_PAGE = 76 + USER_TYPEAHEAD_PREFETCH = 77 + HOME_TIMELINE_RELATABLE_ACCOUNTS = 78 + NUX_GEO_CATEGORY = 79 + NUX_INTERESTS_CATEGORY = 80 + NUX_PYMK_CATEGORY = 81 + TOP_ARTICLES = 82 + HOME_TIMELINE_TWEET_RECS = 83 + HTL_BULK_FRIEND_FOLLOWS = 84 + NUX_AUTO_FOLLOW = 85 + SEARCH_BONUS_FOLLOW = 86 + CONTENT_RECOMMENDER = 87 + HOME_TIMELINE_REVERSE_CHRON = 88 +}(persisted='true') diff --git a/follow-recommendations-service/thrift/src/main/thrift/logging/engagementType.thrift b/follow-recommendations-service/thrift/src/main/thrift/logging/engagementType.thrift new file mode 100644 index 0000000000..75191f16f0 --- /dev/null +++ b/follow-recommendations-service/thrift/src/main/thrift/logging/engagementType.thrift @@ -0,0 +1,11 @@ +namespace java com.twitter.follow_recommendations.logging.thriftjava +#@namespace scala com.twitter.follow_recommendations.logging.thriftscala +#@namespace strato com.twitter.follow_recommendations.logging + +enum EngagementType { + Click = 0 + Like = 1 + Mention = 2 + Retweet = 3 + ProfileView = 4 +} diff --git a/follow-recommendations-service/thrift/src/main/thrift/logging/flows.thrift b/follow-recommendations-service/thrift/src/main/thrift/logging/flows.thrift new file mode 100644 index 0000000000..98551c08ee --- /dev/null +++ b/follow-recommendations-service/thrift/src/main/thrift/logging/flows.thrift @@ -0,0 +1,16 @@ +namespace java com.twitter.follow_recommendations.logging.thriftjava +#@namespace scala com.twitter.follow_recommendations.logging.thriftscala +#@namespace strato com.twitter.follow_recommendations.logging + +struct OfflineFlowRecommendation { + 1: required i64 userId(personalDataType='UserId') +}(persisted='true', hasPersonalData='true') + +struct OfflineRecommendationStep { + 1: required list recommendations + 2: required set followedUserIds(personalDataType='UserId') +}(persisted='true', hasPersonalData='true') + +struct OfflineFlowContext { + 1: required list steps +}(persisted='true', hasPersonalData='true') diff --git a/follow-recommendations-service/thrift/src/main/thrift/logging/logs.thrift b/follow-recommendations-service/thrift/src/main/thrift/logging/logs.thrift new file mode 100644 index 0000000000..33f09cfb95 --- /dev/null +++ b/follow-recommendations-service/thrift/src/main/thrift/logging/logs.thrift @@ -0,0 +1,72 @@ +namespace java com.twitter.follow_recommendations.logging.thriftjava +#@namespace scala com.twitter.follow_recommendations.logging.thriftscala +#@namespace strato com.twitter.follow_recommendations.logging + +include "client_context.thrift" +include "debug.thrift" +include "display_context.thrift" +include "display_location.thrift" +include "recommendations.thrift" + +struct OfflineRecommendationRequest { + 1: required client_context.OfflineClientContext clientContext + 2: required display_location.OfflineDisplayLocation displayLocation + 3: optional display_context.OfflineDisplayContext displayContext + 4: optional i32 maxResults + 5: optional string cursor + 6: optional list excludedIds(personalDataType='UserId') + 7: optional bool fetchPromotedContent + 8: optional debug.OfflineDebugParams debugParams +}(persisted='true', hasPersonalData='true') + +struct OfflineRecommendationResponse { + 1: required list recommendations +}(persisted='true', hasPersonalData='true') + +struct RecommendationLog { + 1: required OfflineRecommendationRequest request + 2: required OfflineRecommendationResponse response + 3: required i64 timestampMs +}(persisted='true', hasPersonalData='true') + +struct OfflineScoringUserRequest { + 1: required client_context.OfflineClientContext clientContext + 2: required display_location.OfflineDisplayLocation displayLocation + 3: required list candidates +}(persisted='true', hasPersonalData='true') + +struct OfflineScoringUserResponse { + 1: required list candidates +}(persisted='true', hasPersonalData='true') + +struct ScoredUsersLog { + 1: required OfflineScoringUserRequest request + 2: required OfflineScoringUserResponse response + 3: required i64 timestampMs +}(persisted='true', hasPersonalData='true') + +struct OfflineRecommendationFlowUserMetadata { + 1: optional i32 userSignupAge(personalDataType = 'AgeOfAccount') + 2: optional string userState(personalDataType = 'UserState') +}(persisted='true', hasPersonalData='true') + +struct OfflineRecommendationFlowSignals { + 1: optional string countryCode(personalDataType='InferredCountry') +}(persisted='true', hasPersonalData='true') + +struct OfflineRecommendationFlowCandidateSourceCandidates { + 1: required string candidateSourceName + 2: required list candidateUserIds(personalDataType='UserId') + 3: optional list candidateUserScores +}(persisted='true', hasPersonalData='true') + +struct RecommendationFlowLog { + 1: required client_context.OfflineClientContext clientContext + 2: optional OfflineRecommendationFlowUserMetadata userMetadata + 3: optional OfflineRecommendationFlowSignals signals + 4: required i64 timestampMs + 5: required string recommendationFlowIdentifier + 6: optional list filteredCandidates + 7: optional list rankedCandidates + 8: optional list truncatedCandidates +}(persisted='true', hasPersonalData='true') diff --git a/follow-recommendations-service/thrift/src/main/thrift/logging/reasons.thrift b/follow-recommendations-service/thrift/src/main/thrift/logging/reasons.thrift new file mode 100644 index 0000000000..6fc24d919b --- /dev/null +++ b/follow-recommendations-service/thrift/src/main/thrift/logging/reasons.thrift @@ -0,0 +1,62 @@ +namespace java com.twitter.follow_recommendations.logging.thriftjava +#@namespace scala com.twitter.follow_recommendations.logging.thriftscala +#@namespace strato com.twitter.follow_recommendations.logging + +// Proof based on Follow relationship +struct FollowProof { + 1: required list userIds(personalDataType='UserId') + 2: required i32 numIds(personalDataType='CountOfFollowersAndFollowees') +}(persisted='true', hasPersonalData='true') + +// Similar to userIds in the context (e.g. profileId) +struct SimilarToProof { + 1: required list userIds(personalDataType='UserId') +}(persisted='true', hasPersonalData='true') + +// Proof based on geo location +struct PopularInGeoProof { + 1: required string location(personalDataType='InferredLocation') +}(persisted='true', hasPersonalData='true') + +// Proof based on ttt interest +struct TttInterestProof { + 1: required i64 interestId(personalDataType='ProvidedInterests') + 2: required string interestDisplayName(personalDataType='ProvidedInterests') +}(persisted='true', hasPersonalData='true') + +// Proof based on topics +struct TopicProof { + 1: required i64 topicId(personalDataType='ProvidedInterests') +}(persisted='true', hasPersonalData='true') + +// Proof based on custom interest / search queries +struct CustomInterestProof { + 1: required string customerInterest(personalDataType='SearchQuery') +}(persisted='true', hasPersonalData='true') + +// Proof based on tweet authors +struct TweetsAuthorProof { + 1: required list tweetIds(personalDataType='TweetId') +}(persisted='true', hasPersonalData='true') + +// Proof candidate is of device follow type +struct DeviceFollowProof { + 1: required bool isDeviceFollow(personalDataType='OtherDeviceInfo') +}(persisted='true', hasPersonalData='true') + +// Account level proof that should be attached to each candidate +struct AccountProof { + 1: optional FollowProof followProof + 2: optional SimilarToProof similarToProof + 3: optional PopularInGeoProof popularInGeoProof + 4: optional TttInterestProof tttInterestProof + 5: optional TopicProof topicProof + 6: optional CustomInterestProof customInterestProof + 7: optional TweetsAuthorProof tweetsAuthorProof + 8: optional DeviceFollowProof deviceFollowProof + +}(persisted='true', hasPersonalData='true') + +struct Reason { + 1: optional AccountProof accountProof +}(persisted='true', hasPersonalData='true') diff --git a/follow-recommendations-service/thrift/src/main/thrift/logging/recently_engaged_user_id.thrift b/follow-recommendations-service/thrift/src/main/thrift/logging/recently_engaged_user_id.thrift new file mode 100644 index 0000000000..f0af960b98 --- /dev/null +++ b/follow-recommendations-service/thrift/src/main/thrift/logging/recently_engaged_user_id.thrift @@ -0,0 +1,10 @@ +namespace java com.twitter.follow_recommendations.logging.thriftjava +#@namespace scala com.twitter.follow_recommendations.logging.thriftscala +#@namespace strato com.twitter.follow_recommendations.logging + +include "engagementType.thrift" + +struct RecentlyEngagedUserId { + 1: required i64 id(personalDataType='UserId') + 2: required engagementType.EngagementType engagementType +}(persisted='true', hasPersonalData='true') diff --git a/follow-recommendations-service/thrift/src/main/thrift/logging/recommendations.thrift b/follow-recommendations-service/thrift/src/main/thrift/logging/recommendations.thrift new file mode 100644 index 0000000000..bf94e41b8f --- /dev/null +++ b/follow-recommendations-service/thrift/src/main/thrift/logging/recommendations.thrift @@ -0,0 +1,26 @@ +namespace java com.twitter.follow_recommendations.logging.thriftjava +#@namespace scala com.twitter.follow_recommendations.logging.thriftscala +#@namespace strato com.twitter.follow_recommendations.logging + +include "com/twitter/ads/adserver/adserver_common.thrift" +include "reasons.thrift" +include "tracking.thrift" +include "scoring.thrift" + +// Offline equal of UserRecommendation +struct OfflineUserRecommendation { + 1: required i64 userId(personalDataType='UserId') + // reason for this suggestions, eg: social context + 2: optional reasons.Reason reason + // present if it is a promoted account + 3: optional adserver_common.AdImpression adImpression + // tracking token (unserialized) for attribution + 4: optional tracking.TrackingToken trackingToken + // scoring details + 5: optional scoring.ScoringDetails scoringDetails +}(persisted='true', hasPersonalData='true') + +// Offline equal of Recommendation +union OfflineRecommendation { + 1: OfflineUserRecommendation user +}(persisted='true', hasPersonalData='true') diff --git a/follow-recommendations-service/thrift/src/main/thrift/logging/scoring.thrift b/follow-recommendations-service/thrift/src/main/thrift/logging/scoring.thrift new file mode 100644 index 0000000000..e1524662d4 --- /dev/null +++ b/follow-recommendations-service/thrift/src/main/thrift/logging/scoring.thrift @@ -0,0 +1,38 @@ +namespace java com.twitter.follow_recommendations.logging.thriftjava +#@namespace scala com.twitter.follow_recommendations.logging.thriftscala +#@namespace strato com.twitter.follow_recommendations.logging + +include "com/twitter/ml/api/data.thrift" + +struct CandidateSourceDetails { + 1: optional map candidateSourceScores + 2: optional i32 primarySource +}(persisted='true', hasPersonalData='false') + +struct Score { + 1: required double value + 2: optional string rankerId + 3: optional string scoreType +}(persisted='true', hasPersonalData='false') // scoring and ranking info per ranking stage + +// Contains (1) the ML-based heavy ranker and score (2) scores and rankers in producer experiment framework +struct Scores { + 1: required list scores + 2: optional string selectedRankerId + 3: required bool isInProducerScoringExperiment +}(persisted='true', hasPersonalData='false') + +struct RankingInfo { + 1: optional Scores scores + 2: optional i32 rank +}(persisted='true', hasPersonalData='false') + +// this encapsulates all information related to the ranking process from generation to scoring +struct ScoringDetails { + 1: optional CandidateSourceDetails candidateSourceDetails + 2: optional double score // The ML-based heavy ranker score + 3: optional data.DataRecord dataRecord + 4: optional list rankerIds // all ranker ids, including (1) ML-based heavy ranker (2) non-ML adhoc rankers + 5: optional map infoPerRankingStage // scoring and ranking info per ranking stage +}(persisted='true', hasPersonalData='true') + diff --git a/follow-recommendations-service/thrift/src/main/thrift/logging/tracking.thrift b/follow-recommendations-service/thrift/src/main/thrift/logging/tracking.thrift new file mode 100644 index 0000000000..067ba1a46f --- /dev/null +++ b/follow-recommendations-service/thrift/src/main/thrift/logging/tracking.thrift @@ -0,0 +1,16 @@ +namespace java com.twitter.follow_recommendations.logging.thriftjava +#@namespace scala com.twitter.follow_recommendations.logging.thriftscala +#@namespace strato com.twitter.follow_recommendations.logging + +include "com/twitter/suggests/controller_data/controller_data.thrift" +include "display_location.thrift" + +struct TrackingToken { + // trace-id of the request + 1: required i64 sessionId (personalDataType='SessionId') + 2: optional display_location.OfflineDisplayLocation displayLocation + // 64-bit encoded binary attributes of our recommendation + 3: optional controller_data.ControllerData controllerData + // WTF Algorithm Id (backward compatibility) + 4: optional i32 algoId +}(persisted='true', hasPersonalData='true') diff --git a/follow-recommendations-service/thrift/src/main/thrift/reasons.thrift b/follow-recommendations-service/thrift/src/main/thrift/reasons.thrift new file mode 100644 index 0000000000..299e888857 --- /dev/null +++ b/follow-recommendations-service/thrift/src/main/thrift/reasons.thrift @@ -0,0 +1,61 @@ +namespace java com.twitter.follow_recommendations.thriftjava +#@namespace scala com.twitter.follow_recommendations.thriftscala +#@namespace strato com.twitter.follow_recommendations + +// Proof based on Follow relationship +struct FollowProof { + 1: required list userIds(personalDataType='UserId') + 2: required i32 numIds(personalDataType='CountOfFollowersAndFollowees') +}(hasPersonalData='true') + +// Similar to userIds in the context (e.g. profileId) +struct SimilarToProof { + 1: required list userIds(personalDataType='UserId') +}(hasPersonalData='true') + +// Proof based on geo location +struct PopularInGeoProof { + 1: required string location(personalDataType='InferredLocation') +}(hasPersonalData='true') + +// Proof based on ttt interest +struct TttInterestProof { + 1: required i64 interestId(personalDataType='ProvidedInterests') + 2: required string interestDisplayName(personalDataType='ProvidedInterests') +}(hasPersonalData='true') + +// Proof based on topics +struct TopicProof { + 1: required i64 topicId(personalDataType='ProvidedInterests') +}(hasPersonalData='true') + +// Proof based on custom interest / search queries +struct CustomInterestProof { + 1: required string query(personalDataType='SearchQuery') +}(hasPersonalData='true') + +// Proof based on tweet authors +struct TweetsAuthorProof { + 1: required list tweetIds(personalDataType='TweetId') +}(hasPersonalData='true') + +// Proof candidate is of device follow type +struct DeviceFollowProof { + 1: required bool isDeviceFollow(personalDataType='OtherDeviceInfo') +}(hasPersonalData='true') + +// Account level proof that should be attached to each candidate +struct AccountProof { + 1: optional FollowProof followProof + 2: optional SimilarToProof similarToProof + 3: optional PopularInGeoProof popularInGeoProof + 4: optional TttInterestProof tttInterestProof + 5: optional TopicProof topicProof + 6: optional CustomInterestProof customInterestProof + 7: optional TweetsAuthorProof tweetsAuthorProof + 8: optional DeviceFollowProof deviceFollowProof +}(hasPersonalData='true') + +struct Reason { + 1: optional AccountProof accountProof +}(hasPersonalData='true') diff --git a/follow-recommendations-service/thrift/src/main/thrift/recently_engaged_user_id.thrift b/follow-recommendations-service/thrift/src/main/thrift/recently_engaged_user_id.thrift new file mode 100644 index 0000000000..6a13bd31e3 --- /dev/null +++ b/follow-recommendations-service/thrift/src/main/thrift/recently_engaged_user_id.thrift @@ -0,0 +1,10 @@ +namespace java com.twitter.follow_recommendations.thriftjava +#@namespace scala com.twitter.follow_recommendations.thriftscala +#@namespace strato com.twitter.follow_recommendations + +include "engagementType.thrift" + +struct RecentlyEngagedUserId { + 1: required i64 id(personalDataType='UserId') + 2: required engagementType.EngagementType engagementType +}(persisted='true', hasPersonalData='true') diff --git a/follow-recommendations-service/thrift/src/main/thrift/recommendations.thrift b/follow-recommendations-service/thrift/src/main/thrift/recommendations.thrift new file mode 100644 index 0000000000..1070bb11c0 --- /dev/null +++ b/follow-recommendations-service/thrift/src/main/thrift/recommendations.thrift @@ -0,0 +1,40 @@ +namespace java com.twitter.follow_recommendations.thriftjava +#@namespace scala com.twitter.follow_recommendations.thriftscala +#@namespace strato com.twitter.follow_recommendations + +include "com/twitter/ads/adserver/adserver_common.thrift" +include "debug.thrift" +include "reasons.thrift" +include "scoring.thrift" + +struct UserRecommendation { + 1: required i64 userId(personalDataType='UserId') + // reason for this suggestions, eg: social context + 2: optional reasons.Reason reason + // present if it is a promoted account + 3: optional adserver_common.AdImpression adImpression + // tracking token for attribution + 4: optional string trackingInfo + // scoring details + 5: optional scoring.ScoringDetails scoringDetails + 6: optional string recommendationFlowIdentifier + // FeatureSwitch overrides for candidates: + 7: optional map featureOverrides +}(hasPersonalData='true') + +union Recommendation { + 1: UserRecommendation user +}(hasPersonalData='true') + +struct HydratedUserRecommendation { + 1: required i64 userId(personalDataType='UserId') + 2: optional string socialProof + // present if it is a promoted account, used by clients for determining ad impression + 3: optional adserver_common.AdImpression adImpression + // tracking token for attribution + 4: optional string trackingInfo +}(hasPersonalData='true') + +union HydratedRecommendation { + 1: HydratedUserRecommendation hydratedUserRecommendation +} diff --git a/follow-recommendations-service/thrift/src/main/thrift/scoring.thrift b/follow-recommendations-service/thrift/src/main/thrift/scoring.thrift new file mode 100644 index 0000000000..33111baf87 --- /dev/null +++ b/follow-recommendations-service/thrift/src/main/thrift/scoring.thrift @@ -0,0 +1,49 @@ +namespace java com.twitter.follow_recommendations.thriftjava +#@namespace scala com.twitter.follow_recommendations.thriftscala +#@namespace strato com.twitter.follow_recommendations + +include "com/twitter/ml/api/data.thrift" + +struct CandidateSourceDetails { + 1: optional map candidateSourceScores + 2: optional i32 primarySource + 3: optional map candidateSourceRanks +}(hasPersonalData='false') + +struct Score { + 1: required double value + 2: optional string rankerId + 3: optional string scoreType +}(hasPersonalData='false') + +// Contains (1) the ML-based heavy ranker and score (2) scores and rankers in producer experiment framework +struct Scores { + 1: required list scores + 2: optional string selectedRankerId + 3: required bool isInProducerScoringExperiment +}(hasPersonalData='false') + +struct RankingInfo { + 1: optional Scores scores + 2: optional i32 rank +}(hasPersonalData='false') + +// this encapsulates all information related to the ranking process from generation to scoring +struct ScoringDetails { + 1: optional CandidateSourceDetails candidateSourceDetails + 2: optional double score + 3: optional data.DataRecord dataRecord + 4: optional list rankerIds + 5: optional DebugDataRecord debugDataRecord // this field is not logged as it's only used for debugging + 6: optional map infoPerRankingStage // scoring and ranking info per ranking stage +}(hasPersonalData='true') + +// exactly the same as a data record, except that we store the feature name instead of the id +struct DebugDataRecord { + 1: optional set binaryFeatures; // stores BINARY features + 2: optional map continuousFeatures; // stores CONTINUOUS features + 3: optional map discreteFeatures; // stores DISCRETE features + 4: optional map stringFeatures; // stores STRING features + 5: optional map> sparseBinaryFeatures; // stores sparse BINARY features + 6: optional map> sparseContinuousFeatures; // sparse CONTINUOUS features +}(hasPersonalData='true') diff --git a/follow-recommendations-service/thrift/src/main/thrift/tracking.thrift b/follow-recommendations-service/thrift/src/main/thrift/tracking.thrift new file mode 100644 index 0000000000..81111ead81 --- /dev/null +++ b/follow-recommendations-service/thrift/src/main/thrift/tracking.thrift @@ -0,0 +1,17 @@ +namespace java com.twitter.follow_recommendations.thriftjava +#@namespace scala com.twitter.follow_recommendations.thriftscala +#@namespace strato com.twitter.follow_recommendations + +include "com/twitter/suggests/controller_data/controller_data.thrift" +include "display_location.thrift" + +// struct used for tracking/attribution purposes in our offline pipelines +struct TrackingToken { + // trace-id of the request + 1: required i64 sessionId (personalDataType='SessionId') + 2: optional display_location.DisplayLocation displayLocation + // 64-bit encoded binary attributes of our recommendation + 3: optional controller_data.ControllerData controllerData + // WTF Algorithm Id (backward compatibility) + 4: optional i32 algoId +}(hasPersonalData='true') diff --git a/graph-feature-service/BUILD.bazel b/graph-feature-service/BUILD.bazel new file mode 100644 index 0000000000..afad5ce5d5 --- /dev/null +++ b/graph-feature-service/BUILD.bazel @@ -0,0 +1,67 @@ +alias( + name = "graph_feature_service-server", + target = ":graph_feature_service-server_lib", +) + +target( + name = "graph_feature_service-server_lib", + dependencies = [ + "graph-feature-service/src/main/scala/com/twitter/graph_feature_service/server", + ], +) + +alias( + name = "graph_feature_service-worker", + target = ":graph_feature_service-worker_lib", +) + +target( + name = "graph_feature_service-worker_lib", + dependencies = [ + "graph-feature-service/src/main/scala/com/twitter/graph_feature_service/worker", + ], +) + +jvm_binary( + name = "server-bin", + basename = "graph_feature_service-server", + main = "com.twitter.graph_feature_service.server.Main", + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + ":graph_feature_service-server", + "3rdparty/jvm/ch/qos/logback:logback-classic", + "finagle/finagle-zipkin-scribe/src/main/scala", + "loglens/loglens-logback/src/main/scala/com/twitter/loglens/logback", + "twitter-server/logback-classic/src/main/scala", + ], +) + +jvm_binary( + name = "worker-bin", + basename = "graph_feature_service-worker", + main = "com.twitter.graph_feature_service.worker.Main", + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + ":graph_feature_service-worker", + "3rdparty/jvm/ch/qos/logback:logback-classic", + "finagle/finagle-zipkin-scribe/src/main/scala", + "loglens/loglens-logback/src/main/scala/com/twitter/loglens/logback", + "twitter-server/logback-classic/src/main/scala", + ], +) + +jvm_app( + name = "server-bundle", + basename = "graph_feature_service-server-dist", + binary = ":server-bin", + tags = ["bazel-compatible"], +) + +jvm_app( + name = "worker-bundle", + basename = "graph_feature_service-worker-dist", + binary = ":worker-bin", + tags = ["bazel-compatible"], +) diff --git a/graph-feature-service/README.md b/graph-feature-service/README.md new file mode 100644 index 0000000000..8305988b6b --- /dev/null +++ b/graph-feature-service/README.md @@ -0,0 +1,3 @@ +# Graph Feature Service + +Graph Feature Service (GFS) is a distributed system that can provide various graph features for given pairs of users. For instance, given source user A and candidate user C, GFS can answer questions like “how many of A’s followings have favorited C”, “how many of A’s followings are following C”, and “how much C is similar to the users that A has favorited“. \ No newline at end of file diff --git a/graph-feature-service/doc/common.md b/graph-feature-service/doc/common.md new file mode 100644 index 0000000000..70d293c848 --- /dev/null +++ b/graph-feature-service/doc/common.md @@ -0,0 +1,62 @@ +# Common thrift types + +GFS uses several thrift datastructures which are common to multiple queries. They are listed below. + +## EdgeType + +`EdgeType` is a thrift enum which specifies which edge types to query for the graph. + +```thrift +enum EdgeType { + FOLLOWING, + FOLLOWED_BY, + FAVORITE, + FAVORITED_BY, + RETWEET, + RETWEETED_BY, + REPLY, + REPLYED_BY, + MENTION, + MENTIONED_BY, + MUTUAL_FOLLOW, + SIMILAR_TO, // more edge types (like block, report, etc.) can be supported later. + RESERVED_12, + RESERVED_13, + RESERVED_14, + RESERVED_15, + RESERVED_16, + RESERVED_17, + RESERVED_18, + RESERVED_19, + RESERVED_20 +} +``` + +For an example of how this is used, consider the `GetNeighbors` query. If we set the `edgeType` field +of the `GfsNeighborsRequest`, the response will contain all the users that the specified user follows. +If, on the other hand, we set `edgeType` to be `FollowedBy` it will return all the users who are +followed by the specified user. + +## FeatureType + +`FeatureType` is a thrift struct which is used in queries which require two edge types. + +```thrift +struct FeatureType { + 1: required EdgeType leftEdgeType // edge type from source user + 2: required EdgeType rightEdgeType // edge type from candidate user +}(persisted="true") +``` + +## UserWithScore + +The candidate generation queries return lists of candidates together with a computed score for the +relevant feature. `UserWithScore` is a thrift struct which bundles together a candidate's ID with +the score. + +```thrift +struct UserWithScore { + 1: required i64 userId + 2: required double score +} +``` diff --git a/graph-feature-service/doc/getintersection.md b/graph-feature-service/doc/getintersection.md new file mode 100644 index 0000000000..6053729bb5 --- /dev/null +++ b/graph-feature-service/doc/getintersection.md @@ -0,0 +1,43 @@ +# GetIntersection + +## Request and response syntax + +A `GetIntersection` call takes as input a `GfsIntersectionRequest` thrift struct. + +```thrift +struct GfsIntersectionRequest { + 1: required i64 userId + 2: required list candidateUserIds + 3: required list featureTypes +} +``` + +The response is returned in a `GfsIntersectionResponse` thrift struct. + +```thrift +struct GfsIntersectionResponse { + 1: required i64 userId + 2: required list results +} + +struct GfsIntersectionResult { + 1: required i64 candidateUserId + 2: required list intersectionValues +} + +struct IntersectionValue { + 1: required FeatureType featureType + 2: optional i32 count + 3: optional list intersectionIds + 4: optional i32 leftNodeDegree + 5: optional i32 rightNodeDegree +}(persisted="true") +``` + +## Behavior + +The `GfsIntersectionResponse` contains in its `results` field a `GfsIntersectionResult` for every candidate in `candidateIds` which contains an `IntersectionValue` for every `FeatureType` in the request's `featureTypes` field. + +The `IntersectionValue` contains the size of the intersection between the `leftEdgeType` edges from `userId` and the `rightEdgeType` edges from `candidateId` in the `count` field, as well as their respective degrees in the graphs in `leftNodeDegree` and `rightNodeDegree` respectively. + +**Note:** the `intersectionIds` field currently only contains `Nil`. diff --git a/graph-feature-service/src/main/scala/com/twitter/graph_feature_service/common/BUILD.bazel b/graph-feature-service/src/main/scala/com/twitter/graph_feature_service/common/BUILD.bazel new file mode 100644 index 0000000000..e6b2e92be3 --- /dev/null +++ b/graph-feature-service/src/main/scala/com/twitter/graph_feature_service/common/BUILD.bazel @@ -0,0 +1,8 @@ +scala_library( + platform = "java8", + tags = [ + "bazel-compatible", + "bazel-only", + ], + dependencies = ["src/scala/com/twitter/storehaus_internal/util"], +) diff --git a/graph-feature-service/src/main/scala/com/twitter/graph_feature_service/common/Configs.scala b/graph-feature-service/src/main/scala/com/twitter/graph_feature_service/common/Configs.scala new file mode 100644 index 0000000000..20647d68cc --- /dev/null +++ b/graph-feature-service/src/main/scala/com/twitter/graph_feature_service/common/Configs.scala @@ -0,0 +1,73 @@ +package com.twitter.graph_feature_service.common + +import com.twitter.conversions.DurationOps._ +import com.twitter.util.Duration +import com.twitter.util.Time +import java.nio.ByteBuffer +import scala.util.hashing.MurmurHash3 + +object Configs { + + // NOTE: notify #recos-platform slack room, if you want to change this. + // This SHOULD be updated together with NUM_SHARDS in worker.aurora + final val NumGraphShards: Int = 40 + + final val TopKRealGraph: Int = 512 + + final val BaseHdfsPath: String = "/user/cassowary/processed/gfs/constant_db/" + + // whether or not to write in_value and out_value graphs. Used in the scalding job. + final val EnableValueGraphs: Boolean = true + // whether or not to write in_key and out_key graphs. Used in the scalding job. + final val EnableKeyGraphs: Boolean = false + + final val FollowOutValPath: String = "follow_out_val/" + final val FollowOutKeyPath: String = "follow_out_key/" + final val FollowInValPath: String = "follow_in_val/" + final val FollowInKeyPath: String = "follow_in_key/" + + final val MutualFollowValPath: String = "mutual_follow_val/" + final val MutualFollowKeyPath: String = "mutual_follow_key/" + + final val FavoriteOutValPath: String = "favorite_out_val/" + final val FavoriteInValPath: String = "favorite_in_val/" + final val FavoriteOutKeyPath: String = "favorite_out_key/" + final val FavoriteInKeyPath: String = "favorite_in_key/" + + final val RetweetOutValPath: String = "retweet_out_val/" + final val RetweetInValPath: String = "retweet_in_val/" + final val RetweetOutKeyPath: String = "retweet_out_key/" + final val RetweetInKeyPath: String = "retweet_in_key/" + + final val MentionOutValPath: String = "mention_out_val/" + final val MentionInValPath: String = "mention_in_val/" + final val MentionOutKeyPath: String = "mention_out_key/" + final val MentionInKeyPath: String = "mention_in_key/" + + final val MemCacheTTL: Duration = 8.hours + + final val RandomSeed: Int = 39582942 + + def getTimedHdfsShardPath(shardId: Int, path: String, time: Time): String = { + val timeStr = time.format("yyyy/MM/dd") + s"$path/$timeStr/shard_$shardId" + } + + def getHdfsPath(path: String, overrideBaseHdfsPath: Option[String] = None): String = { + val basePath = overrideBaseHdfsPath.getOrElse(BaseHdfsPath) + s"$basePath$path" + } + + private def hash(kArr: Array[Byte], seed: Int): Int = { + MurmurHash3.bytesHash(kArr, seed) & 0x7fffffff // keep positive + } + + private def hashLong(l: Long, seed: Int): Int = { + hash(ByteBuffer.allocate(8).putLong(l).array(), seed) + } + + def shardForUser(userId: Long): Int = { + hashLong(userId, RandomSeed) % NumGraphShards + } + +} diff --git a/graph-feature-service/src/main/scala/com/twitter/graph_feature_service/server/BUILD.bazel b/graph-feature-service/src/main/scala/com/twitter/graph_feature_service/server/BUILD.bazel new file mode 100644 index 0000000000..c20a5e04c7 --- /dev/null +++ b/graph-feature-service/src/main/scala/com/twitter/graph_feature_service/server/BUILD.bazel @@ -0,0 +1,35 @@ +scala_library( + sources = ["**/*.scala"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/com/google/inject:guice", + "3rdparty/jvm/com/twitter/storehaus:core", + "3rdparty/jvm/javax/inject:javax.inject", + "3rdparty/jvm/net/codingwell:scala-guice", + "3rdparty/jvm/org/lz4:lz4-java", + "3rdparty/jvm/org/slf4j:slf4j-api", + "discovery-common/src/main/scala/com/twitter/discovery/common/stats", + "finagle/finagle-http/src/main/scala", + "finatra-internal/decider/src/main/scala", + "finatra-internal/mtls-thriftmux/src/main/scala", + "finatra/inject/inject-app/src/main/scala", + "finatra/inject/inject-core/src/main/scala", + "finatra/inject/inject-server/src/main/scala", + "finatra/inject/inject-thrift-client/src/main/scala", + "finatra/inject/inject-utils/src/main/scala", + "finatra/thrift/src/main/scala/com/twitter/finatra/thrift", + "finatra/thrift/src/main/scala/com/twitter/finatra/thrift:controller", + "finatra/thrift/src/main/scala/com/twitter/finatra/thrift/filters", + "finatra/thrift/src/main/scala/com/twitter/finatra/thrift/routing", + "graph-feature-service/src/main/resources", + "graph-feature-service/src/main/scala/com/twitter/graph_feature_service/common", + "graph-feature-service/src/main/scala/com/twitter/graph_feature_service/util", + "graph-feature-service/src/main/thrift/com/twitter/graph_feature_service:graph_feature_service_thrift-scala", + "hermit/hermit-core/src/main/scala/com/twitter/hermit/store/common", + "servo/request/src/main/scala", + "src/scala/com/twitter/storehaus_internal/memcache", + "util/util-app/src/main/scala", + "util/util-slf4j-api/src/main/scala", + ], +) diff --git a/graph-feature-service/src/main/scala/com/twitter/graph_feature_service/server/Main.scala b/graph-feature-service/src/main/scala/com/twitter/graph_feature_service/server/Main.scala new file mode 100644 index 0000000000..5980afdf23 --- /dev/null +++ b/graph-feature-service/src/main/scala/com/twitter/graph_feature_service/server/Main.scala @@ -0,0 +1,56 @@ +package com.twitter.graph_feature_service.server + +import com.google.inject.Module +import com.twitter.finatra.decider.modules.DeciderModule +import com.twitter.finatra.mtls.thriftmux.Mtls +import com.twitter.finatra.thrift.ThriftServer +import com.twitter.finatra.thrift.filters.{ + AccessLoggingFilter, + LoggingMDCFilter, + StatsFilter, + ThriftMDCFilter, + TraceIdMDCFilter +} +import com.twitter.finatra.mtls.thriftmux.modules.MtlsThriftWebFormsModule +import com.twitter.finatra.thrift.routing.ThriftRouter +import com.twitter.graph_feature_service.server.controllers.ServerController +import com.twitter.graph_feature_service.server.handlers.ServerWarmupHandler +import com.twitter.graph_feature_service.server.modules.{ + GetIntersectionStoreModule, + GraphFeatureServiceWorkerClientsModule, + ServerFlagsModule +} +import com.twitter.graph_feature_service.thriftscala +import com.twitter.inject.thrift.modules.ThriftClientIdModule + +object Main extends ServerMain + +class ServerMain extends ThriftServer with Mtls { + + override val name = "graph_feature_service-server" + + override val modules: Seq[Module] = { + Seq( + ServerFlagsModule, + DeciderModule, + ThriftClientIdModule, + GraphFeatureServiceWorkerClientsModule, + GetIntersectionStoreModule, + new MtlsThriftWebFormsModule[thriftscala.Server.MethodPerEndpoint](this) + ) + } + + override def configureThrift(router: ThriftRouter): Unit = { + router + .filter[LoggingMDCFilter] + .filter[TraceIdMDCFilter] + .filter[ThriftMDCFilter] + .filter[AccessLoggingFilter] + .filter[StatsFilter] + .add[ServerController] + } + + override protected def warmup(): Unit = { + handle[ServerWarmupHandler]() + } +} diff --git a/graph-feature-service/src/main/scala/com/twitter/graph_feature_service/server/controllers/ServerController.scala b/graph-feature-service/src/main/scala/com/twitter/graph_feature_service/server/controllers/ServerController.scala new file mode 100644 index 0000000000..ca8973f05d --- /dev/null +++ b/graph-feature-service/src/main/scala/com/twitter/graph_feature_service/server/controllers/ServerController.scala @@ -0,0 +1,46 @@ +package com.twitter.graph_feature_service.server.controllers + +import com.twitter.discovery.common.stats.DiscoveryStatsFilter +import com.twitter.finagle.Service +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.finatra.thrift.Controller +import com.twitter.graph_feature_service.server.handlers.ServerGetIntersectionHandler.GetIntersectionRequest +import com.twitter.graph_feature_service.server.handlers.ServerGetIntersectionHandler +import com.twitter.graph_feature_service.thriftscala +import com.twitter.graph_feature_service.thriftscala.Server.GetIntersection +import com.twitter.graph_feature_service.thriftscala.Server.GetPresetIntersection +import com.twitter.graph_feature_service.thriftscala._ +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class ServerController @Inject() ( + serverGetIntersectionHandler: ServerGetIntersectionHandler +)( + implicit statsReceiver: StatsReceiver) + extends Controller(thriftscala.Server) { + + private val getIntersectionService: Service[GetIntersectionRequest, GfsIntersectionResponse] = + new DiscoveryStatsFilter(statsReceiver.scope("srv").scope("get_intersection")) + .andThen(Service.mk(serverGetIntersectionHandler)) + + val getIntersection: Service[GetIntersection.Args, GfsIntersectionResponse] = { args => + // TODO: Disable updateCache after HTL switch to use PresetIntersection endpoint. + getIntersectionService( + GetIntersectionRequest.fromGfsIntersectionRequest(args.request, cacheable = true)) + } + handle(GetIntersection) { getIntersection } + + def getPresetIntersection: Service[ + GetPresetIntersection.Args, + GfsIntersectionResponse + ] = { args => + // TODO: Refactor after HTL switch to PresetIntersection + val cacheable = args.request.presetFeatureTypes == PresetFeatureTypes.HtlTwoHop + getIntersectionService( + GetIntersectionRequest.fromGfsPresetIntersectionRequest(args.request, cacheable)) + } + + handle(GetPresetIntersection) { getPresetIntersection } + +} diff --git a/graph-feature-service/src/main/scala/com/twitter/graph_feature_service/server/handlers/ServerGetIntersectionHandler.scala b/graph-feature-service/src/main/scala/com/twitter/graph_feature_service/server/handlers/ServerGetIntersectionHandler.scala new file mode 100644 index 0000000000..2c77c1f54d --- /dev/null +++ b/graph-feature-service/src/main/scala/com/twitter/graph_feature_service/server/handlers/ServerGetIntersectionHandler.scala @@ -0,0 +1,198 @@ +package com.twitter.graph_feature_service.server.handlers + +import com.twitter.finagle.stats.Stat +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.graph_feature_service.server.handlers.ServerGetIntersectionHandler.GetIntersectionRequest +import com.twitter.graph_feature_service.server.stores.FeatureTypesEncoder +import com.twitter.graph_feature_service.server.stores.GetIntersectionStore.GetIntersectionQuery +import com.twitter.graph_feature_service.thriftscala.PresetFeatureTypes +import com.twitter.graph_feature_service.thriftscala._ +import com.twitter.graph_feature_service.util.FeatureTypesCalculator +import com.twitter.servo.request.RequestHandler +import com.twitter.storehaus.ReadableStore +import com.twitter.util.Future +import com.twitter.util.Memoize +import javax.inject.Inject +import javax.inject.Named +import javax.inject.Singleton + +@Singleton +class ServerGetIntersectionHandler @Inject() ( + @Named("ReadThroughGetIntersectionStore") + readThroughStore: ReadableStore[GetIntersectionQuery, CachedIntersectionResult], + @Named("BypassCacheGetIntersectionStore") + readOnlyStore: ReadableStore[GetIntersectionQuery, CachedIntersectionResult] +)( + implicit statsReceiver: StatsReceiver) + extends RequestHandler[GetIntersectionRequest, GfsIntersectionResponse] { + + import ServerGetIntersectionHandler._ + + // TODO: Track all the stats based on PresetFeatureType and update the dashboard + private val stats: StatsReceiver = statsReceiver.scope("srv").scope("get_intersection") + private val numCandidatesCount = stats.counter("total_num_candidates") + private val numCandidatesStat = stats.stat("num_candidates") + private val numFeaturesStat = stats.stat("num_features") + private val userEmptyCount = stats.counter("user_empty_count") + private val candidateEmptyRateStat = stats.stat("candidate_empty_rate") + private val candidateNumEmptyStat = stats.stat("candidate_num_empty") + private val missedRateStat = stats.stat("miss_rate") + private val numMissedStat = stats.stat("num_missed") + + // Assume the order from HTL doesn't change. Only log the HTL query now. + private val featureStatMap = FeatureTypesCalculator.presetFeatureTypes.map { feature => + val featureString = s"${feature.leftEdgeType.name}_${feature.rightEdgeType.name}" + feature -> Array( + stats.counter(s"feature_type_${featureString}_total"), + stats.counter(s"feature_type_${featureString}_count_zero"), + stats.counter(s"feature_type_${featureString}_left_zero"), + stats.counter(s"feature_type_${featureString}_right_zero") + ) + }.toMap + + private val sourceCandidateNumStats = Memoize[PresetFeatureTypes, Stat] { presetFeature => + stats.stat(s"source_candidate_num_${presetFeature.name}") + } + + override def apply(request: GetIntersectionRequest): Future[GfsIntersectionResponse] = { + val featureTypes = request.calculatedFeatureTypes + val numCandidates = request.candidateUserIds.length + val numFeatures = featureTypes.length + + numCandidatesCount.incr(numCandidates) + numCandidatesStat.add(numCandidates) + numFeaturesStat.add(numFeatures) + sourceCandidateNumStats(request.presetFeatureTypes).add(numCandidates) + + // Note: do not change the orders of features and candidates. + val candidateIds = request.candidateUserIds + + if (featureTypes.isEmpty || candidateIds.isEmpty) { + Future.value(DefaultGfsIntersectionResponse) + } else { + Future + .collect { + val getIntersectionStore = if (request.cacheable) readThroughStore else readOnlyStore + getIntersectionStore.multiGet(GetIntersectionQuery.buildQueries(request)) + }.map { responses => + val results = responses.collect { + case (query, Some(result)) => + query.candidateId -> GfsIntersectionResult( + query.candidateId, + query.calculatedFeatureTypes.zip(result.values).map { + case (featureType, value) => + IntersectionValue( + featureType, + Some(value.count), + if (value.intersectionIds.isEmpty) None else Some(value.intersectionIds), + Some(value.leftNodeDegree), + Some(value.rightNodeDegree) + ) + } + ) + } + + // Keep the response order same as input + val processedResults = candidateIds.map { candidateId => + results.getOrElse(candidateId, GfsIntersectionResult(candidateId, List.empty)) + } + + val candidateEmptyNum = + processedResults.count( + _.intersectionValues.exists(value => isZero(value.rightNodeDegree))) + + val numMissed = processedResults.count(_.intersectionValues.size != numFeatures) + + if (processedResults.exists( + _.intersectionValues.forall(value => isZero(value.leftNodeDegree)))) { + userEmptyCount.incr() + } + + candidateNumEmptyStat.add(candidateEmptyNum) + candidateEmptyRateStat.add(candidateEmptyNum.toFloat / numCandidates) + numMissedStat.add(numMissed) + missedRateStat.add(numMissed.toFloat / numCandidates) + + processedResults.foreach { result => + result.intersectionValues.zip(featureTypes).foreach { + case (value, featureType) => + featureStatMap.get(featureType).foreach { statsArray => + statsArray(TotalIndex).incr() + if (isZero(value.count)) { + statsArray(CountIndex).incr() + } + if (isZero(value.leftNodeDegree)) { + statsArray(LeftIndex).incr() + } + if (isZero(value.rightNodeDegree)) { + statsArray(RightIndex).incr() + } + } + } + } + + GfsIntersectionResponse(processedResults) + } + } + + } + +} + +private[graph_feature_service] object ServerGetIntersectionHandler { + + case class GetIntersectionRequest( + userId: Long, + candidateUserIds: Seq[Long], + featureTypes: Seq[FeatureType], + presetFeatureTypes: PresetFeatureTypes, + intersectionIdLimit: Option[Int], + cacheable: Boolean) { + + lazy val calculatedFeatureTypes: Seq[FeatureType] = + FeatureTypesCalculator.getFeatureTypes(presetFeatureTypes, featureTypes) + + lazy val calculatedFeatureTypesString: String = + FeatureTypesEncoder(calculatedFeatureTypes) + } + + object GetIntersectionRequest { + + def fromGfsIntersectionRequest( + request: GfsIntersectionRequest, + cacheable: Boolean + ): GetIntersectionRequest = { + GetIntersectionRequest( + request.userId, + request.candidateUserIds, + request.featureTypes, + PresetFeatureTypes.Empty, + request.intersectionIdLimit, + cacheable) + } + + def fromGfsPresetIntersectionRequest( + request: GfsPresetIntersectionRequest, + cacheable: Boolean + ): GetIntersectionRequest = { + GetIntersectionRequest( + request.userId, + request.candidateUserIds, + List.empty, + request.presetFeatureTypes, + request.intersectionIdLimit, + cacheable) + } + } + + private val DefaultGfsIntersectionResponse = GfsIntersectionResponse() + + private val TotalIndex = 0 + private val CountIndex = 1 + private val LeftIndex = 2 + private val RightIndex = 3 + + def isZero(opt: Option[Int]): Boolean = { + !opt.exists(_ != 0) + } +} diff --git a/graph-feature-service/src/main/scala/com/twitter/graph_feature_service/server/handlers/ServerWarmupHandler.scala b/graph-feature-service/src/main/scala/com/twitter/graph_feature_service/server/handlers/ServerWarmupHandler.scala new file mode 100644 index 0000000000..3e31f2c8fd --- /dev/null +++ b/graph-feature-service/src/main/scala/com/twitter/graph_feature_service/server/handlers/ServerWarmupHandler.scala @@ -0,0 +1,45 @@ +package com.twitter.graph_feature_service.server.handlers + +import com.twitter.finatra.thrift.routing.ThriftWarmup +import com.twitter.graph_feature_service.thriftscala.EdgeType.FavoritedBy +import com.twitter.graph_feature_service.thriftscala.EdgeType.FollowedBy +import com.twitter.graph_feature_service.thriftscala.EdgeType.Following +import com.twitter.graph_feature_service.thriftscala.Server.GetIntersection +import com.twitter.graph_feature_service.thriftscala.FeatureType +import com.twitter.graph_feature_service.thriftscala.GfsIntersectionRequest +import com.twitter.inject.utils.Handler +import com.twitter.scrooge.Request +import com.twitter.util.logging.Logger +import javax.inject.Inject +import javax.inject.Singleton +import scala.util.Random + +@Singleton +class ServerWarmupHandler @Inject() (warmup: ThriftWarmup) extends Handler { + + val logger: Logger = Logger("WarmupHandler") + + // TODO: Add the testing accounts to warm-up the service. + private val testingAccounts: Array[Long] = Seq.empty.toArray + + private def getRandomRequest: GfsIntersectionRequest = { + GfsIntersectionRequest( + testingAccounts(Random.nextInt(testingAccounts.length)), + testingAccounts, + Seq(FeatureType(Following, FollowedBy), FeatureType(Following, FavoritedBy)) + ) + } + + override def handle(): Unit = { + warmup.sendRequest( + GetIntersection, + Request( + GetIntersection.Args( + getRandomRequest + )), + 10 + )() + + logger.info("Warmup Done!") + } +} diff --git a/graph-feature-service/src/main/scala/com/twitter/graph_feature_service/server/modules/GetIntersectionStoreModule.scala b/graph-feature-service/src/main/scala/com/twitter/graph_feature_service/server/modules/GetIntersectionStoreModule.scala new file mode 100644 index 0000000000..cae99b3a60 --- /dev/null +++ b/graph-feature-service/src/main/scala/com/twitter/graph_feature_service/server/modules/GetIntersectionStoreModule.scala @@ -0,0 +1,91 @@ +package com.twitter.graph_feature_service.server.modules + +import com.google.inject.Provides +import com.twitter.bijection.scrooge.CompactScalaCodec +import com.twitter.conversions.DurationOps._ +import com.twitter.finagle.mtls.authentication.ServiceIdentifier +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.graph_feature_service.common.Configs._ +import com.twitter.graph_feature_service.server.stores.GetIntersectionStore +import com.twitter.graph_feature_service.server.stores.GetIntersectionStore.GetIntersectionQuery +import com.twitter.graph_feature_service.thriftscala.CachedIntersectionResult +import com.twitter.hermit.store.common.ObservedMemcachedReadableStore +import com.twitter.inject.TwitterModule +import com.twitter.inject.annotations.Flag +import com.twitter.storehaus.ReadableStore +import com.twitter.storehaus_internal.memcache.MemcacheStore +import com.twitter.storehaus_internal.util.{ClientName, ZkEndPoint} +import com.twitter.util.Duration +import javax.inject.{Named, Singleton} + +/** + * Initialize the MemCache based GetIntersectionStore. + * The Key of MemCache is UserId~CandidateId~FeatureTypes~IntersectionIdLimit. + */ +object GetIntersectionStoreModule extends TwitterModule { + + private[this] val requestTimeout: Duration = 25.millis + private[this] val retries: Int = 0 + + @Provides + @Named("ReadThroughGetIntersectionStore") + @Singleton + def provideReadThroughGetIntersectionStore( + graphFeatureServiceWorkerClients: GraphFeatureServiceWorkerClients, + serviceIdentifier: ServiceIdentifier, + @Flag(ServerFlagNames.MemCacheClientName) memCacheName: String, + @Flag(ServerFlagNames.MemCachePath) memCachePath: String + )( + implicit statsReceiver: StatsReceiver + ): ReadableStore[GetIntersectionQuery, CachedIntersectionResult] = { + buildMemcacheStore( + graphFeatureServiceWorkerClients, + memCacheName, + memCachePath, + serviceIdentifier) + } + + @Provides + @Named("BypassCacheGetIntersectionStore") + @Singleton + def provideReadOnlyGetIntersectionStore( + graphFeatureServiceWorkerClients: GraphFeatureServiceWorkerClients, + )( + implicit statsReceiver: StatsReceiver + ): ReadableStore[GetIntersectionQuery, CachedIntersectionResult] = { + // Bypass the Memcache. + GetIntersectionStore(graphFeatureServiceWorkerClients, statsReceiver) + } + + private[this] def buildMemcacheStore( + graphFeatureServiceWorkerClients: GraphFeatureServiceWorkerClients, + memCacheName: String, + memCachePath: String, + serviceIdentifier: ServiceIdentifier, + )( + implicit statsReceiver: StatsReceiver + ): ReadableStore[GetIntersectionQuery, CachedIntersectionResult] = { + val backingStore = GetIntersectionStore(graphFeatureServiceWorkerClients, statsReceiver) + + val cacheClient = MemcacheStore.memcachedClient( + name = ClientName(memCacheName), + dest = ZkEndPoint(memCachePath), + timeout = requestTimeout, + retries = retries, + serviceIdentifier = serviceIdentifier, + statsReceiver = statsReceiver + ) + + ObservedMemcachedReadableStore.fromCacheClient[GetIntersectionQuery, CachedIntersectionResult]( + backingStore = backingStore, + cacheClient = cacheClient, + ttl = MemCacheTTL + )( + valueInjection = LZ4Injection.compose(CompactScalaCodec(CachedIntersectionResult)), + statsReceiver = statsReceiver.scope("mem_cache"), + keyToString = { key => + s"L~${key.userId}~${key.candidateId}~${key.featureTypesString}~${key.intersectionIdLimit}" + } + ) + } +} diff --git a/graph-feature-service/src/main/scala/com/twitter/graph_feature_service/server/modules/GraphFeatureServiceWorkerClientsModule.scala b/graph-feature-service/src/main/scala/com/twitter/graph_feature_service/server/modules/GraphFeatureServiceWorkerClientsModule.scala new file mode 100644 index 0000000000..a6f56827ef --- /dev/null +++ b/graph-feature-service/src/main/scala/com/twitter/graph_feature_service/server/modules/GraphFeatureServiceWorkerClientsModule.scala @@ -0,0 +1,51 @@ +package com.twitter.graph_feature_service.server.modules + +import com.google.inject.Provides +import com.twitter.conversions.DurationOps._ +import com.twitter.finagle.mtls.authentication.ServiceIdentifier +import com.twitter.finagle.mtls.client.MtlsStackClient._ +import com.twitter.finagle.ThriftMux +import com.twitter.finagle.service.RetryBudget +import com.twitter.graph_feature_service.thriftscala +import com.twitter.inject.TwitterModule +import com.twitter.inject.annotations.Flag +import com.twitter.util.{Await, Duration} +import javax.inject.Singleton + +case class GraphFeatureServiceWorkerClients( + workers: Seq[thriftscala.Worker.MethodPerEndpoint]) + +object GraphFeatureServiceWorkerClientsModule extends TwitterModule { + private[this] val closeableGracePeriod: Duration = 1.second + private[this] val requestTimeout: Duration = 25.millis + + @Provides + @Singleton + def provideGraphFeatureServiceWorkerClient( + @Flag(ServerFlagNames.NumWorkers) numWorkers: Int, + @Flag(ServerFlagNames.ServiceRole) serviceRole: String, + @Flag(ServerFlagNames.ServiceEnv) serviceEnv: String, + serviceIdentifier: ServiceIdentifier + ): GraphFeatureServiceWorkerClients = { + + val workers: Seq[thriftscala.Worker.MethodPerEndpoint] = + (0 until numWorkers).map { id => + val dest = s"/srv#/$serviceEnv/local/$serviceRole/graph_feature_service-worker-$id" + + val client = ThriftMux.client + .withRequestTimeout(requestTimeout) + .withRetryBudget(RetryBudget.Empty) + .withMutualTls(serviceIdentifier) + .build[thriftscala.Worker.MethodPerEndpoint](dest, s"worker-$id") + + onExit { + val closeable = client.asClosable + Await.result(closeable.close(closeableGracePeriod), closeableGracePeriod) + } + + client + } + + GraphFeatureServiceWorkerClients(workers) + } +} diff --git a/graph-feature-service/src/main/scala/com/twitter/graph_feature_service/server/modules/LZ4Injection.scala b/graph-feature-service/src/main/scala/com/twitter/graph_feature_service/server/modules/LZ4Injection.scala new file mode 100644 index 0000000000..3018dcd9ca --- /dev/null +++ b/graph-feature-service/src/main/scala/com/twitter/graph_feature_service/server/modules/LZ4Injection.scala @@ -0,0 +1,17 @@ +package com.twitter.graph_feature_service.server.modules + +import com.twitter.bijection.Injection +import scala.util.Try +import net.jpountz.lz4.{LZ4CompressorWithLength, LZ4DecompressorWithLength, LZ4Factory} + +object LZ4Injection extends Injection[Array[Byte], Array[Byte]] { + private val lz4Factory = LZ4Factory.fastestInstance() + private val fastCompressor = new LZ4CompressorWithLength(lz4Factory.fastCompressor()) + private val decompressor = new LZ4DecompressorWithLength(lz4Factory.fastDecompressor()) + + override def apply(a: Array[Byte]): Array[Byte] = LZ4Injection.fastCompressor.compress(a) + + override def invert(b: Array[Byte]): Try[Array[Byte]] = Try { + LZ4Injection.decompressor.decompress(b) + } +} diff --git a/graph-feature-service/src/main/scala/com/twitter/graph_feature_service/server/modules/ServerFlagModule.scala b/graph-feature-service/src/main/scala/com/twitter/graph_feature_service/server/modules/ServerFlagModule.scala new file mode 100644 index 0000000000..d38bfb24a7 --- /dev/null +++ b/graph-feature-service/src/main/scala/com/twitter/graph_feature_service/server/modules/ServerFlagModule.scala @@ -0,0 +1,31 @@ +package com.twitter.graph_feature_service.server.modules + +import com.twitter.inject.TwitterModule + +object ServerFlagNames { + final val NumWorkers = "service.num_workers" + final val ServiceRole = "service.role" + final val ServiceEnv = "service.env" + + final val MemCacheClientName = "service.mem_cache_client_name" + final val MemCachePath = "service.mem_cache_path" +} + +/** + * Initializes references to the flag values defined in the aurora.deploy file. + * To check what the flag values are initialized in runtime, search FlagsModule in stdout + */ +object ServerFlagsModule extends TwitterModule { + + import ServerFlagNames._ + + flag[Int](NumWorkers, "Num of workers") + + flag[String](ServiceRole, "Service Role") + + flag[String](ServiceEnv, "Service Env") + + flag[String](MemCacheClientName, "MemCache Client Name") + + flag[String](MemCachePath, "MemCache Path") +} diff --git a/graph-feature-service/src/main/scala/com/twitter/graph_feature_service/server/stores/FeatureTypesEncoder.scala b/graph-feature-service/src/main/scala/com/twitter/graph_feature_service/server/stores/FeatureTypesEncoder.scala new file mode 100644 index 0000000000..b8ad4d7431 --- /dev/null +++ b/graph-feature-service/src/main/scala/com/twitter/graph_feature_service/server/stores/FeatureTypesEncoder.scala @@ -0,0 +1,16 @@ +package com.twitter.graph_feature_service.server.stores + +import com.twitter.graph_feature_service.common.Configs.RandomSeed +import com.twitter.graph_feature_service.thriftscala.FeatureType +import scala.util.hashing.MurmurHash3 + +object FeatureTypesEncoder { + + def apply(featureTypes: Seq[FeatureType]): String = { + val byteArray = featureTypes.flatMap { featureType => + Array(featureType.leftEdgeType.getValue.toByte, featureType.rightEdgeType.getValue.toByte) + }.toArray + (MurmurHash3.bytesHash(byteArray, RandomSeed) & 0x7fffffff).toString // keep positive + } + +} diff --git a/graph-feature-service/src/main/scala/com/twitter/graph_feature_service/server/stores/GetIntersectionStore.scala b/graph-feature-service/src/main/scala/com/twitter/graph_feature_service/server/stores/GetIntersectionStore.scala new file mode 100644 index 0000000000..7824be511a --- /dev/null +++ b/graph-feature-service/src/main/scala/com/twitter/graph_feature_service/server/stores/GetIntersectionStore.scala @@ -0,0 +1,181 @@ +package com.twitter.graph_feature_service.server.stores + +import com.twitter.finagle.RequestTimeoutException +import com.twitter.finagle.stats.{Stat, StatsReceiver} +import com.twitter.graph_feature_service.server.handlers.ServerGetIntersectionHandler.GetIntersectionRequest +import com.twitter.graph_feature_service.server.modules.GraphFeatureServiceWorkerClients +import com.twitter.graph_feature_service.server.stores.GetIntersectionStore.GetIntersectionQuery +import com.twitter.graph_feature_service.thriftscala._ +import com.twitter.inject.Logging +import com.twitter.storehaus.ReadableStore +import com.twitter.util.Future +import javax.inject.Singleton +import scala.collection.mutable.ArrayBuffer + +@Singleton +case class GetIntersectionStore( + graphFeatureServiceWorkerClients: GraphFeatureServiceWorkerClients, + statsReceiver: StatsReceiver) + extends ReadableStore[GetIntersectionQuery, CachedIntersectionResult] + with Logging { + + import GetIntersectionStore._ + + private val stats = statsReceiver.scope("get_intersection_store") + private val requestCount = stats.counter(name = "request_count") + private val aggregatorLatency = stats.stat("aggregator_latency") + private val timeOutCounter = stats.counter("worker_timeouts") + private val unknownErrorCounter = stats.counter("unknown_errors") + + override def multiGet[K1 <: GetIntersectionQuery]( + ks: Set[K1] + ): Map[K1, Future[Option[CachedIntersectionResult]]] = { + if (ks.isEmpty) { + Map.empty + } else { + requestCount.incr() + + val head = ks.head + // We assume all the GetIntersectionQuery use the same userId and featureTypes + val userId = head.userId + val featureTypes = head.featureTypes + val presetFeatureTypes = head.presetFeatureTypes + val calculatedFeatureTypes = head.calculatedFeatureTypes + val intersectionIdLimit = head.intersectionIdLimit + + val request = WorkerIntersectionRequest( + userId, + ks.map(_.candidateId).toArray, + featureTypes, + presetFeatureTypes, + intersectionIdLimit + ) + + val resultFuture = Future + .collect( + graphFeatureServiceWorkerClients.workers.map { worker => + worker + .getIntersection(request) + .rescue { + case _: RequestTimeoutException => + timeOutCounter.incr() + Future.value(DefaultWorkerIntersectionResponse) + case e => + unknownErrorCounter.incr() + logger.error("Failure to load result.", e) + Future.value(DefaultWorkerIntersectionResponse) + } + } + ).map { responses => + Stat.time(aggregatorLatency) { + gfsIntersectionResponseAggregator( + responses, + calculatedFeatureTypes, + request.candidateUserIds, + intersectionIdLimit + ) + } + } + + ks.map { query => + query -> resultFuture.map(_.get(query.candidateId)) + }.toMap + } + } + + /** + * Function to merge GfsIntersectionResponse from workers into one result. + */ + private def gfsIntersectionResponseAggregator( + responseList: Seq[WorkerIntersectionResponse], + features: Seq[FeatureType], + candidates: Seq[Long], + intersectionIdLimit: Int + ): Map[Long, CachedIntersectionResult] = { + + // Map of (candidate -> features -> type -> value) + val cube = Array.fill[Int](candidates.length, features.length, 3)(0) + // Map of (candidate -> features -> intersectionIds) + val ids = Array.fill[Option[ArrayBuffer[Long]]](candidates.length, features.length)(None) + val notZero = intersectionIdLimit != 0 + + for { + response <- responseList + (features, candidateIndex) <- response.results.zipWithIndex + (workerValue, featureIndex) <- features.zipWithIndex + } { + cube(candidateIndex)(featureIndex)(CountIndex) += workerValue.count + cube(candidateIndex)(featureIndex)(LeftDegreeIndex) += workerValue.leftNodeDegree + cube(candidateIndex)(featureIndex)(RightDegreeIndex) += workerValue.rightNodeDegree + + if (notZero && workerValue.intersectionIds.nonEmpty) { + val arrayBuffer = ids(candidateIndex)(featureIndex) match { + case Some(buffer) => buffer + case None => + val buffer = ArrayBuffer[Long]() + ids(candidateIndex)(featureIndex) = Some(buffer) + buffer + } + val intersectionIds = workerValue.intersectionIds + + // Scan the intersectionId based on the Shard. The response order is consistent. + if (arrayBuffer.size < intersectionIdLimit) { + if (intersectionIds.size > intersectionIdLimit - arrayBuffer.size) { + arrayBuffer ++= intersectionIds.slice(0, intersectionIdLimit - arrayBuffer.size) + } else { + arrayBuffer ++= intersectionIds + } + } + } + } + + candidates.zipWithIndex.map { + case (candidate, candidateIndex) => + candidate -> CachedIntersectionResult(features.indices.map { featureIndex => + WorkerIntersectionValue( + cube(candidateIndex)(featureIndex)(CountIndex), + cube(candidateIndex)(featureIndex)(LeftDegreeIndex), + cube(candidateIndex)(featureIndex)(RightDegreeIndex), + ids(candidateIndex)(featureIndex).getOrElse(Nil) + ) + }) + }.toMap + } + +} + +object GetIntersectionStore { + + private[graph_feature_service] case class GetIntersectionQuery( + userId: Long, + candidateId: Long, + featureTypes: Seq[FeatureType], + presetFeatureTypes: PresetFeatureTypes, + featureTypesString: String, + calculatedFeatureTypes: Seq[FeatureType], + intersectionIdLimit: Int) + + private[graph_feature_service] object GetIntersectionQuery { + def buildQueries(request: GetIntersectionRequest): Set[GetIntersectionQuery] = { + request.candidateUserIds.toSet.map { candidateId: Long => + GetIntersectionQuery( + request.userId, + candidateId, + request.featureTypes, + request.presetFeatureTypes, + request.calculatedFeatureTypesString, + request.calculatedFeatureTypes, + request.intersectionIdLimit.getOrElse(DefaultIntersectionIdLimit) + ) + } + } + } + + // Don't return the intersectionId for better performance + private val DefaultIntersectionIdLimit = 0 + private val DefaultWorkerIntersectionResponse = WorkerIntersectionResponse() + + private val CountIndex = 0 + private val LeftDegreeIndex = 1 + private val RightDegreeIndex = 2 +} diff --git a/graph-feature-service/src/main/scala/com/twitter/graph_feature_service/util/BUILD b/graph-feature-service/src/main/scala/com/twitter/graph_feature_service/util/BUILD new file mode 100644 index 0000000000..7aa2bc51cd --- /dev/null +++ b/graph-feature-service/src/main/scala/com/twitter/graph_feature_service/util/BUILD @@ -0,0 +1,7 @@ +scala_library( + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "graph-feature-service/src/main/thrift/com/twitter/graph_feature_service:graph_feature_service_thrift-scala", + ], +) diff --git a/graph-feature-service/src/main/scala/com/twitter/graph_feature_service/util/FeatureTypesCalculator.scala b/graph-feature-service/src/main/scala/com/twitter/graph_feature_service/util/FeatureTypesCalculator.scala new file mode 100644 index 0000000000..86caad2bf3 --- /dev/null +++ b/graph-feature-service/src/main/scala/com/twitter/graph_feature_service/util/FeatureTypesCalculator.scala @@ -0,0 +1,58 @@ +package com.twitter.graph_feature_service.util + +import com.twitter.graph_feature_service.thriftscala.EdgeType._ +import com.twitter.graph_feature_service.thriftscala.{FeatureType, PresetFeatureTypes} + +object FeatureTypesCalculator { + + final val DefaultTwoHop = Seq( + FeatureType(Following, FollowedBy), + FeatureType(Following, FavoritedBy), + FeatureType(Following, RetweetedBy), + FeatureType(Following, MentionedBy), + FeatureType(Following, MutualFollow), + FeatureType(Favorite, FollowedBy), + FeatureType(Favorite, FavoritedBy), + FeatureType(Favorite, RetweetedBy), + FeatureType(Favorite, MentionedBy), + FeatureType(Favorite, MutualFollow), + FeatureType(MutualFollow, FollowedBy), + FeatureType(MutualFollow, FavoritedBy), + FeatureType(MutualFollow, RetweetedBy), + FeatureType(MutualFollow, MentionedBy), + FeatureType(MutualFollow, MutualFollow) + ) + + final val SocialProofTwoHop = Seq(FeatureType(Following, FollowedBy)) + + final val HtlTwoHop = DefaultTwoHop + + final val WtfTwoHop = SocialProofTwoHop + + final val SqTwoHop = DefaultTwoHop + + final val RuxTwoHop = DefaultTwoHop + + final val MRTwoHop = DefaultTwoHop + + final val UserTypeaheadTwoHop = SocialProofTwoHop + + final val presetFeatureTypes = + (HtlTwoHop ++ WtfTwoHop ++ SqTwoHop ++ RuxTwoHop ++ MRTwoHop ++ UserTypeaheadTwoHop).toSet + + def getFeatureTypes( + presetFeatureTypes: PresetFeatureTypes, + featureTypes: Seq[FeatureType] + ): Seq[FeatureType] = { + presetFeatureTypes match { + case PresetFeatureTypes.HtlTwoHop => HtlTwoHop + case PresetFeatureTypes.WtfTwoHop => WtfTwoHop + case PresetFeatureTypes.SqTwoHop => SqTwoHop + case PresetFeatureTypes.RuxTwoHop => RuxTwoHop + case PresetFeatureTypes.MrTwoHop => MRTwoHop + case PresetFeatureTypes.UserTypeaheadTwoHop => UserTypeaheadTwoHop + case _ => featureTypes + } + } + +} diff --git a/graph-feature-service/src/main/scala/com/twitter/graph_feature_service/util/IntersectionValueCalculator.scala b/graph-feature-service/src/main/scala/com/twitter/graph_feature_service/util/IntersectionValueCalculator.scala new file mode 100644 index 0000000000..4e1376cc45 --- /dev/null +++ b/graph-feature-service/src/main/scala/com/twitter/graph_feature_service/util/IntersectionValueCalculator.scala @@ -0,0 +1,242 @@ +package com.twitter.graph_feature_service.util + +import com.twitter.graph_feature_service.thriftscala.{ + FeatureType, + IntersectionValue, + WorkerIntersectionValue +} +import java.nio.ByteBuffer +import scala.collection.mutable.ArrayBuffer + +/** + * Functions for computing feature values based on the values returned by constantDB. + */ +object IntersectionValueCalculator { + + /** + * Compute the size of the array in a ByteBuffer. + * Note that this function assumes the ByteBuffer is encoded using Injections.seqLong2ByteBuffer + */ + def computeArraySize(x: ByteBuffer): Int = { + x.remaining() >> 3 // divide 8 + } + + /** + * + */ + def apply(x: ByteBuffer, y: ByteBuffer, intersectionIdLimit: Int): WorkerIntersectionValue = { + + val xSize = computeArraySize(x) + val ySize = computeArraySize(y) + + val largerArray = if (xSize > ySize) x else y + val smallerArray = if (xSize > ySize) y else x + + if (intersectionIdLimit == 0) { + val result = computeIntersectionUsingBinarySearchOnLargerByteBuffer(smallerArray, largerArray) + WorkerIntersectionValue(result, xSize, ySize) + } else { + val (result, ids) = computeIntersectionWithIds(smallerArray, largerArray, intersectionIdLimit) + WorkerIntersectionValue(result, xSize, ySize, ids) + } + } + + /** + * Note that this function assumes the ByteBuffer is encoded using Injections.seqLong2ByteBuffer + * + */ + def computeIntersectionUsingBinarySearchOnLargerByteBuffer( + smallArray: ByteBuffer, + largeArray: ByteBuffer + ): Int = { + var res: Int = 0 + var i: Int = 0 + + while (i < smallArray.remaining()) { + if (binarySearch(largeArray, smallArray.getLong(i)) >= 0) { + res += 1 + } + i += 8 + } + res + } + + def computeIntersectionWithIds( + smallArray: ByteBuffer, + largeArray: ByteBuffer, + intersectionLimit: Int + ): (Int, Seq[Long]) = { + var res: Int = 0 + var i: Int = 0 + // Most of the intersectionLimit is smaller than default size: 16 + val idBuffer = ArrayBuffer[Long]() + + while (i < smallArray.remaining()) { + val value = smallArray.getLong(i) + if (binarySearch(largeArray, value) >= 0) { + res += 1 + // Always get the smaller ids + if (idBuffer.size < intersectionLimit) { + idBuffer += value + } + } + i += 8 + } + (res, idBuffer) + } + + /** + * Note that this function assumes the ByteBuffer is encoded using Injections.seqLong2ByteBuffer + * + */ + private[util] def binarySearch(arr: ByteBuffer, value: Long): Int = { + var start = 0 + var end = arr.remaining() + + while (start <= end && start < arr.remaining()) { + val mid = ((start + end) >> 1) & ~7 // take mid - mid % 8 + if (arr.getLong(mid) == value) { + return mid // return the index of the value + } else if (arr.getLong(mid) < value) { + start = mid + 8 + } else { + end = mid - 1 + } + } + // if not existed, return -1 + -1 + } + + /** + * TODO: for now it only computes intersection size. Will add more feature types (e.g., dot + * product, maximum value). + * + * NOTE that this function assumes both x and y are SORTED arrays. + * In graph feature service, the sorting is done in the offline Scalding job. + * + * @param x source user's array + * @param y candidate user's array + * @param featureType feature type + * @return + */ + def apply(x: Array[Long], y: Array[Long], featureType: FeatureType): IntersectionValue = { + + val xSize = x.length + val ySize = y.length + + val intersection = + if (xSize.min(ySize) * math.log(xSize.max(ySize)) < (xSize + ySize).toDouble) { + if (xSize < ySize) { + computeIntersectionUsingBinarySearchOnLargerArray(x, y) + } else { + computeIntersectionUsingBinarySearchOnLargerArray(y, x) + } + } else { + computeIntersectionUsingListMerging(x, y) + } + + IntersectionValue( + featureType, + Some(intersection.toInt), + None, // return None for now + Some(xSize), + Some(ySize) + ) + } + + /** + * Function for computing the intersections of two SORTED arrays by list merging. + * + * @param x one array + * @param y another array + * @param ordering ordering function for comparing values of T + * @tparam T type + * @return The intersection size and the list of intersected elements + */ + private[util] def computeIntersectionUsingListMerging[T]( + x: Array[T], + y: Array[T] + )( + implicit ordering: Ordering[T] + ): Int = { + + var res: Int = 0 + var i: Int = 0 + var j: Int = 0 + + while (i < x.length && j < y.length) { + val comp = ordering.compare(x(i), y(j)) + if (comp > 0) j += 1 + else if (comp < 0) i += 1 + else { + res += 1 + i += 1 + j += 1 + } + } + res + } + + /** + * Function for computing the intersections of two arrays by binary search on the larger array. + * Note that the larger array MUST be SORTED. + * + * @param smallArray smaller array + * @param largeArray larger array + * @param ordering ordering function for comparing values of T + * @tparam T type + * + * @return The intersection size and the list of intersected elements + */ + private[util] def computeIntersectionUsingBinarySearchOnLargerArray[T]( + smallArray: Array[T], + largeArray: Array[T] + )( + implicit ordering: Ordering[T] + ): Int = { + var res: Int = 0 + var i: Int = 0 + while (i < smallArray.length) { + val currentValue: T = smallArray(i) + if (binarySearch(largeArray, currentValue) >= 0) { + res += 1 + } + i += 1 + } + res + } + + /** + * Function for doing the binary search + * + * @param arr array + * @param value the target value for searching + * @param ordering ordering function + * @tparam T type + * @return the index of element in the larger array. + * If there is no such element in the array, return -1. + */ + private[util] def binarySearch[T]( + arr: Array[T], + value: T + )( + implicit ordering: Ordering[T] + ): Int = { + var start = 0 + var end = arr.length - 1 + + while (start <= end) { + val mid = (start + end) >> 1 + val comp = ordering.compare(arr(mid), value) + if (comp == 0) { + return mid // return the index of the value + } else if (comp < 0) { + start = mid + 1 + } else { + end = mid - 1 + } + } + // if not existed, return -1 + -1 + } +} diff --git a/graph-feature-service/src/main/scala/com/twitter/graph_feature_service/worker/BUILD.bazel b/graph-feature-service/src/main/scala/com/twitter/graph_feature_service/worker/BUILD.bazel new file mode 100644 index 0000000000..7f0d975d7d --- /dev/null +++ b/graph-feature-service/src/main/scala/com/twitter/graph_feature_service/worker/BUILD.bazel @@ -0,0 +1,30 @@ +scala_library( + sources = ["**/*.scala"], + platform = "java8", + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/com/google/inject:guice", + "3rdparty/jvm/javax/inject:javax.inject", + "3rdparty/jvm/net/codingwell:scala-guice", + "discovery-common/src/main/scala/com/twitter/discovery/common/stats", + "finatra-internal/decider/src/main/scala", + "finatra-internal/gizmoduck/src/main/scala", + "finatra-internal/mtls-thriftmux/src/main/scala", + "finatra/inject/inject-app/src/main/scala", + "finatra/inject/inject-core/src/main/scala", + "finatra/inject/inject-server/src/main/scala", + "finatra/inject/inject-thrift-client/src/main/scala", + "finatra/inject/inject-utils/src/main/scala", + "frigate/frigate-common:constdb_util", + "graph-feature-service/src/main/resources", + "graph-feature-service/src/main/scala/com/twitter/graph_feature_service/common", + "graph-feature-service/src/main/scala/com/twitter/graph_feature_service/util", + "graph-feature-service/src/main/thrift/com/twitter/graph_feature_service:graph_feature_service_thrift-scala", + "hermit/hermit-core/src/main/scala/com/twitter/hermit/store/common", + "servo/request/src/main/scala", + "twitter-server-internal/src/main/scala", + "twitter-server/server/src/main/scala", + "util/util-app/src/main/scala", + "util/util-slf4j-api/src/main/scala", + ], +) diff --git a/graph-feature-service/src/main/scala/com/twitter/graph_feature_service/worker/Main.scala b/graph-feature-service/src/main/scala/com/twitter/graph_feature_service/worker/Main.scala new file mode 100644 index 0000000000..10e8ec0e2f --- /dev/null +++ b/graph-feature-service/src/main/scala/com/twitter/graph_feature_service/worker/Main.scala @@ -0,0 +1,58 @@ +package com.twitter.graph_feature_service.worker + +import com.google.inject.Module +import com.twitter.finatra.decider.modules.DeciderModule +import com.twitter.finatra.gizmoduck.modules.TimerModule +import com.twitter.finatra.mtls.thriftmux.Mtls +import com.twitter.finatra.thrift.ThriftServer +import com.twitter.finatra.thrift.filters.{ + LoggingMDCFilter, + StatsFilter, + ThriftMDCFilter, + TraceIdMDCFilter +} +import com.twitter.finatra.mtls.thriftmux.modules.MtlsThriftWebFormsModule +import com.twitter.finatra.thrift.routing.ThriftRouter +import com.twitter.graph_feature_service.thriftscala +import com.twitter.graph_feature_service.worker.controllers.WorkerController +import com.twitter.graph_feature_service.worker.handlers.WorkerWarmupHandler +import com.twitter.graph_feature_service.worker.modules.{ + GraphContainerProviderModule, + WorkerFlagModule +} +import com.twitter.graph_feature_service.worker.util.GraphContainer +import com.twitter.inject.thrift.modules.ThriftClientIdModule +import com.twitter.util.Await + +object Main extends WorkerMain + +class WorkerMain extends ThriftServer with Mtls { + + override val name = "graph_feature_service-worker" + + override val modules: Seq[Module] = { + Seq( + WorkerFlagModule, + DeciderModule, + TimerModule, + ThriftClientIdModule, + GraphContainerProviderModule, + new MtlsThriftWebFormsModule[thriftscala.Worker.MethodPerEndpoint](this) + ) + } + + override def configureThrift(router: ThriftRouter): Unit = { + router + .filter[LoggingMDCFilter] + .filter[TraceIdMDCFilter] + .filter[ThriftMDCFilter] + .filter[StatsFilter] + .add[WorkerController] + } + + override protected def warmup(): Unit = { + val graphContainer = injector.instance[GraphContainer] + Await.result(graphContainer.warmup) + handle[WorkerWarmupHandler]() + } +} diff --git a/graph-feature-service/src/main/scala/com/twitter/graph_feature_service/worker/controllers/WorkerController.scala b/graph-feature-service/src/main/scala/com/twitter/graph_feature_service/worker/controllers/WorkerController.scala new file mode 100644 index 0000000000..f30305b2c1 --- /dev/null +++ b/graph-feature-service/src/main/scala/com/twitter/graph_feature_service/worker/controllers/WorkerController.scala @@ -0,0 +1,38 @@ +package com.twitter.graph_feature_service.worker.controllers + +import com.twitter.discovery.common.stats.DiscoveryStatsFilter +import com.twitter.finagle.Service +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.finatra.thrift.Controller +import com.twitter.graph_feature_service.thriftscala +import com.twitter.graph_feature_service.thriftscala.Worker.GetIntersection +import com.twitter.graph_feature_service.thriftscala._ +import com.twitter.graph_feature_service.worker.handlers._ +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class WorkerController @Inject() ( + workerGetIntersectionHandler: WorkerGetIntersectionHandler +)( + implicit statsReceiver: StatsReceiver) + extends Controller(thriftscala.Worker) { + + // use DiscoveryStatsFilter to filter out exceptions out of our control + private val getIntersectionService: Service[ + WorkerIntersectionRequest, + WorkerIntersectionResponse + ] = + new DiscoveryStatsFilter[WorkerIntersectionRequest, WorkerIntersectionResponse]( + statsReceiver.scope("srv").scope("get_intersection") + ).andThen(Service.mk(workerGetIntersectionHandler)) + + val getIntersection: Service[GetIntersection.Args, WorkerIntersectionResponse] = { args => + getIntersectionService(args.request).onFailure { throwable => + logger.error(s"Failure to get intersection for request $args.", throwable) + } + } + + handle(GetIntersection) { getIntersection } + +} diff --git a/graph-feature-service/src/main/scala/com/twitter/graph_feature_service/worker/handlers/WorkerGetIntersectionHandler.scala b/graph-feature-service/src/main/scala/com/twitter/graph_feature_service/worker/handlers/WorkerGetIntersectionHandler.scala new file mode 100644 index 0000000000..7acf8b1d3e --- /dev/null +++ b/graph-feature-service/src/main/scala/com/twitter/graph_feature_service/worker/handlers/WorkerGetIntersectionHandler.scala @@ -0,0 +1,105 @@ +package com.twitter.graph_feature_service.worker.handlers + +import com.twitter.finagle.stats.{Stat, StatsReceiver} +import com.twitter.graph_feature_service.thriftscala.{ + WorkerIntersectionRequest, + WorkerIntersectionResponse, + WorkerIntersectionValue +} +import com.twitter.graph_feature_service.util.{FeatureTypesCalculator, IntersectionValueCalculator} +import com.twitter.graph_feature_service.util.IntersectionValueCalculator._ +import com.twitter.graph_feature_service.worker.util.GraphContainer +import com.twitter.servo.request.RequestHandler +import com.twitter.util.Future +import java.nio.ByteBuffer +import javax.inject.{Inject, Singleton} + +@Singleton +class WorkerGetIntersectionHandler @Inject() ( + graphContainer: GraphContainer, + statsReceiver: StatsReceiver) + extends RequestHandler[WorkerIntersectionRequest, WorkerIntersectionResponse] { + + import WorkerGetIntersectionHandler._ + + private val stats: StatsReceiver = statsReceiver.scope("srv/get_intersection") + private val numCandidatesCount = stats.counter("total_num_candidates") + private val toPartialGraphQueryStat = stats.stat("to_partial_graph_query_latency") + private val fromPartialGraphQueryStat = stats.stat("from_partial_graph_query_latency") + private val intersectionCalculationStat = stats.stat("computation_latency") + + override def apply(request: WorkerIntersectionRequest): Future[WorkerIntersectionResponse] = { + + numCandidatesCount.incr(request.candidateUserIds.length) + + val userId = request.userId + + // NOTE: do not change the order of candidates + val candidateIds = request.candidateUserIds + + // NOTE: do not change the order of features + val featureTypes = + FeatureTypesCalculator.getFeatureTypes(request.presetFeatureTypes, request.featureTypes) + + val leftEdges = featureTypes.map(_.leftEdgeType).distinct + val rightEdges = featureTypes.map(_.rightEdgeType).distinct + + val rightEdgeMap = Stat.time(toPartialGraphQueryStat) { + rightEdges.map { rightEdge => + val map = graphContainer.toPartialMap.get(rightEdge) match { + case Some(graph) => + candidateIds.flatMap { candidateId => + graph.apply(candidateId).map(candidateId -> _) + }.toMap + case None => + Map.empty[Long, ByteBuffer] + } + rightEdge -> map + }.toMap + } + + val leftEdgeMap = Stat.time(fromPartialGraphQueryStat) { + leftEdges.flatMap { leftEdge => + graphContainer.toPartialMap.get(leftEdge).flatMap(_.apply(userId)).map(leftEdge -> _) + }.toMap + } + + val res = Stat.time(intersectionCalculationStat) { + WorkerIntersectionResponse( + // NOTE that candidate ordering is important + candidateIds.map { candidateId => + // NOTE that the featureTypes ordering is important + featureTypes.map { + featureType => + val leftNeighborsOpt = leftEdgeMap.get(featureType.leftEdgeType) + val rightNeighborsOpt = + rightEdgeMap.get(featureType.rightEdgeType).flatMap(_.get(candidateId)) + + if (leftNeighborsOpt.isEmpty && rightNeighborsOpt.isEmpty) { + EmptyWorkerIntersectionValue + } else if (rightNeighborsOpt.isEmpty) { + EmptyWorkerIntersectionValue.copy( + leftNodeDegree = computeArraySize(leftNeighborsOpt.get) + ) + } else if (leftNeighborsOpt.isEmpty) { + EmptyWorkerIntersectionValue.copy( + rightNodeDegree = computeArraySize(rightNeighborsOpt.get) + ) + } else { + IntersectionValueCalculator( + leftNeighborsOpt.get, + rightNeighborsOpt.get, + request.intersectionIdLimit) + } + } + } + ) + } + + Future.value(res) + } +} + +object WorkerGetIntersectionHandler { + val EmptyWorkerIntersectionValue: WorkerIntersectionValue = WorkerIntersectionValue(0, 0, 0, Nil) +} diff --git a/graph-feature-service/src/main/scala/com/twitter/graph_feature_service/worker/handlers/WorkerWarmupHandler.scala b/graph-feature-service/src/main/scala/com/twitter/graph_feature_service/worker/handlers/WorkerWarmupHandler.scala new file mode 100644 index 0000000000..d89d215c1e --- /dev/null +++ b/graph-feature-service/src/main/scala/com/twitter/graph_feature_service/worker/handlers/WorkerWarmupHandler.scala @@ -0,0 +1,14 @@ +package com.twitter.graph_feature_service.worker.handlers + +import com.twitter.finatra.thrift.routing.ThriftWarmup +import com.twitter.inject.Logging +import com.twitter.inject.utils.Handler +import javax.inject.{Inject, Singleton} + +@Singleton +class WorkerWarmupHandler @Inject() (warmup: ThriftWarmup) extends Handler with Logging { + + override def handle(): Unit = { + info("Warmup Done!") + } +} diff --git a/graph-feature-service/src/main/scala/com/twitter/graph_feature_service/worker/modules/GraphContainerProviderModule.scala b/graph-feature-service/src/main/scala/com/twitter/graph_feature_service/worker/modules/GraphContainerProviderModule.scala new file mode 100644 index 0000000000..6c66922d4a --- /dev/null +++ b/graph-feature-service/src/main/scala/com/twitter/graph_feature_service/worker/modules/GraphContainerProviderModule.scala @@ -0,0 +1,62 @@ +package com.twitter.graph_feature_service.worker.modules + +import com.google.inject.Provides +import com.twitter.concurrent.AsyncSemaphore +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.graph_feature_service.common.Configs._ +import com.twitter.graph_feature_service.worker.util +import com.twitter.graph_feature_service.worker.util.AutoUpdatingGraph +import com.twitter.graph_feature_service.worker.util.FollowedByPartialValueGraph +import com.twitter.graph_feature_service.worker.util.FollowingPartialValueGraph +import com.twitter.graph_feature_service.worker.util.GraphContainer +import com.twitter.graph_feature_service.worker.util.GraphKey +import com.twitter.graph_feature_service.worker.util.MutualFollowPartialValueGraph +import com.twitter.inject.TwitterModule +import com.twitter.inject.annotations.Flag +import com.twitter.util.Timer +import javax.inject.Singleton + +object GraphContainerProviderModule extends TwitterModule { + + @Provides + @Singleton + def provideAutoUpdatingGraphs( + @Flag(WorkerFlagNames.HdfsCluster) hdfsCluster: String, + @Flag(WorkerFlagNames.HdfsClusterUrl) hdfsClusterUrl: String, + @Flag(WorkerFlagNames.ShardId) shardId: Int + )( + implicit statsReceiver: StatsReceiver, + timer: Timer + ): GraphContainer = { + + // NOTE that we do not load some the graphs for saving RAM at this moment. + val enabledGraphPaths: Map[GraphKey, String] = + Map( + FollowingPartialValueGraph -> FollowOutValPath, + FollowedByPartialValueGraph -> FollowInValPath + ) + + // Only allow one graph to update at the same time. + val sharedSemaphore = new AsyncSemaphore(1) + + val graphs: Map[GraphKey, AutoUpdatingGraph] = + enabledGraphPaths.map { + case (graphKey, path) => + graphKey -> AutoUpdatingGraph( + dataPath = getHdfsPath(path), + hdfsCluster = hdfsCluster, + hdfsClusterUrl = hdfsClusterUrl, + shard = shardId, + minimumSizeForCompleteGraph = 1e6.toLong, + sharedSemaphore = Some(sharedSemaphore) + )( + statsReceiver + .scope("graphs") + .scope(graphKey.getClass.getSimpleName), + timer + ) + } + + util.GraphContainer(graphs) + } +} diff --git a/graph-feature-service/src/main/scala/com/twitter/graph_feature_service/worker/modules/WorkerFlagModule.scala b/graph-feature-service/src/main/scala/com/twitter/graph_feature_service/worker/modules/WorkerFlagModule.scala new file mode 100644 index 0000000000..2188d169b2 --- /dev/null +++ b/graph-feature-service/src/main/scala/com/twitter/graph_feature_service/worker/modules/WorkerFlagModule.scala @@ -0,0 +1,33 @@ +package com.twitter.graph_feature_service.worker.modules + +import com.twitter.inject.TwitterModule + +object WorkerFlagNames { + final val ServiceRole = "service.role" + final val ServiceEnv = "service.env" + final val ShardId = "service.shardId" + final val NumShards = "service.numShards" + final val HdfsCluster = "service.hdfsCluster" + final val HdfsClusterUrl = "service.hdfsClusterUrl" +} + +/** + * Initializes references to the flag values defined in the aurora.deploy file. + * To check what the flag values are initialized in runtime, search FlagsModule in stdout + */ +object WorkerFlagModule extends TwitterModule { + + import WorkerFlagNames._ + + flag[Int](ShardId, "Shard Id") + + flag[Int](NumShards, "Num of Graph Shards") + + flag[String](ServiceRole, "Service Role") + + flag[String](ServiceEnv, "Service Env") + + flag[String](HdfsCluster, "Hdfs cluster to download graph files from") + + flag[String](HdfsClusterUrl, "Hdfs cluster url to download graph files from") +} diff --git a/graph-feature-service/src/main/scala/com/twitter/graph_feature_service/worker/util/AutoUpdatingGraph.scala b/graph-feature-service/src/main/scala/com/twitter/graph_feature_service/worker/util/AutoUpdatingGraph.scala new file mode 100644 index 0000000000..e5c1c367fc --- /dev/null +++ b/graph-feature-service/src/main/scala/com/twitter/graph_feature_service/worker/util/AutoUpdatingGraph.scala @@ -0,0 +1,69 @@ +package com.twitter.graph_feature_service.worker.util + +import com.twitter.bijection.Injection +import com.twitter.concurrent.AsyncSemaphore +import com.twitter.conversions.DurationOps._ +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.frigate.common.constdb_util.{ + AutoUpdatingReadOnlyGraph, + ConstDBImporter, + Injections +} +import com.twitter.graph_feature_service.common.Configs +import com.twitter.util.{Duration, Future, Timer} +import java.nio.ByteBuffer + +/** + * @param dataPath the path to the data on HDFS + * @param hdfsCluster cluster where we check for updates and download graph files from + * @param hdfsClusterUrl url to HDFS cluster + * @param shard The shard of the graph to download + * @param minimumSizeForCompleteGraph minimumSize for complete graph - otherwise we don't load it + * @param updateIntervalMin The interval after which the first update is tried and the interval between such updates + * @param updateIntervalMax the maximum time before an update is triggered + * @param deleteInterval The interval after which older data is deleted from disk + * @param sharedSemaphore The semaphore controls the number of graph loads at same time on the instance. + */ +case class AutoUpdatingGraph( + dataPath: String, + hdfsCluster: String, + hdfsClusterUrl: String, + shard: Int, + minimumSizeForCompleteGraph: Long, + updateIntervalMin: Duration = 1.hour, + updateIntervalMax: Duration = 12.hours, + deleteInterval: Duration = 2.seconds, + sharedSemaphore: Option[AsyncSemaphore] = None +)( + implicit statsReceiver: StatsReceiver, + timer: Timer) + extends AutoUpdatingReadOnlyGraph[Long, ByteBuffer]( + hdfsCluster, + hdfsClusterUrl, + shard, + minimumSizeForCompleteGraph, + updateIntervalMin, + updateIntervalMax, + deleteInterval, + sharedSemaphore + ) + with ConstDBImporter[Long, ByteBuffer] { + + override def numGraphShards: Int = Configs.NumGraphShards + + override def basePath: String = dataPath + + override val keyInj: Injection[Long, ByteBuffer] = Injections.long2Varint + + override val valueInj: Injection[ByteBuffer, ByteBuffer] = Injection.identity + + override def get(targetId: Long): Future[Option[ByteBuffer]] = + super + .get(targetId) + .map { res => + res.foreach(r => arraySizeStat.add(r.remaining())) + res + } + + private val arraySizeStat = stats.scope("get").stat("size") +} diff --git a/graph-feature-service/src/main/scala/com/twitter/graph_feature_service/worker/util/GfsQuery.scala b/graph-feature-service/src/main/scala/com/twitter/graph_feature_service/worker/util/GfsQuery.scala new file mode 100644 index 0000000000..e5d822e2ba --- /dev/null +++ b/graph-feature-service/src/main/scala/com/twitter/graph_feature_service/worker/util/GfsQuery.scala @@ -0,0 +1,14 @@ +package com.twitter.graph_feature_service.worker.util + +import com.twitter.graph_feature_service.thriftscala.EdgeType + +sealed trait GfsQuery { + def edgeType: EdgeType + def userId: Long +} + +/** + * Search for edges for any users to users in local partition. + */ +case class ToPartialQuery(edgeType: EdgeType, userId: Long) extends GfsQuery + diff --git a/graph-feature-service/src/main/scala/com/twitter/graph_feature_service/worker/util/GraphContainer.scala b/graph-feature-service/src/main/scala/com/twitter/graph_feature_service/worker/util/GraphContainer.scala new file mode 100644 index 0000000000..9ac626bb97 --- /dev/null +++ b/graph-feature-service/src/main/scala/com/twitter/graph_feature_service/worker/util/GraphContainer.scala @@ -0,0 +1,19 @@ +package com.twitter.graph_feature_service.worker.util + +import com.twitter.graph_feature_service.thriftscala.EdgeType +import com.twitter.util.Future + +case class GraphContainer( + graphs: Map[GraphKey, AutoUpdatingGraph]) { + + final val toPartialMap: Map[EdgeType, AutoUpdatingGraph] = + graphs.collect { + case (partialValueGraph: PartialValueGraph, graph) => + partialValueGraph.edgeType -> graph + } + + // load all the graphs from constantDB format to memory + def warmup: Future[Unit] = { + Future.collect(graphs.mapValues(_.warmup())).unit + } +} diff --git a/graph-feature-service/src/main/scala/com/twitter/graph_feature_service/worker/util/GraphKey.scala b/graph-feature-service/src/main/scala/com/twitter/graph_feature_service/worker/util/GraphKey.scala new file mode 100644 index 0000000000..2b174eb8c6 --- /dev/null +++ b/graph-feature-service/src/main/scala/com/twitter/graph_feature_service/worker/util/GraphKey.scala @@ -0,0 +1,32 @@ +package com.twitter.graph_feature_service.worker.util + +import com.twitter.graph_feature_service.thriftscala.EdgeType +import com.twitter.graph_feature_service.thriftscala.EdgeType._ + +sealed trait GraphKey { + + def edgeType: EdgeType +} + +sealed trait PartialValueGraph extends GraphKey + +/** + * Follow Graphs + */ +object FollowingPartialValueGraph extends PartialValueGraph { + + override def edgeType: EdgeType = Following +} + +object FollowedByPartialValueGraph extends PartialValueGraph { + + override def edgeType: EdgeType = FollowedBy +} + +/** + * Mutual Follow Graphs + */ +object MutualFollowPartialValueGraph extends PartialValueGraph { + + override def edgeType: EdgeType = MutualFollow +} diff --git a/graph-feature-service/src/main/scala/com/twitter/graph_feature_service/worker/util/GraphType.scala b/graph-feature-service/src/main/scala/com/twitter/graph_feature_service/worker/util/GraphType.scala new file mode 100644 index 0000000000..a2a7634af5 --- /dev/null +++ b/graph-feature-service/src/main/scala/com/twitter/graph_feature_service/worker/util/GraphType.scala @@ -0,0 +1,16 @@ +package com.twitter.graph_feature_service.worker.util + +//These classes are to help the GraphContainer choose the right data structure to answer queries +sealed trait GraphType + +object FollowGraph extends GraphType + +object FavoriteGraph extends GraphType + +object RetweetGraph extends GraphType + +object ReplyGraph extends GraphType + +object MentionGraph extends GraphType + +object MutualFollowGraph extends GraphType diff --git a/graph-feature-service/src/main/scalding/com/twitter/graph_feature_service/scalding/BUILD.bazel b/graph-feature-service/src/main/scalding/com/twitter/graph_feature_service/scalding/BUILD.bazel new file mode 100644 index 0000000000..66e27fb7fc --- /dev/null +++ b/graph-feature-service/src/main/scalding/com/twitter/graph_feature_service/scalding/BUILD.bazel @@ -0,0 +1,66 @@ +scala_library( + platform = "java8", + tags = [ + "bazel-compatible", + "bazel-only", + ], + dependencies = [ + "3rdparty/jvm/com/twitter/bijection:core", + "frigate/frigate-common/src/main/scala/com/twitter/frigate/common/constdb_util", + "graph-feature-service/src/main/scala/com/twitter/graph_feature_service/common", + "src/scala/com/twitter/interaction_graph/scio/agg_all:interaction_graph_history_aggregated_edge_snapshot-scala", + "src/scala/com/twitter/interaction_graph/scio/ml/scores:real_graph_in_scores-scala", + "src/scala/com/twitter/pluck/source/user_audits:user_audit_final-scala", + "src/scala/com/twitter/scalding_internal/dalv2", + "src/scala/com/twitter/scalding_internal/job", + "src/scala/com/twitter/scalding_internal/job/analytics_batch", + ], +) + +scalding_job( + name = "graph_feature_service_adhoc_job", + main = "com.twitter.graph_feature_service.scalding.GraphFeatureServiceAdhocApp", + args = [ + "--date 2022-10-24", + ], + config = [ + ("hadoop.map.jvm.total-memory", "3072m"), + ("hadoop.reduce.jvm.total-memory", "3072m"), + ("hadoop.submitter.jvm.total-memory", "5120m"), + ("submitter.tier", "preemptible"), + ], + contact = "recos-platform-alerts@twitter.com", + hadoop_cluster = "atla-proc", + hadoop_properties = [("mapreduce.job.hdfs-servers", "/atla/proc/user/cassowary")], + platform = "java8", + role = "cassowary", + runtime_platform = "java8", + tags = [ + "bazel-compatible:migrated", + "bazel-only", + ], + dependencies = [":scalding"], +) + +scalding_job( + name = "graph_feature_service_daily_job", + main = "com.twitter.graph_feature_service.scalding.GraphFeatureServiceScheduledApp", + config = [ + ("hadoop.map.jvm.total-memory", "3072m"), + ("hadoop.reduce.jvm.total-memory", "3072m"), + ("hadoop.submitter.jvm.total-memory", "5120m"), + ("submitter.tier", "preemptible"), + ], + contact = "recos-platform-alerts@twitter.com", + cron = "01,31 * * * *", + hadoop_cluster = "atla-proc", + hadoop_properties = [("mapreduce.job.hdfs-servers", "/atla/proc/user/cassowary")], + platform = "java8", + role = "cassowary", + runtime_platform = "java8", + tags = [ + "bazel-compatible:migrated", + "bazel-only", + ], + dependencies = [":scalding"], +) diff --git a/graph-feature-service/src/main/scalding/com/twitter/graph_feature_service/scalding/EdgeFeature.scala b/graph-feature-service/src/main/scalding/com/twitter/graph_feature_service/scalding/EdgeFeature.scala new file mode 100644 index 0000000000..76005c3aea --- /dev/null +++ b/graph-feature-service/src/main/scalding/com/twitter/graph_feature_service/scalding/EdgeFeature.scala @@ -0,0 +1,9 @@ +package com.twitter.graph_feature_service.scalding + +case class EdgeFeature( + realGraphScore: Float, + followScore: Option[Float] = None, + mutualFollowScore: Option[Float] = None, + favoriteScore: Option[Float] = None, + retweetScore: Option[Float] = None, + mentionScore: Option[Float] = None) diff --git a/graph-feature-service/src/main/scalding/com/twitter/graph_feature_service/scalding/GraphFeatureServiceAppBase.scala b/graph-feature-service/src/main/scalding/com/twitter/graph_feature_service/scalding/GraphFeatureServiceAppBase.scala new file mode 100644 index 0000000000..993dc5a391 --- /dev/null +++ b/graph-feature-service/src/main/scalding/com/twitter/graph_feature_service/scalding/GraphFeatureServiceAppBase.scala @@ -0,0 +1,85 @@ +package com.twitter.graph_feature_service.scalding + +import com.twitter.scalding._ +import com.twitter.scalding_internal.job.TwitterExecutionApp +import com.twitter.scalding_internal.job.analytics_batch.{ + AnalyticsBatchExecution, + AnalyticsBatchExecutionArgs, + BatchDescription, + BatchFirstTime, + BatchIncrement, + TwitterScheduledExecutionApp +} +import java.util.TimeZone + +/** + * Each job only needs to implement this runOnDateRange() function. It makes it easier for testing. + */ +trait GraphFeatureServiceBaseJob { + implicit val timeZone: TimeZone = DateOps.UTC + implicit val dateParser: DateParser = DateParser.default + + def runOnDateRange( + enableValueGraphs: Option[Boolean] = None, + enableKeyGraphs: Option[Boolean] = None + )( + implicit dateRange: DateRange, + timeZone: TimeZone, + uniqueID: UniqueID + ): Execution[Unit] + + /** + * Print customized counters in the log + */ + def printerCounters[T](execution: Execution[T]): Execution[Unit] = { + execution.getCounters + .flatMap { + case (_, counters) => + counters.toMap.toSeq + .sortBy(e => (e._1.group, e._1.counter)) + .foreach { + case (statKey, value) => + println(s"${statKey.group}\t${statKey.counter}\t$value") + } + Execution.unit + } + } +} + +/** + * Trait that wraps things about adhoc jobs. + */ +trait GraphFeatureServiceAdhocBaseApp extends TwitterExecutionApp with GraphFeatureServiceBaseJob { + override def job: Execution[Unit] = Execution.withId { implicit uniqueId => + Execution.getArgs.flatMap { args: Args => + implicit val dateRange: DateRange = DateRange.parse(args.list("date"))(timeZone, dateParser) + printerCounters(runOnDateRange()) + } + } +} + +/** + * Trait that wraps things about scheduled jobs. + * + * A new daily app only needs to declare the starting date. + */ +trait GraphFeatureServiceScheduledBaseApp + extends TwitterScheduledExecutionApp + with GraphFeatureServiceBaseJob { + + def firstTime: RichDate // for example: RichDate("2018-02-21") + + def batchIncrement: Duration = Days(1) + + override def scheduledJob: Execution[Unit] = Execution.withId { implicit uniqueId => + val analyticsArgs = AnalyticsBatchExecutionArgs( + batchDesc = BatchDescription(getClass.getName), + firstTime = BatchFirstTime(firstTime), + batchIncrement = BatchIncrement(batchIncrement) + ) + + AnalyticsBatchExecution(analyticsArgs) { implicit dateRange => + printerCounters(runOnDateRange()) + } + } +} diff --git a/graph-feature-service/src/main/scalding/com/twitter/graph_feature_service/scalding/GraphFeatureServiceApps.scala b/graph-feature-service/src/main/scalding/com/twitter/graph_feature_service/scalding/GraphFeatureServiceApps.scala new file mode 100644 index 0000000000..e6086526be --- /dev/null +++ b/graph-feature-service/src/main/scalding/com/twitter/graph_feature_service/scalding/GraphFeatureServiceApps.scala @@ -0,0 +1,52 @@ +package com.twitter.graph_feature_service.scalding + +import com.twitter.scalding.DateRange +import com.twitter.scalding.Execution +import com.twitter.scalding.RichDate +import com.twitter.scalding.UniqueID +import java.util.Calendar +import java.util.TimeZone +import sun.util.calendar.BaseCalendar + +/** + * To launch an adhoc run: + * + scalding remote run --target graph-feature-service/src/main/scalding/com/twitter/graph_feature_service/scalding:graph_feature_service_adhoc_job + */ +object GraphFeatureServiceAdhocApp + extends GraphFeatureServiceMainJob + with GraphFeatureServiceAdhocBaseApp {} + +/** + * To schedule the job, upload the workflows config (only required for the first time and subsequent config changes): + * scalding workflow upload --jobs graph-feature-service/src/main/scalding/com/twitter/graph_feature_service/scalding:graph_feature_service_daily_job --autoplay --build-cron-schedule "20 23 1 * *" + * You can then build from the UI by clicking "Build" and pasting in your remote branch, or leave it empty if you're redeploying from master. + * The workflows config above should automatically trigger once each month. + */ +object GraphFeatureServiceScheduledApp + extends GraphFeatureServiceMainJob + with GraphFeatureServiceScheduledBaseApp { + override def firstTime: RichDate = RichDate("2018-05-18") + + override def runOnDateRange( + enableValueGraphs: Option[Boolean], + enableKeyGraphs: Option[Boolean] + )( + implicit dateRange: DateRange, + timeZone: TimeZone, + uniqueID: UniqueID + ): Execution[Unit] = { + // Only run the value Graphs on Tuesday, Thursday, Saturday + val overrideEnableValueGraphs = { + val dayOfWeek = dateRange.start.toCalendar.get(Calendar.DAY_OF_WEEK) + dayOfWeek == BaseCalendar.TUESDAY | + dayOfWeek == BaseCalendar.THURSDAY | + dayOfWeek == BaseCalendar.SATURDAY + } + + super.runOnDateRange( + Some(true), + Some(false) // disable key Graphs since we are not using them in production + ) + } +} diff --git a/graph-feature-service/src/main/scalding/com/twitter/graph_feature_service/scalding/GraphFeatureServiceMainJob.scala b/graph-feature-service/src/main/scalding/com/twitter/graph_feature_service/scalding/GraphFeatureServiceMainJob.scala new file mode 100644 index 0000000000..f0446285cc --- /dev/null +++ b/graph-feature-service/src/main/scalding/com/twitter/graph_feature_service/scalding/GraphFeatureServiceMainJob.scala @@ -0,0 +1,297 @@ +package com.twitter.graph_feature_service.scalding + +import com.twitter.bijection.Injection +import com.twitter.frigate.common.constdb_util.Injections +import com.twitter.frigate.common.constdb_util.ScaldingUtil +import com.twitter.graph_feature_service.common.Configs +import com.twitter.graph_feature_service.common.Configs._ +import com.twitter.interaction_graph.scio.agg_all.InteractionGraphHistoryAggregatedEdgeSnapshotScalaDataset +import com.twitter.interaction_graph.scio.ml.scores.RealGraphInScoresScalaDataset +import com.twitter.interaction_graph.thriftscala.FeatureName +import com.twitter.interaction_graph.thriftscala.{EdgeFeature => TEdgeFeature} +import com.twitter.pluck.source.user_audits.UserAuditFinalScalaDataset +import com.twitter.scalding.DateRange +import com.twitter.scalding.Days +import com.twitter.scalding.Execution +import com.twitter.scalding.Stat +import com.twitter.scalding.UniqueID +import com.twitter.scalding.typed.TypedPipe +import com.twitter.scalding_internal.dalv2.DAL +import com.twitter.scalding_internal.dalv2.remote_access.AllowCrossClusterSameDC +import com.twitter.scalding_internal.multiformat.format.keyval.KeyVal +import com.twitter.util.Time +import com.twitter.wtf.candidate.thriftscala.CandidateSeq +import java.nio.ByteBuffer +import java.util.TimeZone + +trait GraphFeatureServiceMainJob extends GraphFeatureServiceBaseJob { + + // keeping hdfsPath as a separate variable in order to override it in unit tests + protected val hdfsPath: String = BaseHdfsPath + + protected def getShardIdForUser(userId: Long): Int = shardForUser(userId) + + protected implicit val keyInj: Injection[Long, ByteBuffer] = Injections.long2Varint + + protected implicit val valueInj: Injection[Long, ByteBuffer] = Injections.long2ByteBuffer + + protected val bufferSize: Int = 1 << 26 + + protected val maxNumKeys: Int = 1 << 24 + + protected val numReducers: Int = NumGraphShards + + protected val outputStreamBufferSize: Int = 1 << 26 + + protected final val shardingByKey = { (k: Long, _: Long) => + getShardIdForUser(k) + } + + protected final val shardingByValue = { (_: Long, v: Long) => + getShardIdForUser(v) + } + + private def writeGraphToDB( + graph: TypedPipe[(Long, Long)], + shardingFunction: (Long, Long) => Int, + path: String + )( + implicit dateRange: DateRange + ): Execution[TypedPipe[(Int, Unit)]] = { + ScaldingUtil + .writeConstDB[Long, Long]( + graph.withDescription(s"sharding $path"), + shardingFunction, + shardId => + getTimedHdfsShardPath( + shardId, + getHdfsPath(path, Some(hdfsPath)), + Time.fromMilliseconds(dateRange.end.timestamp) + ), + Int.MaxValue, + bufferSize, + maxNumKeys, + numReducers, + outputStreamBufferSize + )( + keyInj, + valueInj, + Ordering[(Long, Long)] + ) + .forceToDiskExecution + } + + def extractFeature( + featureList: Seq[TEdgeFeature], + featureName: FeatureName + ): Option[Float] = { + featureList + .find(_.name == featureName) + .map(_.tss.ewma.toFloat) + .filter(_ > 0.0) + } + + /** + * Function to extract a subgraph (e.g., follow graph) from real graph and take top K by real graph + * weight. + * + * @param input input real graph + * @param edgeFilter filter function to only get the edges needed (e.g., only follow edges) + * @param counter counter + * @return a subgroup that contains topK, e.g., follow graph for each user. + */ + private def getSubGraph( + input: TypedPipe[(Long, Long, EdgeFeature)], + edgeFilter: EdgeFeature => Boolean, + counter: Stat + ): TypedPipe[(Long, Long)] = { + input + .filter(c => edgeFilter(c._3)) + .map { + case (srcId, destId, features) => + (srcId, (destId, features.realGraphScore)) + } + .group + // auto reducer estimation only allocates 15 reducers, so setting an explicit number here + .withReducers(2000) + .sortedReverseTake(TopKRealGraph)(Ordering.by(_._2)) + .flatMap { + case (srcId, topKNeighbors) => + counter.inc() + topKNeighbors.map { + case (destId, _) => + (srcId, destId) + } + } + } + + def getMauIds()(implicit dateRange: DateRange, uniqueID: UniqueID): TypedPipe[Long] = { + val numMAUs = Stat("NUM_MAUS") + val uniqueMAUs = Stat("UNIQUE_MAUS") + + DAL + .read(UserAuditFinalScalaDataset) + .withRemoteReadPolicy(AllowCrossClusterSameDC) + .toTypedPipe + .collect { + case user_audit if user_audit.isValid => + numMAUs.inc() + user_audit.userId + } + .distinct + .map { u => + uniqueMAUs.inc() + u + } + } + + def getRealGraphWithMAUOnly( + implicit dateRange: DateRange, + timeZone: TimeZone, + uniqueID: UniqueID + ): TypedPipe[(Long, Long, EdgeFeature)] = { + val numMAUs = Stat("NUM_MAUS") + val uniqueMAUs = Stat("UNIQUE_MAUS") + + val monthlyActiveUsers = DAL + .read(UserAuditFinalScalaDataset) + .withRemoteReadPolicy(AllowCrossClusterSameDC) + .toTypedPipe + .collect { + case user_audit if user_audit.isValid => + numMAUs.inc() + user_audit.userId + } + .distinct + .map { u => + uniqueMAUs.inc() + u + } + .asKeys + + val realGraphAggregates = DAL + .readMostRecentSnapshot( + InteractionGraphHistoryAggregatedEdgeSnapshotScalaDataset, + dateRange.embiggen(Days(5))) + .withRemoteReadPolicy(AllowCrossClusterSameDC) + .toTypedPipe + .map { edge => + val featureList = edge.features + val edgeFeature = EdgeFeature( + edge.weight.getOrElse(0.0).toFloat, + extractFeature(featureList, FeatureName.NumMutualFollows), + extractFeature(featureList, FeatureName.NumFavorites), + extractFeature(featureList, FeatureName.NumRetweets), + extractFeature(featureList, FeatureName.NumMentions) + ) + (edge.sourceId, (edge.destinationId, edgeFeature)) + } + .join(monthlyActiveUsers) + .map { + case (srcId, ((destId, feature), _)) => + (destId, (srcId, feature)) + } + .join(monthlyActiveUsers) + .map { + case (destId, ((srcId, feature), _)) => + (srcId, destId, feature) + } + realGraphAggregates + } + + def getTopKFollowGraph( + implicit dateRange: DateRange, + timeZone: TimeZone, + uniqueID: UniqueID + ): TypedPipe[(Long, Long)] = { + val followGraphMauStat = Stat("NumFollowEdges_MAU") + val mau: TypedPipe[Long] = getMauIds() + DAL + .readMostRecentSnapshot(RealGraphInScoresScalaDataset, dateRange.embiggen(Days(7))) + .withRemoteReadPolicy(AllowCrossClusterSameDC) + .toTypedPipe + .groupBy(_.key) + .join(mau.asKeys) + .withDescription("filtering srcId by mau") + .flatMap { + case (_, (KeyVal(srcId, CandidateSeq(candidates)), _)) => + followGraphMauStat.inc() + val topK = candidates.sortBy(-_.score).take(TopKRealGraph) + topK.map { c => (srcId, c.userId) } + } + } + + override def runOnDateRange( + enableValueGraphs: Option[Boolean], + enableKeyGraphs: Option[Boolean] + )( + implicit dateRange: DateRange, + timeZone: TimeZone, + uniqueID: UniqueID + ): Execution[Unit] = { + + val processValueGraphs = enableValueGraphs.getOrElse(Configs.EnableValueGraphs) + val processKeyGraphs = enableKeyGraphs.getOrElse(Configs.EnableKeyGraphs) + + if (!processKeyGraphs && !processValueGraphs) { + // Skip the batch job + Execution.unit + } else { + // val favoriteGraphStat = Stat("NumFavoriteEdges") + // val retweetGraphStat = Stat("NumRetweetEdges") + // val mentionGraphStat = Stat("NumMentionEdges") + + // val realGraphAggregates = getRealGraphWithMAUOnly + + val followGraph = getTopKFollowGraph + // val mutualFollowGraph = followGraph.asKeys.join(followGraph.swap.asKeys).keys + + // val favoriteGraph = + // getSubGraph(realGraphAggregates, _.favoriteScore.isDefined, favoriteGraphStat) + + // val retweetGraph = + // getSubGraph(realGraphAggregates, _.retweetScore.isDefined, retweetGraphStat) + + // val mentionGraph = + // getSubGraph(realGraphAggregates, _.mentionScore.isDefined, mentionGraphStat) + + val writeValDataSetExecutions = if (processValueGraphs) { + Seq( + (followGraph, shardingByValue, FollowOutValPath), + (followGraph.swap, shardingByValue, FollowInValPath) + // (mutualFollowGraph, shardingByValue, MutualFollowValPath), + // (favoriteGraph, shardingByValue, FavoriteOutValPath), + // (favoriteGraph.swap, shardingByValue, FavoriteInValPath), + // (retweetGraph, shardingByValue, RetweetOutValPath), + // (retweetGraph.swap, shardingByValue, RetweetInValPath), + // (mentionGraph, shardingByValue, MentionOutValPath), + // (mentionGraph.swap, shardingByValue, MentionInValPath) + ) + } else { + Seq.empty + } + + val writeKeyDataSetExecutions = if (processKeyGraphs) { + Seq( + (followGraph, shardingByKey, FollowOutKeyPath), + (followGraph.swap, shardingByKey, FollowInKeyPath) + // (favoriteGraph, shardingByKey, FavoriteOutKeyPath), + // (favoriteGraph.swap, shardingByKey, FavoriteInKeyPath), + // (retweetGraph, shardingByKey, RetweetOutKeyPath), + // (retweetGraph.swap, shardingByKey, RetweetInKeyPath), + // (mentionGraph, shardingByKey, MentionOutKeyPath), + // (mentionGraph.swap, shardingByKey, MentionInKeyPath), + // (mutualFollowGraph, shardingByKey, MutualFollowKeyPath) + ) + } else { + Seq.empty + } + + Execution + .sequence((writeValDataSetExecutions ++ writeKeyDataSetExecutions).map { + case (graph, shardingMethod, path) => + writeGraphToDB(graph, shardingMethod, path) + }).unit + } + } +} diff --git a/graph-feature-service/src/main/scalding/com/twitter/graph_feature_service/scalding/adhoc/BUILD.bazel b/graph-feature-service/src/main/scalding/com/twitter/graph_feature_service/scalding/adhoc/BUILD.bazel new file mode 100644 index 0000000000..6378b0a830 --- /dev/null +++ b/graph-feature-service/src/main/scalding/com/twitter/graph_feature_service/scalding/adhoc/BUILD.bazel @@ -0,0 +1,27 @@ +scala_library( + platform = "java8", + tags = ["bazel-only"], + dependencies = [ + "3rdparty/jvm/com/twitter/bijection:core", + "3rdparty/jvm/com/twitter/bijection:scrooge", + "frigate/frigate-common/src/main/scala/com/twitter/frigate/common/constdb_util", + "src/java/com/twitter/ml/api:api-base", + "src/scala/com/twitter/ml/api:api-base", + "src/scala/com/twitter/scalding_internal/job", + "src/scala/com/twitter/scalding_internal/job/analytics_batch", + "src/thrift/com/twitter/ml/api:data-java", + ], +) + +hadoop_binary( + name = "gfs_random_request-adhoc", + main = "com.twitter.graph_feature_service.scalding.adhoc.RandomRequestGenerationApp", + platform = "java8", + runtime_platform = "java8", + tags = [ + "bazel-compatible", + "bazel-compatible:migrated", + "bazel-only", + ], + dependencies = [":adhoc"], +) diff --git a/graph-feature-service/src/main/scalding/com/twitter/graph_feature_service/scalding/adhoc/RandomRequestGenerationApp.scala b/graph-feature-service/src/main/scalding/com/twitter/graph_feature_service/scalding/adhoc/RandomRequestGenerationApp.scala new file mode 100644 index 0000000000..7163a96ac4 --- /dev/null +++ b/graph-feature-service/src/main/scalding/com/twitter/graph_feature_service/scalding/adhoc/RandomRequestGenerationApp.scala @@ -0,0 +1,77 @@ +package com.twitter.graph_feature_service.scalding.adhoc + +import com.twitter.bijection.Injection +import com.twitter.frigate.common.constdb_util.Injections +import com.twitter.ml.api.Feature.Discrete +import com.twitter.ml.api.{DailySuffixFeatureSource, DataSetPipe, RichDataRecord} +import com.twitter.scalding._ +import com.twitter.scalding_internal.job.TwitterExecutionApp +import java.nio.ByteBuffer +import java.util.TimeZone + +object RandomRequestGenerationJob { + implicit val timeZone: TimeZone = DateOps.UTC + implicit val dateParser: DateParser = DateParser.default + + val timelineRecapDataSetPath: String = + "/atla/proc2/user/timelines/processed/suggests/recap/data_records" + + val USER_ID = new Discrete("meta.user_id") + val AUTHOR_ID = new Discrete("meta.author_id") + + val timelineRecapOutPutPath: String = "/user/cassowary/gfs/adhoc/timeline_data" + + implicit val inj: Injection[Long, ByteBuffer] = Injections.long2Varint + + def run( + dataSetPath: String, + outPutPath: String, + numOfPairsToTake: Int + )( + implicit dateRange: DateRange, + uniqueID: UniqueID + ): Execution[Unit] = { + + val NumUserAuthorPairs = Stat("NumUserAuthorPairs") + + val dataSet: DataSetPipe = DailySuffixFeatureSource(dataSetPath).read + + val userAuthorPairs: TypedPipe[(Long, Long)] = dataSet.records.map { record => + val richRecord = new RichDataRecord(record, dataSet.featureContext) + + val userId = richRecord.getFeatureValue(USER_ID) + val authorId = richRecord.getFeatureValue(AUTHOR_ID) + NumUserAuthorPairs.inc() + (userId, authorId) + } + + userAuthorPairs + .limit(numOfPairsToTake) + .writeExecution( + TypedTsv[(Long, Long)](outPutPath) + ) + } +} + +/** + * ./bazel bundle graph-feature-service/src/main/scalding/com/twitter/graph_feature_service/scalding/adhoc:all + * + * oscar hdfs --screen --user cassowary --tee gfs_log --bundle gfs_random_request-adhoc \ + --tool com.twitter.graph_feature_service.scalding.adhoc.RandomRequestGenerationApp \ + -- --date 2018-08-11 \ + --input /atla/proc2/user/timelines/processed/suggests/recap/data_records \ + --output /user/cassowary/gfs/adhoc/timeline_data + */ +object RandomRequestGenerationApp extends TwitterExecutionApp { + import RandomRequestGenerationJob._ + override def job: Execution[Unit] = Execution.withId { implicit uniqueId => + Execution.getArgs.flatMap { args: Args => + implicit val dateRange: DateRange = DateRange.parse(args.list("date"))(timeZone, dateParser) + run( + args.optional("input").getOrElse(timelineRecapDataSetPath), + args.optional("output").getOrElse(timelineRecapOutPutPath), + args.int("num_pairs", 3000) + ) + } + } +} diff --git a/graph-feature-service/src/main/thrift/com/twitter/graph_feature_service/BUILD b/graph-feature-service/src/main/thrift/com/twitter/graph_feature_service/BUILD new file mode 100644 index 0000000000..72b7d516b9 --- /dev/null +++ b/graph-feature-service/src/main/thrift/com/twitter/graph_feature_service/BUILD @@ -0,0 +1,15 @@ +create_thrift_libraries( + base_name = "graph_feature_service_thrift", + sources = ["*.thrift"], + platform = "java8", + tags = ["bazel-compatible"], + generate_languages = [ + "java", + # ruby is added due to ruby dependees in timelines + "ruby", + "scala", + "strato", + ], + provides_java_name = "graph_feature_service_thrift_java", + provides_scala_name = "graph_feature_service_thrift_scala", +) diff --git a/graph-feature-service/src/main/thrift/com/twitter/graph_feature_service/graph_feature_service.thrift b/graph-feature-service/src/main/thrift/com/twitter/graph_feature_service/graph_feature_service.thrift new file mode 100644 index 0000000000..232f8488de --- /dev/null +++ b/graph-feature-service/src/main/thrift/com/twitter/graph_feature_service/graph_feature_service.thrift @@ -0,0 +1,123 @@ +namespace java com.twitter.graph_feature_service.thriftjava +#@namespace scala com.twitter.graph_feature_service.thriftscala +#@namespace strato com.twitter.graph_feature_service.thriftscala + +// edge type to differentiate different types of graphs (we can also add a lot of other types of edges) +enum EdgeType { + FOLLOWING, + FOLLOWED_BY, + FAVORITE, + FAVORITED_BY, + RETWEET, + RETWEETED_BY, + REPLY, + REPLYED_BY, + MENTION, + MENTIONED_BY, + MUTUAL_FOLLOW, + SIMILAR_TO, // more edge types (like block, report, etc.) can be supported later. + RESERVED_12, + RESERVED_13, + RESERVED_14, + RESERVED_15, + RESERVED_16, + RESERVED_17, + RESERVED_18, + RESERVED_19, + RESERVED_20 +} + +enum PresetFeatureTypes { + EMPTY, + HTL_TWO_HOP, + WTF_TWO_HOP, + SQ_TWO_HOP, + RUX_TWO_HOP, + MR_TWO_HOP, + USER_TYPEAHEAD_TWO_HOP +} + +struct UserWithCount { + 1: required i64 userId(personalDataType = 'UserId') + 2: required i32 count +}(hasPersonalData = 'true') + +struct UserWithScore { + 1: required i64 userId(personalDataType = 'UserId') + 2: required double score +}(hasPersonalData = 'true') + +// Feature Type +// For example, to compute how many of source user's following's have favorited candidate user, +// we need to compute the intersection between source user's FOLLOWING edges, and candidate user's +// FAVORITED_BY edge. In this case, we should user FeatureType(FOLLOWING, FAVORITED_BY) +struct FeatureType { + 1: required EdgeType leftEdgeType // edge type from source user + 2: required EdgeType rightEdgeType // edge type from candidate user +}(persisted="true") + +struct IntersectionValue { + 1: required FeatureType featureType + 2: optional i32 count + 3: optional list intersectionIds(personalDataType = 'UserId') + 4: optional i32 leftNodeDegree + 5: optional i32 rightNodeDegree +}(persisted="true", hasPersonalData = 'true') + +struct GfsIntersectionResult { + 1: required i64 candidateUserId(personalDataType = 'UserId') + 2: required list intersectionValues +}(hasPersonalData = 'true') + +struct GfsIntersectionRequest { + 1: required i64 userId(personalDataType = 'UserId') + 2: required list candidateUserIds(personalDataType = 'UserId') + 3: required list featureTypes + 4: optional i32 intersectionIdLimit +} + +struct GfsPresetIntersectionRequest { + 1: required i64 userId(personalDataType = 'UserId') + 2: required list candidateUserIds(personalDataType = 'UserId') + 3: required PresetFeatureTypes presetFeatureTypes + 4: optional i32 intersectionIdLimit +}(hasPersonalData = 'true') + +struct GfsIntersectionResponse { + 1: required list results +} + +service Server { + GfsIntersectionResponse getIntersection(1: GfsIntersectionRequest request) + GfsIntersectionResponse getPresetIntersection(1: GfsPresetIntersectionRequest request) +} + +################################################################################################### +## For internal usage only +################################################################################################### +struct WorkerIntersectionRequest { + 1: required i64 userId(personalDataType = 'UserId') + 2: required list candidateUserIds(personalDataType = 'UserId') + 3: required list featureTypes + 4: required PresetFeatureTypes presetFeatureTypes + 5: required i32 intersectionIdLimit +}(hasPersonalData = 'true') + +struct WorkerIntersectionResponse { + 1: required list> results +} + +struct WorkerIntersectionValue { + 1: i32 count + 2: i32 leftNodeDegree + 3: i32 rightNodeDegree + 4: list intersectionIds(personalDataType = 'UserId') +}(hasPersonalData = 'true') + +struct CachedIntersectionResult { + 1: required list values +} + +service Worker { + WorkerIntersectionResponse getIntersection(1: WorkerIntersectionRequest request) +} diff --git a/home-mixer/BUILD.bazel b/home-mixer/BUILD.bazel new file mode 100644 index 0000000000..ab7b358e71 --- /dev/null +++ b/home-mixer/BUILD.bazel @@ -0,0 +1,30 @@ +jvm_binary( + name = "bin", + basename = "home-mixer", + main = "com.twitter.home_mixer.HomeMixerServerMain", + runtime_platform = "java11", + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/ch/qos/logback:logback-classic", + "finagle/finagle-zipkin-scribe/src/main/scala", + "finatra/inject/inject-logback/src/main/scala", + "home-mixer/server/src/main/scala/com/twitter/home_mixer", + "loglens/loglens-logback/src/main/scala/com/twitter/loglens/logback", + "twitter-server-internal/src/main/scala", + "twitter-server/logback-classic/src/main/scala", + ], +) + +# Aurora Workflows build phase convention requires a jvm_app named with home-mixer-app +jvm_app( + name = "home-mixer-app", + archive = "zip", + binary = ":bin", + bundles = [ + bundle( + fileset = ["config/**/*"], + owning_target = "home-mixer/config:files", + ), + ], + tags = ["bazel-compatible"], +) diff --git a/home-mixer/README.md b/home-mixer/README.md new file mode 100644 index 0000000000..a1be737be7 --- /dev/null +++ b/home-mixer/README.md @@ -0,0 +1,102 @@ +Home Mixer +========== + +Home Mixer is the main service used to construct and serve Twitter's Home Timelines. It currently +powers: +- For you - best Tweets from people you follow + recommended out-of-network content +- Following - reverse chronological Tweets from people you follow +- Lists - reverse chronological Tweets from List members + +Home Mixer is built on Product Mixer, our custom Scala framework that facilitates building +feeds of content. + +## Overview + +The For You recommendation algorithm in Home Mixer involves the following stages: + +- Candidate Generation - fetch Tweets from various Candidate Sources. For example: + - Earlybird Search Index + - User Tweet Entity Graph + - Cr Mixer + - Follow Recommendations Service +- Feature Hydration + - Fetch the ~6000 features needed for ranking +- Scoring and Ranking using ML model +- Filters and Heuristics. For example: + - Author Diversity + - Content Balance (In network vs Out of Network) + - Feedback fatigue + - Deduplication / previously seen Tweets removal + - Visibility Filtering (blocked, muted authors/tweets, NSFW settings) +- Mixing - integrate Tweets with non-Tweet content + - Ads + - Who-to-follow modules + - Prompts +- Product Features and Serving + - Conversation Modules for replies + - Social Context + - Timeline Navigation + - Edited Tweets + - Feedback options + - Pagination and cursoring + - Observability and logging + - Client instructions and content marshalling + +## Pipeline Structure + +### General + +Product Mixer services like Home Mixer are structured around Pipelines that split the execution +into transparent and structured steps. + +Requests first go to Product Pipelines, which are used to select which Mixer Pipeline or +Recommendation Pipeline to run for a given request. Each Mixer or Recommendation +Pipeline may run multiple Candidate Pipelines to fetch candidates to include in the response. + +Mixer Pipelines combine the results of multiple heterogeneous Candidate Pipelines together +(e.g. ads, tweets, users) while Recommendation Pipelines are used to score (via Scoring Pipelines) +and rank the results of homogenous Candidate Pipelines so that the top ranked ones can be returned. +These pipelines also marshall candidates into a domain object and then into a transport object +to return to the caller. + +Candidate Pipelines fetch candidates from underlying Candidate Sources and perform some basic +operations on the Candidates, such as filtering out unwanted candidates, applying decorations, +and hydrating features. + +The sections below describe the high level pipeline structure (non-exhaustive) for the main Home +Timeline tabs powered by Home Mixer. + +### For You + +- ForYouProductPipelineConfig + - ForYouScoredTweetsMixerPipelineConfig (main orchestration layer - mixes Tweets with ads and users) + - ForYouScoredTweetsCandidatePipelineConfig (fetch Tweets) + - ScoredTweetsRecommendationPipelineConfig (main Tweet recommendation layer) + - Fetch Tweet Candidates + - ScoredTweetsInNetworkCandidatePipelineConfig + - ScoredTweetsCrMixerCandidatePipelineConfig + - ScoredTweetsUtegCandidatePipelineConfig + - ScoredTweetsFrsCandidatePipelineConfig + - Feature Hydration and Scoring + - ScoredTweetsScoringPipelineConfig + - ForYouConversationServiceCandidatePipelineConfig (backup reverse chron pipeline in case Scored Tweets fails) + - ForYouAdsCandidatePipelineConfig (fetch ads) + - ForYouWhoToFollowCandidatePipelineConfig (fetch users to recommend) + +### Following + +- FollowingProductPipelineConfig + - FollowingMixerPipelineConfig + - FollowingEarlybirdCandidatePipelineConfig (fetch tweets from Search Index) + - ConversationServiceCandidatePipelineConfig (fetch ancestors for conversation modules) + - FollowingAdsCandidatePipelineConfig (fetch ads) + - FollowingWhoToFollowCandidatePipelineConfig (fetch users to recommend) + +### Lists + +- ListTweetsProductPipelineConfig + - ListTweetsMixerPipelineConfig + - ListTweetsTimelineServiceCandidatePipelineConfig (fetch tweets from timeline service) + - ConversationServiceCandidatePipelineConfig (fetch ancestors for conversation modules) + - ListTweetsAdsCandidatePipelineConfig (fetch ads) + diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/BUILD.bazel b/home-mixer/server/src/main/scala/com/twitter/home_mixer/BUILD.bazel new file mode 100644 index 0000000000..a9474ec099 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/BUILD.bazel @@ -0,0 +1,46 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + strict_deps = True, + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/com/google/inject:guice", + "3rdparty/jvm/javax/inject:javax.inject", + "3rdparty/jvm/net/codingwell:scala-guice", + "3rdparty/jvm/org/slf4j:slf4j-api", + "finagle/finagle-core/src/main", + "finagle/finagle-http/src/main/scala", + "finagle/finagle-thriftmux/src/main/scala", + "finatra-internal/mtls-http/src/main/scala", + "finatra-internal/mtls-thriftmux/src/main/scala", + "finatra/http-core/src/main/java/com/twitter/finatra/http", + "finatra/inject/inject-app/src/main/java/com/twitter/inject/annotations", + "finatra/inject/inject-app/src/main/scala", + "finatra/inject/inject-core/src/main/scala", + "finatra/inject/inject-server/src/main/scala", + "finatra/inject/inject-utils/src/main/scala", + "home-mixer/server/src/main/resources", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/controller", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/module", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/param", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/product", + "home-mixer/thrift/src/main/thrift:thrift-scala", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/module", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/controllers", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/module", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/module/stringcenter", + "product-mixer/core/src/main/thrift/com/twitter/product_mixer/core:thrift-scala", + "src/thrift/com/twitter/timelines/render:thrift-scala", + "stringcenter/client", + "stringcenter/client/src/main/java", + "stringcenter/client/src/main/scala/com/twitter/stringcenter/client", + "thrift-web-forms/src/main/scala/com/twitter/thriftwebforms/view", + "timelines/src/main/scala/com/twitter/timelines/config", + "timelines/src/main/scala/com/twitter/timelines/features/app", + "twitter-server-internal", + "twitter-server/server/src/main/scala", + "util/util-app/src/main/scala", + "util/util-core:scala", + "util/util-slf4j-api/src/main/scala", + ], +) diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/HomeMixerHttpServerWarmupHandler.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/HomeMixerHttpServerWarmupHandler.scala new file mode 100644 index 0000000000..16a9d9c58d --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/HomeMixerHttpServerWarmupHandler.scala @@ -0,0 +1,18 @@ +package com.twitter.home_mixer + +import com.twitter.finatra.http.routing.HttpWarmup +import com.twitter.finatra.httpclient.RequestBuilder._ +import com.twitter.inject.Logging +import com.twitter.inject.utils.Handler +import com.twitter.util.Try +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class HomeMixerHttpServerWarmupHandler @Inject() (warmup: HttpWarmup) extends Handler with Logging { + + override def handle(): Unit = { + Try(warmup.send(get("/admin/product-mixer/product-pipelines"), admin = true)()) + .onFailure(e => error(e.getMessage, e)) + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/HomeMixerServer.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/HomeMixerServer.scala new file mode 100644 index 0000000000..ff7c757277 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/HomeMixerServer.scala @@ -0,0 +1,118 @@ +package com.twitter.home_mixer + +import com.google.inject.Module +import com.twitter.finagle.Filter +import com.twitter.finatra.annotations.DarkTrafficFilterType +import com.twitter.finatra.http.HttpServer +import com.twitter.finatra.http.routing.HttpRouter +import com.twitter.finatra.mtls.http.{Mtls => HttpMtls} +import com.twitter.finatra.mtls.thriftmux.Mtls +import com.twitter.finatra.mtls.thriftmux.modules.MtlsThriftWebFormsModule +import com.twitter.finatra.thrift.ThriftServer +import com.twitter.finatra.thrift.filters._ +import com.twitter.finatra.thrift.routing.ThriftRouter +import com.twitter.home_mixer.controller.HomeThriftController +import com.twitter.home_mixer.module._ +import com.twitter.home_mixer.param.GlobalParamConfigModule +import com.twitter.home_mixer.product.HomeMixerProductModule +import com.twitter.home_mixer.{thriftscala => st} +import com.twitter.product_mixer.component_library.module.AccountRecommendationsMixerModule +import com.twitter.product_mixer.component_library.module.CrMixerClientModule +import com.twitter.product_mixer.component_library.module.DarkTrafficFilterModule +import com.twitter.product_mixer.component_library.module.EarlybirdModule +import com.twitter.product_mixer.component_library.module.ExploreRankerClientModule +import com.twitter.product_mixer.component_library.module.GizmoduckClientModule +import com.twitter.product_mixer.component_library.module.OnboardingTaskServiceModule +import com.twitter.product_mixer.component_library.module.SocialGraphServiceModule +import com.twitter.product_mixer.component_library.module.TimelineMixerClientModule +import com.twitter.product_mixer.component_library.module.TimelineRankerClientModule +import com.twitter.product_mixer.component_library.module.TimelineScorerClientModule +import com.twitter.product_mixer.component_library.module.TimelineServiceClientModule +import com.twitter.product_mixer.component_library.module.TweetImpressionStoreModule +import com.twitter.product_mixer.component_library.module.UserSessionStoreModule +import com.twitter.product_mixer.core.controllers.ProductMixerController +import com.twitter.product_mixer.core.module.LoggingThrowableExceptionMapper +import com.twitter.product_mixer.core.module.ProductMixerModule +import com.twitter.product_mixer.core.module.StratoClientModule +import com.twitter.product_mixer.core.module.stringcenter.ProductScopeStringCenterModule + +object HomeMixerServerMain extends HomeMixerServer + +class HomeMixerServer extends ThriftServer with Mtls with HttpServer with HttpMtls { + override val name = "home-mixer-server" + + override val modules: Seq[Module] = Seq( + AccountRecommendationsMixerModule, + AdvertiserBrandSafetySettingsStoreModule, + ClientSentImpressionsPublisherModule, + ConversationServiceModule, + CrMixerClientModule, + EarlybirdModule, + ExploreRankerClientModule, + GizmoduckClientModule, + GlobalParamConfigModule, + HomeAdsCandidateSourceModule, + HomeMixerFlagsModule, + HomeMixerProductModule, + HomeMixerResourcesModule, + HomeNaviModelClientModule, + ImpressionBloomFilterModule, + InjectionHistoryClientModule, + FeedbackHistoryClientModule, + ManhattanClientsModule, + ManhattanFeatureRepositoryModule, + ManhattanTweetImpressionStoreModule, + MemcachedFeatureRepositoryModule, + OnboardingTaskServiceModule, + OptimizedStratoClientModule, + PeopleDiscoveryServiceModule, + ProductMixerModule, + RealGraphInNetworkScoresModule, + RealtimeAggregateFeatureRepositoryModule, + ScoredTweetsMemcacheModule, + ScribeEventPublisherModule, + SimClustersRecentEngagementsClientModule, + SocialGraphServiceModule, + StaleTweetsCacheModule, + StratoClientModule, + ThriftFeatureRepositoryModule, + TimelineMixerClientModule, + TimelineRankerClientModule, + TimelineScorerClientModule, + TimelineServiceClientModule, + TimelinesPersistenceStoreClientModule, + TweetImpressionStoreModule, + TweetyPieClientModule, + TweetypieStaticEntitiesCacheClientModule, + UserMetadataStoreModule, + UserSessionStoreModule, + new DarkTrafficFilterModule[st.HomeMixer.ReqRepServicePerEndpoint](), + new MtlsThriftWebFormsModule[st.HomeMixer.MethodPerEndpoint](this), + new ProductScopeStringCenterModule() + ) + + def configureThrift(router: ThriftRouter): Unit = { + router + .filter[LoggingMDCFilter] + .filter[TraceIdMDCFilter] + .filter[ThriftMDCFilter] + .filter[StatsFilter] + .filter[AccessLoggingFilter] + .filter[ExceptionMappingFilter] + .filter[Filter.TypeAgnostic, DarkTrafficFilterType] + .exceptionMapper[LoggingThrowableExceptionMapper] + .exceptionMapper[PipelineFailureExceptionMapper] + .add[HomeThriftController] + } + + override def configureHttp(router: HttpRouter): Unit = + router.add( + ProductMixerController[st.HomeMixer.MethodPerEndpoint]( + this.injector, + st.HomeMixer.ExecutePipeline)) + + override protected def warmup(): Unit = { + handle[HomeMixerThriftServerWarmupHandler]() + handle[HomeMixerHttpServerWarmupHandler]() + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/HomeMixerThriftServerWarmupHandler.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/HomeMixerThriftServerWarmupHandler.scala new file mode 100644 index 0000000000..df70010721 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/HomeMixerThriftServerWarmupHandler.scala @@ -0,0 +1,73 @@ +package com.twitter.home_mixer + +import com.twitter.finagle.thrift.ClientId +import com.twitter.finatra.thrift.routing.ThriftWarmup +import com.twitter.home_mixer.{thriftscala => st} +import com.twitter.inject.Logging +import com.twitter.inject.utils.Handler +import com.twitter.product_mixer.core.{thriftscala => pt} +import com.twitter.scrooge.Request +import com.twitter.scrooge.Response +import com.twitter.util.Return +import com.twitter.util.Throw +import com.twitter.util.Try +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class HomeMixerThriftServerWarmupHandler @Inject() (warmup: ThriftWarmup) + extends Handler + with Logging { + + private val clientId = ClientId("thrift-warmup-client") + + def handle(): Unit = { + val testIds = Seq(1, 2, 3) + try { + clientId.asCurrent { + testIds.foreach { id => + val warmupReq = warmupQuery(id) + info(s"Sending warm-up request to service with query: $warmupReq") + warmup.sendRequest( + method = st.HomeMixer.GetUrtResponse, + req = Request(st.HomeMixer.GetUrtResponse.Args(warmupReq)))(assertWarmupResponse) + } + } + } catch { + case e: Throwable => error(e.getMessage, e) + } + info("Warm-up done.") + } + + private def warmupQuery(userId: Long): st.HomeMixerRequest = { + val clientContext = pt.ClientContext( + userId = Some(userId), + guestId = None, + appId = Some(12345L), + ipAddress = Some("0.0.0.0"), + userAgent = Some("FAKE_USER_AGENT_FOR_WARMUPS"), + countryCode = Some("US"), + languageCode = Some("en"), + isTwoffice = None, + userRoles = None, + deviceId = Some("FAKE_DEVICE_ID_FOR_WARMUPS") + ) + st.HomeMixerRequest( + clientContext = clientContext, + product = st.Product.Following, + productContext = Some(st.ProductContext.Following(st.Following())), + maxResults = Some(3) + ) + } + + private def assertWarmupResponse( + result: Try[Response[st.HomeMixer.GetUrtResponse.SuccessType]] + ): Unit = { + result match { + case Return(_) => // ok + case Throw(exception) => + warn("Error performing warm-up request.") + error(exception.getMessage, exception) + } + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/candidate_pipeline/BUILD.bazel b/home-mixer/server/src/main/scala/com/twitter/home_mixer/candidate_pipeline/BUILD.bazel new file mode 100644 index 0000000000..526efd7183 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/candidate_pipeline/BUILD.bazel @@ -0,0 +1,36 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + strict_deps = True, + tags = ["bazel-compatible"], + dependencies = [ + "home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/candidate_source", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/builder", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/urt/builder", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/filter", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/gate", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/query_transformer", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/marshaller/timelines", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/model", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/model/request", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/param", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/product/following/model", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/product/following/param", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/service", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/candidate_source/tweetconvosvc", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/filter", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/gate", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/candidate", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/transformer", + "timelinemixer/common/src/main/scala/com/twitter/timelinemixer/clients/manhattan", + "timelinemixer/server/src/main/scala/com/twitter/timelinemixer/injection/store/persistence", + "timelines/src/main/scala/com/twitter/timelines/injection/scribe", + "timelineservice/common/src/main/scala/com/twitter/timelineservice/model", + ], + exports = [ + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/candidate_source/tweetconvosvc", + ], +) diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/candidate_pipeline/ConversationServiceCandidatePipelineConfig.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/candidate_pipeline/ConversationServiceCandidatePipelineConfig.scala new file mode 100644 index 0000000000..8a255b7bbc --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/candidate_pipeline/ConversationServiceCandidatePipelineConfig.scala @@ -0,0 +1,107 @@ +package com.twitter.home_mixer.candidate_pipeline + +import com.twitter.home_mixer.functional_component.feature_hydrator.NamesFeatureHydrator +import com.twitter.home_mixer.functional_component.feature_hydrator.SocialGraphServiceFeatureHydrator +import com.twitter.home_mixer.functional_component.feature_hydrator.TweetypieFeatureHydrator +import com.twitter.home_mixer.functional_component.filter.InvalidConversationModuleFilter +import com.twitter.home_mixer.functional_component.filter.PredicateFeatureFilter +import com.twitter.home_mixer.functional_component.filter.RetweetDeduplicationFilter +import com.twitter.home_mixer.model.HomeFeatures.IsHydratedFeature +import com.twitter.home_mixer.model.HomeFeatures.QuotedTweetDroppedFeature +import com.twitter.home_mixer.service.HomeMixerAlertConfig +import com.twitter.product_mixer.component_library.candidate_source.tweetconvosvc.ConversationServiceCandidateSource +import com.twitter.product_mixer.component_library.candidate_source.tweetconvosvc.ConversationServiceCandidateSourceRequest +import com.twitter.product_mixer.component_library.candidate_source.tweetconvosvc.TweetWithConversationMetadata +import com.twitter.product_mixer.component_library.filter.FeatureFilter +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.functional_component.candidate_source.BaseCandidateSource +import com.twitter.product_mixer.core.functional_component.decorator.CandidateDecorator +import com.twitter.product_mixer.core.functional_component.feature_hydrator.BaseCandidateFeatureHydrator +import com.twitter.product_mixer.core.functional_component.filter.Filter +import com.twitter.product_mixer.core.functional_component.gate.BaseGate +import com.twitter.product_mixer.core.functional_component.transformer.CandidateFeatureTransformer +import com.twitter.product_mixer.core.functional_component.transformer.CandidatePipelineResultsTransformer +import com.twitter.product_mixer.core.functional_component.transformer.DependentCandidatePipelineQueryTransformer +import com.twitter.product_mixer.core.model.common.identifier.CandidatePipelineIdentifier +import com.twitter.product_mixer.core.model.common.identifier.FilterIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.product_mixer.core.pipeline.candidate.DependentCandidatePipelineConfig + +/** + * Candidate Pipeline Config that fetches tweets from the Conversation Service Candidate Source + */ +class ConversationServiceCandidatePipelineConfig[Query <: PipelineQuery]( + conversationServiceCandidateSource: ConversationServiceCandidateSource, + tweetypieFeatureHydrator: TweetypieFeatureHydrator, + socialGraphServiceFeatureHydrator: SocialGraphServiceFeatureHydrator, + namesFeatureHydrator: NamesFeatureHydrator, + override val gates: Seq[BaseGate[Query]], + override val decorator: Option[CandidateDecorator[Query, TweetCandidate]]) + extends DependentCandidatePipelineConfig[ + Query, + ConversationServiceCandidateSourceRequest, + TweetWithConversationMetadata, + TweetCandidate + ] { + + override val identifier: CandidatePipelineIdentifier = + CandidatePipelineIdentifier("ConversationService") + + private val TweetypieHydratedFilterId = "TweetypieHydrated" + private val QuotedTweetDroppedFilterId = "QuotedTweetDropped" + + override val candidateSource: BaseCandidateSource[ + ConversationServiceCandidateSourceRequest, + TweetWithConversationMetadata + ] = conversationServiceCandidateSource + + override val queryTransformer: DependentCandidatePipelineQueryTransformer[ + Query, + ConversationServiceCandidateSourceRequest + ] = { (_, candidates) => + val tweetsWithConversationMetadata = candidates.map { candidate => + TweetWithConversationMetadata( + tweetId = candidate.candidateIdLong, + userId = None, + sourceTweetId = None, + sourceUserId = None, + inReplyToTweetId = None, + conversationId = None, + ancestors = Seq.empty + ) + } + ConversationServiceCandidateSourceRequest(tweetsWithConversationMetadata) + } + + override val featuresFromCandidateSourceTransformers: Seq[ + CandidateFeatureTransformer[TweetWithConversationMetadata] + ] = Seq(ConversationServiceResponseFeatureTransformer) + + override val resultTransformer: CandidatePipelineResultsTransformer[ + TweetWithConversationMetadata, + TweetCandidate + ] = { sourceResult => TweetCandidate(id = sourceResult.tweetId) } + + override val preFilterFeatureHydrationPhase1: Seq[ + BaseCandidateFeatureHydrator[Query, TweetCandidate, _] + ] = Seq(tweetypieFeatureHydrator, socialGraphServiceFeatureHydrator) + + override def filters: Seq[Filter[Query, TweetCandidate]] = Seq( + RetweetDeduplicationFilter, + FeatureFilter.fromFeature(FilterIdentifier(TweetypieHydratedFilterId), IsHydratedFeature), + PredicateFeatureFilter.fromPredicate( + FilterIdentifier(QuotedTweetDroppedFilterId), + shouldKeepCandidate = { features => !features.getOrElse(QuotedTweetDroppedFeature, false) } + ), + InvalidConversationModuleFilter + ) + + override val postFilterFeatureHydration: Seq[ + BaseCandidateFeatureHydrator[Query, TweetCandidate, _] + ] = Seq(namesFeatureHydrator) + + override val alerts = Seq( + HomeMixerAlertConfig.BusinessHours.defaultSuccessRateAlert(), + HomeMixerAlertConfig.BusinessHours.defaultEmptyResponseRateAlert() + ) +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/candidate_pipeline/ConversationServiceCandidatePipelineConfigBuilder.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/candidate_pipeline/ConversationServiceCandidatePipelineConfigBuilder.scala new file mode 100644 index 0000000000..219069816e --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/candidate_pipeline/ConversationServiceCandidatePipelineConfigBuilder.scala @@ -0,0 +1,34 @@ +package com.twitter.home_mixer.candidate_pipeline + +import com.twitter.product_mixer.component_library.candidate_source.tweetconvosvc.ConversationServiceCandidateSource +import com.twitter.home_mixer.functional_component.feature_hydrator.NamesFeatureHydrator +import com.twitter.home_mixer.functional_component.feature_hydrator.SocialGraphServiceFeatureHydrator +import com.twitter.home_mixer.functional_component.feature_hydrator.TweetypieFeatureHydrator +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.functional_component.decorator.CandidateDecorator +import com.twitter.product_mixer.core.functional_component.gate.BaseGate +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class ConversationServiceCandidatePipelineConfigBuilder[Query <: PipelineQuery] @Inject() ( + conversationServiceCandidateSource: ConversationServiceCandidateSource, + tweetypieFeatureHydrator: TweetypieFeatureHydrator, + socialGraphServiceFeatureHydrator: SocialGraphServiceFeatureHydrator, + namesFeatureHydrator: NamesFeatureHydrator) { + + def build( + gates: Seq[BaseGate[Query]] = Seq.empty, + decorator: Option[CandidateDecorator[Query, TweetCandidate]] = None + ): ConversationServiceCandidatePipelineConfig[Query] = { + new ConversationServiceCandidatePipelineConfig( + conversationServiceCandidateSource, + tweetypieFeatureHydrator, + socialGraphServiceFeatureHydrator, + namesFeatureHydrator, + gates, + decorator + ) + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/candidate_pipeline/ConversationServiceResponseFeatureTransformer.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/candidate_pipeline/ConversationServiceResponseFeatureTransformer.scala new file mode 100644 index 0000000000..154c080ad9 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/candidate_pipeline/ConversationServiceResponseFeatureTransformer.scala @@ -0,0 +1,39 @@ +package com.twitter.home_mixer.candidate_pipeline + +import com.twitter.home_mixer.model.HomeFeatures._ +import com.twitter.product_mixer.component_library.candidate_source.tweetconvosvc.TweetWithConversationMetadata +import com.twitter.product_mixer.core.feature.Feature +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.feature.featuremap.FeatureMapBuilder +import com.twitter.product_mixer.core.functional_component.transformer.CandidateFeatureTransformer +import com.twitter.product_mixer.core.model.common.identifier.TransformerIdentifier +import com.twitter.timelineservice.suggests.thriftscala.SuggestType + +object ConversationServiceResponseFeatureTransformer + extends CandidateFeatureTransformer[TweetWithConversationMetadata] { + + override val identifier: TransformerIdentifier = + TransformerIdentifier("ConversationServiceResponse") + + override val features: Set[Feature[_, _]] = Set( + AuthorIdFeature, + InReplyToTweetIdFeature, + IsRetweetFeature, + SourceTweetIdFeature, + SourceUserIdFeature, + ConversationModuleFocalTweetIdFeature, + AncestorsFeature, + SuggestTypeFeature + ) + + override def transform(candidate: TweetWithConversationMetadata): FeatureMap = FeatureMapBuilder() + .add(AuthorIdFeature, candidate.userId) + .add(InReplyToTweetIdFeature, candidate.inReplyToTweetId) + .add(IsRetweetFeature, candidate.sourceTweetId.isDefined) + .add(SourceTweetIdFeature, candidate.sourceTweetId) + .add(SourceUserIdFeature, candidate.sourceUserId) + .add(ConversationModuleFocalTweetIdFeature, candidate.conversationId) + .add(AncestorsFeature, candidate.ancestors) + .add(SuggestTypeFeature, Some(SuggestType.RankedOrganicTweet)) + .build() +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/candidate_pipeline/EditedTweetsCandidatePipelineConfig.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/candidate_pipeline/EditedTweetsCandidatePipelineConfig.scala new file mode 100644 index 0000000000..8f824a8ff6 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/candidate_pipeline/EditedTweetsCandidatePipelineConfig.scala @@ -0,0 +1,84 @@ +package com.twitter.home_mixer.candidate_pipeline + +import com.twitter.home_mixer.functional_component.candidate_source.StaleTweetsCacheCandidateSource +import com.twitter.home_mixer.functional_component.decorator.HomeFeedbackActionInfoBuilder +import com.twitter.home_mixer.functional_component.feature_hydrator.NamesFeatureHydrator +import com.twitter.home_mixer.functional_component.query_transformer.EditedTweetsCandidatePipelineQueryTransformer +import com.twitter.home_mixer.service.HomeMixerAlertConfig +import com.twitter.product_mixer.component_library.decorator.urt.UrtItemCandidateDecorator +import com.twitter.product_mixer.component_library.decorator.urt.builder.contextual_ref.ContextualTweetRefBuilder +import com.twitter.product_mixer.component_library.decorator.urt.builder.item.tweet.TweetCandidateUrtItemBuilder +import com.twitter.product_mixer.component_library.decorator.urt.builder.metadata.EmptyClientEventInfoBuilder +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.functional_component.candidate_source.BaseCandidateSource +import com.twitter.product_mixer.core.functional_component.decorator.CandidateDecorator +import com.twitter.product_mixer.core.functional_component.feature_hydrator.BaseCandidateFeatureHydrator +import com.twitter.product_mixer.core.functional_component.transformer.CandidatePipelineQueryTransformer +import com.twitter.product_mixer.core.functional_component.transformer.CandidatePipelineResultsTransformer +import com.twitter.product_mixer.core.model.common.identifier.CandidatePipelineIdentifier +import com.twitter.product_mixer.core.model.marshalling.response.rtf.safety_level.TimelineFocalTweetSafetyLevel +import com.twitter.product_mixer.core.model.marshalling.response.urt.contextual_ref.TweetHydrationContext +import com.twitter.product_mixer.core.model.marshalling.response.urt.item.tweet.TweetItem +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.product_mixer.core.pipeline.candidate.DependentCandidatePipelineConfig +import javax.inject.Inject +import javax.inject.Singleton + +/** + * Candidate Pipeline Config that fetches edited tweets from the Stale Tweets Cache + */ +@Singleton +case class EditedTweetsCandidatePipelineConfig @Inject() ( + staleTweetsCacheCandidateSource: StaleTweetsCacheCandidateSource, + namesFeatureHydrator: NamesFeatureHydrator, + homeFeedbackActionInfoBuilder: HomeFeedbackActionInfoBuilder) + extends DependentCandidatePipelineConfig[ + PipelineQuery, + Seq[Long], + Long, + TweetCandidate + ] { + + override val identifier: CandidatePipelineIdentifier = CandidatePipelineIdentifier("EditedTweets") + + override val candidateSource: BaseCandidateSource[Seq[Long], Long] = + staleTweetsCacheCandidateSource + + override val queryTransformer: CandidatePipelineQueryTransformer[ + PipelineQuery, + Seq[Long] + ] = EditedTweetsCandidatePipelineQueryTransformer + + override val resultTransformer: CandidatePipelineResultsTransformer[ + Long, + TweetCandidate + ] = { candidate => TweetCandidate(id = candidate) } + + override val postFilterFeatureHydration: Seq[ + BaseCandidateFeatureHydrator[PipelineQuery, TweetCandidate, _] + ] = Seq(namesFeatureHydrator) + + override val decorator: Option[CandidateDecorator[PipelineQuery, TweetCandidate]] = { + val tweetItemBuilder = TweetCandidateUrtItemBuilder[PipelineQuery, TweetCandidate]( + clientEventInfoBuilder = EmptyClientEventInfoBuilder, + entryIdToReplaceBuilder = Some((_, candidate, _) => + Some(s"${TweetItem.TweetEntryNamespace}-${candidate.id.toString}")), + contextualTweetRefBuilder = Some( + ContextualTweetRefBuilder( + TweetHydrationContext( + // Apply safety level that includes canonical VF treatments that apply regardless of context. + safetyLevelOverride = Some(TimelineFocalTweetSafetyLevel), + outerTweetContext = None + ) + ) + ), + feedbackActionInfoBuilder = Some(homeFeedbackActionInfoBuilder) + ) + + Some(UrtItemCandidateDecorator(tweetItemBuilder)) + } + + override val alerts = Seq( + HomeMixerAlertConfig.BusinessHours.defaultSuccessRateAlert(99.5, 50, 60, 60) + ) +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/candidate_pipeline/NewTweetsPillCandidatePipelineConfig.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/candidate_pipeline/NewTweetsPillCandidatePipelineConfig.scala new file mode 100644 index 0000000000..e1a92b9ca7 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/candidate_pipeline/NewTweetsPillCandidatePipelineConfig.scala @@ -0,0 +1,123 @@ +package com.twitter.home_mixer.candidate_pipeline + +import com.twitter.conversions.DurationOps._ +import com.twitter.home_mixer.functional_component.gate.RequestContextNotGate +import com.twitter.home_mixer.model.HomeFeatures.GetNewerFeature +import com.twitter.home_mixer.model.request.DeviceContext +import com.twitter.home_mixer.model.request.HasDeviceContext +import com.twitter.home_mixer.service.HomeMixerAlertConfig +import com.twitter.product_mixer.component_library.decorator.urt.UrtItemCandidateDecorator +import com.twitter.product_mixer.component_library.decorator.urt.builder.item.alert.DurationParamBuilder +import com.twitter.product_mixer.component_library.decorator.urt.builder.item.alert.ShowAlertCandidateUrtItemBuilder +import com.twitter.product_mixer.component_library.decorator.urt.builder.item.alert.StaticShowAlertColorConfigurationBuilder +import com.twitter.product_mixer.component_library.decorator.urt.builder.item.alert.StaticShowAlertDisplayLocationBuilder +import com.twitter.product_mixer.component_library.decorator.urt.builder.item.alert.StaticShowAlertIconDisplayInfoBuilder +import com.twitter.product_mixer.component_library.gate.FeatureGate +import com.twitter.product_mixer.component_library.model.candidate.ShowAlertCandidate +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.functional_component.candidate_source.CandidateSource +import com.twitter.product_mixer.core.functional_component.candidate_source.StaticCandidateSource +import com.twitter.product_mixer.core.functional_component.configapi.StaticParam +import com.twitter.product_mixer.core.functional_component.decorator.CandidateDecorator +import com.twitter.product_mixer.core.functional_component.decorator.urt.builder.item.alert.BaseDurationBuilder +import com.twitter.product_mixer.core.functional_component.gate.Gate +import com.twitter.product_mixer.core.functional_component.transformer.CandidatePipelineQueryTransformer +import com.twitter.product_mixer.core.functional_component.transformer.CandidatePipelineResultsTransformer +import com.twitter.product_mixer.core.model.common.identifier.CandidatePipelineIdentifier +import com.twitter.product_mixer.core.model.common.identifier.CandidateSourceIdentifier +import com.twitter.product_mixer.core.model.marshalling.response.urt.alert.NewTweets +import com.twitter.product_mixer.core.model.marshalling.response.urt.alert.ShowAlertColorConfiguration +import com.twitter.product_mixer.core.model.marshalling.response.urt.alert.ShowAlertIconDisplayInfo +import com.twitter.product_mixer.core.model.marshalling.response.urt.alert.Top +import com.twitter.product_mixer.core.model.marshalling.response.urt.alert.UpArrow +import com.twitter.product_mixer.core.model.marshalling.response.urt.color.TwitterBlueRosettaColor +import com.twitter.product_mixer.core.model.marshalling.response.urt.color.WhiteRosettaColor +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.product_mixer.core.pipeline.candidate.DependentCandidatePipelineConfig +import com.twitter.util.Duration +import javax.inject.Inject +import javax.inject.Singleton + +/** + * Candidate Pipeline Config that creates the New Tweets Pill + */ +@Singleton +class NewTweetsPillCandidatePipelineConfig[Query <: PipelineQuery with HasDeviceContext] @Inject() ( +) extends DependentCandidatePipelineConfig[ + Query, + Unit, + ShowAlertCandidate, + ShowAlertCandidate + ] { + import NewTweetsPillCandidatePipelineConfig._ + + override val identifier: CandidatePipelineIdentifier = + CandidatePipelineIdentifier("NewTweetsPill") + + override val gates: Seq[Gate[Query]] = Seq( + RequestContextNotGate(Seq(DeviceContext.RequestContext.PullToRefresh)), + FeatureGate.fromFeature(GetNewerFeature) + ) + + override val candidateSource: CandidateSource[Unit, ShowAlertCandidate] = + StaticCandidateSource( + CandidateSourceIdentifier(identifier.name), + Seq(ShowAlertCandidate(id = identifier.name, userIds = Seq.empty)) + ) + + override val queryTransformer: CandidatePipelineQueryTransformer[Query, Unit] = { _ => Unit } + + override val resultTransformer: CandidatePipelineResultsTransformer[ + ShowAlertCandidate, + ShowAlertCandidate + ] = { candidate => candidate } + + override val decorator: Option[CandidateDecorator[Query, ShowAlertCandidate]] = { + val triggerDelayBuilder = new BaseDurationBuilder[Query] { + override def apply( + query: Query, + candidate: ShowAlertCandidate, + features: FeatureMap + ): Option[Duration] = { + val delay = query.deviceContext.flatMap(_.requestContextValue) match { + case Some(DeviceContext.RequestContext.TweetSelfThread) => 0.millis + case Some(DeviceContext.RequestContext.ManualRefresh) => 0.millis + case _ => TriggerDelay + } + + Some(delay) + } + } + + val homeShowAlertCandidateBuilder = ShowAlertCandidateUrtItemBuilder( + alertType = NewTweets, + colorConfigBuilder = StaticShowAlertColorConfigurationBuilder(DefaultColorConfig), + displayLocationBuilder = StaticShowAlertDisplayLocationBuilder(Top), + triggerDelayBuilder = Some(triggerDelayBuilder), + displayDurationBuilder = Some(DurationParamBuilder(StaticParam(DisplayDuration))), + iconDisplayInfoBuilder = Some(StaticShowAlertIconDisplayInfoBuilder(DefaultIconDisplayInfo)) + ) + + Some(UrtItemCandidateDecorator(homeShowAlertCandidateBuilder)) + } + + override val alerts = Seq( + HomeMixerAlertConfig.BusinessHours.defaultSuccessRateAlert(), + HomeMixerAlertConfig.BusinessHours.defaultEmptyResponseRateAlert() + ) +} + +object NewTweetsPillCandidatePipelineConfig { + val DefaultColorConfig: ShowAlertColorConfiguration = ShowAlertColorConfiguration( + background = TwitterBlueRosettaColor, + text = WhiteRosettaColor, + border = Some(WhiteRosettaColor) + ) + + val DefaultIconDisplayInfo: ShowAlertIconDisplayInfo = + ShowAlertIconDisplayInfo(icon = UpArrow, tint = WhiteRosettaColor) + + // Unlimited display time (until user takes action) + val DisplayDuration = -1.millisecond + val TriggerDelay = 4.minutes +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/candidate_pipeline/TimelineServiceResponseFeatureTransformer.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/candidate_pipeline/TimelineServiceResponseFeatureTransformer.scala new file mode 100644 index 0000000000..3f0a932c96 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/candidate_pipeline/TimelineServiceResponseFeatureTransformer.scala @@ -0,0 +1,34 @@ +package com.twitter.home_mixer.candidate_pipeline + +import com.twitter.home_mixer.model.HomeFeatures.AuthorIdFeature +import com.twitter.home_mixer.model.HomeFeatures.InReplyToTweetIdFeature +import com.twitter.home_mixer.model.HomeFeatures.IsRetweetFeature +import com.twitter.home_mixer.model.HomeFeatures.SourceTweetIdFeature +import com.twitter.home_mixer.model.HomeFeatures.SourceUserIdFeature +import com.twitter.product_mixer.core.feature.Feature +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.feature.featuremap.FeatureMapBuilder +import com.twitter.product_mixer.core.functional_component.transformer.CandidateFeatureTransformer +import com.twitter.product_mixer.core.model.common.identifier.TransformerIdentifier +import com.twitter.timelineservice.{thriftscala => t} + +object TimelineServiceResponseFeatureTransformer extends CandidateFeatureTransformer[t.Tweet] { + + override val identifier: TransformerIdentifier = TransformerIdentifier("TimelineServiceResponse") + + override val features: Set[Feature[_, _]] = Set( + AuthorIdFeature, + InReplyToTweetIdFeature, + IsRetweetFeature, + SourceTweetIdFeature, + SourceUserIdFeature, + ) + + override def transform(candidate: t.Tweet): FeatureMap = FeatureMapBuilder() + .add(AuthorIdFeature, candidate.userId) + .add(InReplyToTweetIdFeature, candidate.inReplyToStatusId) + .add(IsRetweetFeature, candidate.sourceStatusId.isDefined) + .add(SourceTweetIdFeature, candidate.sourceStatusId) + .add(SourceUserIdFeature, candidate.sourceUserId) + .build() +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/controller/BUILD.bazel b/home-mixer/server/src/main/scala/com/twitter/home_mixer/controller/BUILD.bazel new file mode 100644 index 0000000000..614dc58eb5 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/controller/BUILD.bazel @@ -0,0 +1,20 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + strict_deps = True, + tags = ["bazel-compatible"], + dependencies = [ + "finatra/thrift/src/main/scala/com/twitter/finatra/thrift:controller", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/marshaller/request", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/service", + "home-mixer/thrift/src/main/thrift:thrift-scala", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/controllers", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/configapi", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/debug_query", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/service/urt", + "snowflake/src/main/scala/com/twitter/snowflake/id", + "src/thrift/com/twitter/context:twitter-context-scala", + "src/thrift/com/twitter/timelines/render:thrift-scala", + "twitter-context/src/main/scala", + ], +) diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/controller/HomeThriftController.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/controller/HomeThriftController.scala new file mode 100644 index 0000000000..fb09b405ea --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/controller/HomeThriftController.scala @@ -0,0 +1,50 @@ +package com.twitter.home_mixer.controller + +import com.twitter.finatra.thrift.Controller +import com.twitter.home_mixer.marshaller.request.HomeMixerRequestUnmarshaller +import com.twitter.home_mixer.model.request.HomeMixerRequest +import com.twitter.home_mixer.service.ScoredTweetsService +import com.twitter.home_mixer.{thriftscala => t} +import com.twitter.product_mixer.core.controllers.DebugTwitterContext +import com.twitter.product_mixer.core.functional_component.configapi.ParamsBuilder +import com.twitter.product_mixer.core.service.urt.UrtService +import com.twitter.snowflake.id.SnowflakeId +import com.twitter.stitch.Stitch +import com.twitter.timelines.configapi.Params +import javax.inject.Inject + +class HomeThriftController @Inject() ( + homeRequestUnmarshaller: HomeMixerRequestUnmarshaller, + urtService: UrtService, + scoredTweetsService: ScoredTweetsService, + paramsBuilder: ParamsBuilder) + extends Controller(t.HomeMixer) + with DebugTwitterContext { + + handle(t.HomeMixer.GetUrtResponse) { args: t.HomeMixer.GetUrtResponse.Args => + val request = homeRequestUnmarshaller(args.request) + val params = buildParams(request) + Stitch.run(urtService.getUrtResponse[HomeMixerRequest](request, params)) + } + + handle(t.HomeMixer.GetScoredTweetsResponse) { args: t.HomeMixer.GetScoredTweetsResponse.Args => + val request = homeRequestUnmarshaller(args.request) + val params = buildParams(request) + withDebugTwitterContext(request.clientContext) { + Stitch.run(scoredTweetsService.getScoredTweetsResponse[HomeMixerRequest](request, params)) + } + } + + private def buildParams(request: HomeMixerRequest): Params = { + val userAgeOpt = request.clientContext.userId.map { userId => + SnowflakeId.timeFromIdOpt(userId).map(_.untilNow.inDays).getOrElse(Int.MaxValue) + } + val fsCustomMapInput = userAgeOpt.map("account_age_in_days" -> _).toMap + paramsBuilder.build( + clientContext = request.clientContext, + product = request.product, + featureOverrides = request.debugParams.flatMap(_.featureOverrides).getOrElse(Map.empty), + fsCustomMapInput = fsCustomMapInput + ) + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/candidate_source/BUILD.bazel b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/candidate_source/BUILD.bazel new file mode 100644 index 0000000000..706f52e2ec --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/candidate_source/BUILD.bazel @@ -0,0 +1,22 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + strict_deps = True, + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/javax/inject:javax.inject", + "finagle/finagle-memcached/src/main/scala", + "finatra/inject/inject-core/src/main/scala", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/param", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/candidate_source", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline", + "src/thrift/com/twitter/hermit/candidate:hermit-candidate-scala", + "src/thrift/com/twitter/search:earlybird-scala", + "stitch/stitch-timelineservice/src/main/scala", + "strato/config/columns/recommendations/similarity:similarity-strato-client", + "strato/src/main/scala/com/twitter/strato/client", + ], + exports = [ + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/candidate_source", + ], +) diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/candidate_source/EarlybirdCandidateSource.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/candidate_source/EarlybirdCandidateSource.scala new file mode 100644 index 0000000000..2ddd05814c --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/candidate_source/EarlybirdCandidateSource.scala @@ -0,0 +1,44 @@ +package com.twitter.home_mixer.functional_component.candidate_source + +import com.twitter.product_mixer.core.feature.FeatureWithDefaultOnFailure +import com.twitter.product_mixer.core.feature.featuremap.FeatureMapBuilder +import com.twitter.product_mixer.core.functional_component.candidate_source.CandidateSourceWithExtractedFeatures +import com.twitter.product_mixer.core.functional_component.candidate_source.CandidatesWithSourceFeatures +import com.twitter.product_mixer.core.model.common.identifier.CandidateSourceIdentifier +import com.twitter.search.earlybird.{thriftscala => t} +import com.twitter.stitch.Stitch +import javax.inject.Inject +import javax.inject.Singleton + +case object EarlybirdResponseTruncatedFeature + extends FeatureWithDefaultOnFailure[t.EarlybirdRequest, Boolean] { + override val defaultValue: Boolean = false +} + +case object EarlybirdBottomTweetFeature + extends FeatureWithDefaultOnFailure[t.EarlybirdRequest, Option[Long]] { + override val defaultValue: Option[Long] = None +} + +@Singleton +case class EarlybirdCandidateSource @Inject() ( + earlybird: t.EarlybirdService.MethodPerEndpoint) + extends CandidateSourceWithExtractedFeatures[t.EarlybirdRequest, t.ThriftSearchResult] { + + override val identifier = CandidateSourceIdentifier("Earlybird") + + override def apply( + request: t.EarlybirdRequest + ): Stitch[CandidatesWithSourceFeatures[t.ThriftSearchResult]] = { + Stitch.callFuture(earlybird.search(request)).map { response => + val candidates = response.searchResults.map(_.results).getOrElse(Seq.empty) + + val features = FeatureMapBuilder() + .add(EarlybirdResponseTruncatedFeature, candidates.size == request.searchQuery.numResults) + .add(EarlybirdBottomTweetFeature, candidates.lastOption.map(_.id)) + .build() + + CandidatesWithSourceFeatures(candidates, features) + } + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/candidate_source/SimilarityBasedUsersCandidateSource.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/candidate_source/SimilarityBasedUsersCandidateSource.scala new file mode 100644 index 0000000000..f117e91f92 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/candidate_source/SimilarityBasedUsersCandidateSource.scala @@ -0,0 +1,34 @@ +package com.twitter.home_mixer.functional_component.candidate_source + +import com.twitter.hermit.candidate.{thriftscala => t} +import com.twitter.product_mixer.core.functional_component.candidate_source.CandidateSource +import com.twitter.product_mixer.core.model.common.identifier.CandidateSourceIdentifier +import com.twitter.stitch.Stitch +import com.twitter.strato.client.Fetcher +import com.twitter.strato.generated.client.recommendations.similarity.SimilarUsersBySimsOnUserClientColumn + +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class SimilarityBasedUsersCandidateSource @Inject() ( + similarUsersBySimsOnUserClientColumn: SimilarUsersBySimsOnUserClientColumn) + extends CandidateSource[Seq[Long], t.Candidate] { + + override val identifier: CandidateSourceIdentifier = + CandidateSourceIdentifier("SimilarityBasedUsers") + + private val fetcher: Fetcher[Long, Unit, t.Candidates] = + similarUsersBySimsOnUserClientColumn.fetcher + + override def apply(request: Seq[Long]): Stitch[Seq[t.Candidate]] = { + Stitch + .collect { + request.map { userId => + fetcher.fetch(userId, Unit).map { result => + result.v.map(_.candidates).getOrElse(Seq.empty) + } + } + }.map(_.flatten) + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/candidate_source/StaleTweetsCacheCandidateSource.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/candidate_source/StaleTweetsCacheCandidateSource.scala new file mode 100644 index 0000000000..9a346129d2 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/candidate_source/StaleTweetsCacheCandidateSource.scala @@ -0,0 +1,30 @@ +package com.twitter.home_mixer.functional_component.candidate_source + +import com.google.inject.name.Named +import com.twitter.finagle.memcached.{Client => MemcachedClient} +import com.twitter.home_mixer.param.HomeMixerInjectionNames.StaleTweetsCache +import com.twitter.product_mixer.core.functional_component.candidate_source.CandidateSource +import com.twitter.product_mixer.core.model.common.identifier.CandidateSourceIdentifier +import com.twitter.stitch.Stitch +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class StaleTweetsCacheCandidateSource @Inject() ( + @Named(StaleTweetsCache) staleTweetsCache: MemcachedClient) + extends CandidateSource[Seq[Long], Long] { + + override val identifier: CandidateSourceIdentifier = CandidateSourceIdentifier("StaleTweetsCache") + + private val StaleTweetsCacheKeyPrefix = "v1_" + + override def apply(request: Seq[Long]): Stitch[Seq[Long]] = { + val keys = request.map(StaleTweetsCacheKeyPrefix + _) + + Stitch.callFuture(staleTweetsCache.get(keys).map { tweets => + tweets.map { + case (k, _) => k.replaceFirst(StaleTweetsCacheKeyPrefix, "").toLong + }.toSeq + }) + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/AuthorChildFeedbackActionBuilder.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/AuthorChildFeedbackActionBuilder.scala new file mode 100644 index 0000000000..712d853605 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/AuthorChildFeedbackActionBuilder.scala @@ -0,0 +1,34 @@ +package com.twitter.home_mixer.functional_component.decorator + +import com.twitter.home_mixer.model.HomeFeatures.ScreenNamesFeature +import com.twitter.home_mixer.model.HomeFeatures.SuggestTypeFeature +import com.twitter.home_mixer.product.following.model.HomeMixerExternalStrings +import com.twitter.home_mixer.util.CandidatesUtil +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.model.marshalling.response.urt.metadata.ChildFeedbackAction +import com.twitter.product_mixer.core.product.guice.scope.ProductScoped +import com.twitter.stringcenter.client.StringCenter +import com.twitter.timelines.service.{thriftscala => t} + +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +case class AuthorChildFeedbackActionBuilder @Inject() ( + @ProductScoped stringCenter: StringCenter, + externalStrings: HomeMixerExternalStrings) { + + def apply(candidateFeatures: FeatureMap): Option[ChildFeedbackAction] = { + CandidatesUtil.getOriginalAuthorId(candidateFeatures).flatMap { authorId => + FeedbackUtil.buildUserSeeFewerChildFeedbackAction( + userId = authorId, + namesByUserId = candidateFeatures.getOrElse(ScreenNamesFeature, Map.empty[Long, String]), + promptExternalString = externalStrings.showFewerTweetsString, + confirmationExternalString = externalStrings.showFewerTweetsConfirmationString, + engagementType = t.FeedbackEngagementType.Tweet, + stringCenter = stringCenter, + injectionType = candidateFeatures.getOrElse(SuggestTypeFeature, None) + ) + } + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/BUILD.bazel b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/BUILD.bazel new file mode 100644 index 0000000000..3ba2d22ae3 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/BUILD.bazel @@ -0,0 +1,32 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + strict_deps = True, + tags = ["bazel-compatible"], + dependencies = [ + "finagle/finagle-core/src/main", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/builder", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/model", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/model/request", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/param", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/product/following/model", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/util", + "joinkey/src/main/scala/com/twitter/joinkey/context", + "joinkey/src/main/thrift/com/twitter/joinkey/context:joinkey-context-scala", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/candidate", + "product-mixer/core/src/main/java/com/twitter/product_mixer/core/product/guice/scope", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/decorator/urt/builder", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline", + "src/scala/com/twitter/suggests/controller_data", + "src/thrift/com/twitter/suggests/controller_data:controller_data-scala", + "src/thrift/com/twitter/timelinescorer/common/scoredtweetcandidate:thrift-scala", + "src/thrift/com/twitter/timelineservice/server/suggests/logging:thrift-scala", + "stringcenter/client", + "stringcenter/client/src/main/java", + "timelinemixer/server/src/main/scala/com/twitter/timelinemixer/injection/model/candidate", + "timelinemixer/server/src/main/scala/com/twitter/timelinemixer/translation", + "timelines/src/main/scala/com/twitter/timelines/injection/scribe", + ], +) diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/BlockUserChildFeedbackActionBuilder.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/BlockUserChildFeedbackActionBuilder.scala new file mode 100644 index 0000000000..2535b24b29 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/BlockUserChildFeedbackActionBuilder.scala @@ -0,0 +1,54 @@ +package com.twitter.home_mixer.functional_component.decorator + +import com.twitter.home_mixer.model.HomeFeatures.AuthorIdFeature +import com.twitter.home_mixer.model.HomeFeatures.IsRetweetFeature +import com.twitter.home_mixer.model.HomeFeatures.ScreenNamesFeature +import com.twitter.home_mixer.model.HomeFeatures.SourceUserIdFeature +import com.twitter.home_mixer.product.following.model.HomeMixerExternalStrings +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.model.marshalling.response.urt.icon +import com.twitter.product_mixer.core.model.marshalling.response.urt.metadata.BottomSheet +import com.twitter.product_mixer.core.model.marshalling.response.urt.metadata.ChildFeedbackAction +import com.twitter.product_mixer.core.model.marshalling.response.urt.metadata.RichBehavior +import com.twitter.product_mixer.core.model.marshalling.response.urt.metadata.RichFeedbackBehaviorBlockUser +import com.twitter.product_mixer.core.product.guice.scope.ProductScoped +import com.twitter.stringcenter.client.StringCenter + +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +case class BlockUserChildFeedbackActionBuilder @Inject() ( + @ProductScoped stringCenter: StringCenter, + externalStrings: HomeMixerExternalStrings) { + + def apply(candidateFeatures: FeatureMap): Option[ChildFeedbackAction] = { + val userIdOpt = + if (candidateFeatures.getOrElse(IsRetweetFeature, false)) + candidateFeatures.getOrElse(SourceUserIdFeature, None) + else candidateFeatures.getOrElse(AuthorIdFeature, None) + + userIdOpt.flatMap { userId => + val screenNamesMap = candidateFeatures.getOrElse(ScreenNamesFeature, Map.empty[Long, String]) + val userScreenNameOpt = screenNamesMap.get(userId) + userScreenNameOpt.map { userScreenName => + val prompt = stringCenter.prepare( + externalStrings.blockUserString, + Map("username" -> userScreenName) + ) + ChildFeedbackAction( + feedbackType = RichBehavior, + prompt = Some(prompt), + confirmation = None, + feedbackUrl = None, + hasUndoAction = Some(true), + confirmationDisplayType = Some(BottomSheet), + clientEventInfo = None, + icon = Some(icon.No), + richBehavior = Some(RichFeedbackBehaviorBlockUser(userId)), + subprompt = None + ) + } + } + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/DontLikeFeedbackActionBuilder.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/DontLikeFeedbackActionBuilder.scala new file mode 100644 index 0000000000..344ce668d4 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/DontLikeFeedbackActionBuilder.scala @@ -0,0 +1,88 @@ +package com.twitter.home_mixer.functional_component.decorator + +import com.twitter.conversions.DurationOps._ +import com.twitter.home_mixer.model.HomeFeatures.SuggestTypeFeature +import com.twitter.home_mixer.param.HomeGlobalParams.EnableNahFeedbackInfoParam +import com.twitter.home_mixer.product.following.model.HomeMixerExternalStrings +import com.twitter.home_mixer.util.CandidatesUtil +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.model.marshalling.response.urt.icon.Frown +import com.twitter.product_mixer.core.model.marshalling.response.urt.metadata.DontLike +import com.twitter.product_mixer.core.model.marshalling.response.urt.metadata.FeedbackAction +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.product_mixer.core.product.guice.scope.ProductScoped +import com.twitter.stringcenter.client.StringCenter +import com.twitter.timelines.common.{thriftscala => tlc} +import com.twitter.timelineservice.model.FeedbackInfo +import com.twitter.timelineservice.model.FeedbackMetadata +import com.twitter.timelineservice.{thriftscala => tls} + +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +case class DontLikeFeedbackActionBuilder @Inject() ( + @ProductScoped stringCenter: StringCenter, + externalStrings: HomeMixerExternalStrings, + authorChildFeedbackActionBuilder: AuthorChildFeedbackActionBuilder, + retweeterChildFeedbackActionBuilder: RetweeterChildFeedbackActionBuilder, + notRelevantChildFeedbackActionBuilder: NotRelevantChildFeedbackActionBuilder, + unfollowUserChildFeedbackActionBuilder: UnfollowUserChildFeedbackActionBuilder, + muteUserChildFeedbackActionBuilder: MuteUserChildFeedbackActionBuilder, + blockUserChildFeedbackActionBuilder: BlockUserChildFeedbackActionBuilder, + reportTweetChildFeedbackActionBuilder: ReportTweetChildFeedbackActionBuilder) { + + def apply( + query: PipelineQuery, + candidate: TweetCandidate, + candidateFeatures: FeatureMap + ): Option[FeedbackAction] = { + CandidatesUtil.getOriginalAuthorId(candidateFeatures).map { authorId => + val feedbackEntities = Seq( + tlc.FeedbackEntity.TweetId(candidate.id), + tlc.FeedbackEntity.UserId(authorId) + ) + val feedbackMetadata = FeedbackMetadata( + engagementType = None, + entityIds = feedbackEntities, + ttl = Some(30.days) + ) + val feedbackUrl = FeedbackInfo.feedbackUrl( + feedbackType = tls.FeedbackType.DontLike, + feedbackMetadata = feedbackMetadata, + injectionType = candidateFeatures.getOrElse(SuggestTypeFeature, None) + ) + val childFeedbackActions = if (query.params(EnableNahFeedbackInfoParam)) { + Seq( + unfollowUserChildFeedbackActionBuilder(candidateFeatures), + muteUserChildFeedbackActionBuilder(candidateFeatures), + blockUserChildFeedbackActionBuilder(candidateFeatures), + reportTweetChildFeedbackActionBuilder(candidate) + ).flatten + } else { + Seq( + authorChildFeedbackActionBuilder(candidateFeatures), + retweeterChildFeedbackActionBuilder(candidateFeatures), + notRelevantChildFeedbackActionBuilder(candidate, candidateFeatures) + ).flatten + } + + FeedbackAction( + feedbackType = DontLike, + prompt = Some(stringCenter.prepare(externalStrings.dontLikeString)), + confirmation = Some(stringCenter.prepare(externalStrings.dontLikeConfirmationString)), + childFeedbackActions = + if (childFeedbackActions.nonEmpty) Some(childFeedbackActions) else None, + feedbackUrl = Some(feedbackUrl), + hasUndoAction = Some(true), + confirmationDisplayType = None, + clientEventInfo = None, + icon = Some(Frown), + richBehavior = None, + subprompt = None, + encodedFeedbackRequest = None + ) + } + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/EngagerSocialContextBuilder.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/EngagerSocialContextBuilder.scala new file mode 100644 index 0000000000..5e9f405602 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/EngagerSocialContextBuilder.scala @@ -0,0 +1,119 @@ +package com.twitter.home_mixer.functional_component.decorator + +import com.twitter.home_mixer.model.HomeFeatures.RealNamesFeature +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.model.marshalling.response.urt.metadata._ +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.stringcenter.client.StringCenter +import com.twitter.stringcenter.client.core.ExternalString + +private[decorator] case class SocialContextIdAndScreenName( + socialContextId: Long, + screenName: String) + +object EngagerSocialContextBuilder { + private val UserIdRequestParamName = "user_id" + private val DirectInjectionContentSourceRequestParamName = "dis" + private val DirectInjectionIdRequestParamName = "diid" + private val DirectInjectionContentSourceSocialProofUsers = "socialproofusers" + private val SocialProofUrl = "" +} + +case class EngagerSocialContextBuilder( + contextType: GeneralContextType, + stringCenter: StringCenter, + oneUserString: ExternalString, + twoUsersString: ExternalString, + moreUsersString: ExternalString, + timelineTitle: ExternalString) { + import EngagerSocialContextBuilder._ + + def apply( + socialContextIds: Seq[Long], + query: PipelineQuery, + candidateFeatures: FeatureMap + ): Option[SocialContext] = { + val realNames = candidateFeatures.getOrElse(RealNamesFeature, Map.empty[Long, String]) + val validSocialContextIdAndScreenNames = socialContextIds.flatMap { socialContextId => + realNames + .get(socialContextId).map(screenName => + SocialContextIdAndScreenName(socialContextId, screenName)) + } + + validSocialContextIdAndScreenNames match { + case Seq(user) => + val socialContextString = + stringCenter.prepare(oneUserString, Map("user" -> user.screenName)) + Some(mkOneUserSocialContext(socialContextString, user.socialContextId)) + case Seq(firstUser, secondUser) => + val socialContextString = + stringCenter + .prepare( + twoUsersString, + Map("user1" -> firstUser.screenName, "user2" -> secondUser.screenName)) + Some( + mkManyUserSocialContext( + socialContextString, + query.getRequiredUserId, + validSocialContextIdAndScreenNames.map(_.socialContextId))) + + case firstUser +: otherUsers => + val otherUsersCount = otherUsers.size + val socialContextString = + stringCenter + .prepare( + moreUsersString, + Map("user" -> firstUser.screenName, "count" -> otherUsersCount)) + Some( + mkManyUserSocialContext( + socialContextString, + query.getRequiredUserId, + validSocialContextIdAndScreenNames.map(_.socialContextId))) + case _ => None + } + } + + private def mkOneUserSocialContext(socialContextString: String, userId: Long): GeneralContext = { + GeneralContext( + contextType = contextType, + text = socialContextString, + url = None, + contextImageUrls = None, + landingUrl = Some( + Url( + urlType = DeepLink, + url = "", + urtEndpointOptions = None + ) + ) + ) + } + + private def mkManyUserSocialContext( + socialContextString: String, + viewerId: Long, + socialContextIds: Seq[Long] + ): GeneralContext = { + GeneralContext( + contextType = contextType, + text = socialContextString, + url = None, + contextImageUrls = None, + landingUrl = Some( + Url( + urlType = UrtEndpoint, + url = SocialProofUrl, + urtEndpointOptions = Some(UrtEndpointOptions( + requestParams = Some(Map( + UserIdRequestParamName -> viewerId.toString, + DirectInjectionContentSourceRequestParamName -> DirectInjectionContentSourceSocialProofUsers, + DirectInjectionIdRequestParamName -> socialContextIds.mkString(",") + )), + title = Some(stringCenter.prepare(timelineTitle)), + cacheId = None, + subtitle = None + )) + )) + ) + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/ExtendedReplySocialContextBuilder.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/ExtendedReplySocialContextBuilder.scala new file mode 100644 index 0000000000..88ffa1cd1b --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/ExtendedReplySocialContextBuilder.scala @@ -0,0 +1,78 @@ +package com.twitter.home_mixer.functional_component.decorator + +import com.twitter.home_mixer.model.HomeFeatures.FocalTweetAuthorIdFeature +import com.twitter.home_mixer.model.HomeFeatures.FocalTweetInNetworkFeature +import com.twitter.home_mixer.model.HomeFeatures.FocalTweetRealNamesFeature +import com.twitter.home_mixer.model.HomeFeatures.InNetworkFeature +import com.twitter.home_mixer.product.following.model.HomeMixerExternalStrings +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.functional_component.decorator.urt.builder.social_context.BaseSocialContextBuilder +import com.twitter.product_mixer.core.model.marshalling.response.urt.metadata.SocialContext +import com.twitter.product_mixer.core.model.marshalling.response.urt.metadata._ +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.product_mixer.core.product.guice.scope.ProductScoped +import com.twitter.stringcenter.client.StringCenter +import javax.inject.Inject +import javax.inject.Provider +import javax.inject.Singleton + +/** + * Use '@A replied' when the root tweet is out-of-network and the reply is in network. + * + * This function should only be called for the root Tweet of convo modules. This is enforced by + * [[HomeTweetSocialContextBuilder]]. + */ +@Singleton +case class ExtendedReplySocialContextBuilder @Inject() ( + externalStrings: HomeMixerExternalStrings, + @ProductScoped stringCenterProvider: Provider[StringCenter]) + extends BaseSocialContextBuilder[PipelineQuery, TweetCandidate] { + + private val stringCenter = stringCenterProvider.get() + private val extendedReplyString = externalStrings.socialContextExtendedReply + + def apply( + query: PipelineQuery, + candidate: TweetCandidate, + candidateFeatures: FeatureMap + ): Option[SocialContext] = { + + // If these values are missing default to not showing an extended reply banner + val inNetworkRoot = candidateFeatures.getOrElse(InNetworkFeature, true) + + val inNetworkFocalTweet = + candidateFeatures.getOrElse(FocalTweetInNetworkFeature, None).getOrElse(false) + + if (!inNetworkRoot && inNetworkFocalTweet) { + + val focalTweetAuthorIdOpt = candidateFeatures.getOrElse(FocalTweetAuthorIdFeature, None) + val focalTweetRealNames = + candidateFeatures + .getOrElse(FocalTweetRealNamesFeature, None).getOrElse(Map.empty[Long, String]) + val focalTweetAuthorNameOpt = focalTweetAuthorIdOpt.flatMap(focalTweetRealNames.get) + + (focalTweetAuthorIdOpt, focalTweetAuthorNameOpt) match { + case (Some(focalTweetAuthorId), Some(focalTweetAuthorName)) => + Some( + GeneralContext( + contextType = ConversationGeneralContextType, + text = stringCenter + .prepare(extendedReplyString, placeholders = Map("user1" -> focalTweetAuthorName)), + url = None, + contextImageUrls = None, + landingUrl = Some( + Url( + urlType = DeepLink, + url = "", + urtEndpointOptions = None + )) + )) + case _ => + None + } + } else { + None + } + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/FeedbackUtil.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/FeedbackUtil.scala new file mode 100644 index 0000000000..49e24d0b48 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/FeedbackUtil.scala @@ -0,0 +1,61 @@ +package com.twitter.home_mixer.functional_component.decorator + +import com.twitter.conversions.DurationOps._ +import com.twitter.product_mixer.core.model.marshalling.response.urt.metadata.ChildFeedbackAction +import com.twitter.product_mixer.core.model.marshalling.response.urt.metadata.SeeFewer +import com.twitter.stringcenter.client.StringCenter +import com.twitter.stringcenter.client.core.ExternalString +import com.twitter.timelines.common.{thriftscala => tlc} +import com.twitter.timelines.service.{thriftscala => t} +import com.twitter.timelineservice.model.FeedbackInfo +import com.twitter.timelineservice.model.FeedbackMetadata +import com.twitter.timelineservice.suggests.{thriftscala => st} +import com.twitter.timelineservice.{thriftscala => tlst} + +object FeedbackUtil { + + val FeedbackTtl = 30.days + + def buildUserSeeFewerChildFeedbackAction( + userId: Long, + namesByUserId: Map[Long, String], + promptExternalString: ExternalString, + confirmationExternalString: ExternalString, + engagementType: t.FeedbackEngagementType, + stringCenter: StringCenter, + injectionType: Option[st.SuggestType] + ): Option[ChildFeedbackAction] = { + namesByUserId.get(userId).map { userScreenName => + val prompt = stringCenter.prepare( + promptExternalString, + Map("user" -> userScreenName) + ) + val confirmation = stringCenter.prepare( + confirmationExternalString, + Map("user" -> userScreenName) + ) + val feedbackMetadata = FeedbackMetadata( + engagementType = Some(engagementType), + entityIds = Seq(tlc.FeedbackEntity.UserId(userId)), + ttl = Some(FeedbackTtl)) + val feedbackUrl = FeedbackInfo.feedbackUrl( + feedbackType = tlst.FeedbackType.SeeFewer, + feedbackMetadata = feedbackMetadata, + injectionType = injectionType + ) + + ChildFeedbackAction( + feedbackType = SeeFewer, + prompt = Some(prompt), + confirmation = Some(confirmation), + feedbackUrl = Some(feedbackUrl), + hasUndoAction = Some(true), + confirmationDisplayType = None, + clientEventInfo = None, + icon = None, + richBehavior = None, + subprompt = None + ) + } + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/FollowedBySocialContextBuilder.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/FollowedBySocialContextBuilder.scala new file mode 100644 index 0000000000..0b4882c21f --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/FollowedBySocialContextBuilder.scala @@ -0,0 +1,53 @@ +package com.twitter.home_mixer.functional_component.decorator + +import com.twitter.home_mixer.model.HomeFeatures.InNetworkFeature +import com.twitter.home_mixer.model.HomeFeatures.SGSValidFollowedByUserIdsFeature +import com.twitter.home_mixer.product.following.model.HomeMixerExternalStrings +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.functional_component.decorator.urt.builder.social_context.BaseSocialContextBuilder +import com.twitter.product_mixer.core.model.marshalling.response.urt.metadata._ +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.product_mixer.core.product.guice.scope.ProductScoped +import com.twitter.stringcenter.client.StringCenter +import javax.inject.Inject +import javax.inject.Provider +import javax.inject.Singleton + +@Singleton +case class FollowedBySocialContextBuilder @Inject() ( + externalStrings: HomeMixerExternalStrings, + @ProductScoped stringCenterProvider: Provider[StringCenter]) + extends BaseSocialContextBuilder[PipelineQuery, TweetCandidate] { + + private val stringCenter = stringCenterProvider.get() + + private val engagerSocialContextBuilder = EngagerSocialContextBuilder( + contextType = FollowGeneralContextType, + stringCenter = stringCenter, + oneUserString = externalStrings.socialContextOneUserFollowsString, + twoUsersString = externalStrings.socialContextTwoUsersFollowString, + moreUsersString = externalStrings.socialContextMoreUsersFollowString, + timelineTitle = externalStrings.socialContextFollowedByTimelineTitle + ) + + def apply( + query: PipelineQuery, + candidate: TweetCandidate, + candidateFeatures: FeatureMap + ): Option[SocialContext] = { + // Only apply followed-by social context for OON Tweets + val inNetwork = candidateFeatures.getOrElse(InNetworkFeature, true) + if (!inNetwork) { + val validFollowedByUserIds = + candidateFeatures.getOrElse(SGSValidFollowedByUserIdsFeature, Nil) + engagerSocialContextBuilder( + socialContextIds = validFollowedByUserIds, + query = query, + candidateFeatures = candidateFeatures + ) + } else { + None + } + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/HomeAdsClientEventDetailsBuilder.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/HomeAdsClientEventDetailsBuilder.scala new file mode 100644 index 0000000000..c81223a535 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/HomeAdsClientEventDetailsBuilder.scala @@ -0,0 +1,46 @@ +package com.twitter.home_mixer.functional_component.decorator + +import com.twitter.finagle.tracing.Trace +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.functional_component.decorator.urt.builder.metadata.BaseClientEventDetailsBuilder +import com.twitter.product_mixer.core.model.common.UniversalNoun +import com.twitter.product_mixer.core.model.marshalling.response.urt.metadata.ClientEventDetails +import com.twitter.product_mixer.core.model.marshalling.response.urt.metadata.TimelinesDetails +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.suggests.controller_data.home_tweets.v1.{thriftscala => v1ht} +import com.twitter.suggests.controller_data.home_tweets.{thriftscala => ht} +import com.twitter.suggests.controller_data.thriftscala.ControllerData +import com.twitter.suggests.controller_data.v2.thriftscala.{ControllerData => ControllerDataV2} + +case class HomeAdsClientEventDetailsBuilder(injectionType: Option[String]) + extends BaseClientEventDetailsBuilder[PipelineQuery, UniversalNoun[Any]] { + + override def apply( + query: PipelineQuery, + candidate: UniversalNoun[Any], + candidateFeatures: FeatureMap + ): Option[ClientEventDetails] = { + val homeTweetsControllerDataV1 = v1ht.HomeTweetsControllerData( + tweetTypesBitmap = 0L, + traceId = Some(Trace.id.traceId.toLong), + requestJoinId = None) + + val serializedControllerData = HomeClientEventDetailsBuilder.ControllerDataSerializer( + ControllerData.V2( + ControllerDataV2.HomeTweets(ht.HomeTweetsControllerData.V1(homeTweetsControllerDataV1)))) + + val clientEventDetails = ClientEventDetails( + conversationDetails = None, + timelinesDetails = Some( + TimelinesDetails( + injectionType = injectionType, + controllerData = Some(serializedControllerData), + sourceData = None)), + articleDetails = None, + liveEventDetails = None, + commerceDetails = None + ) + + Some(clientEventDetails) + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/HomeClientEventDetailsBuilder.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/HomeClientEventDetailsBuilder.scala new file mode 100644 index 0000000000..801ac0f846 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/HomeClientEventDetailsBuilder.scala @@ -0,0 +1,92 @@ +package com.twitter.home_mixer.functional_component.decorator + +import com.twitter.bijection.Base64String +import com.twitter.bijection.scrooge.BinaryScalaCodec +import com.twitter.bijection.{Injection => Serializer} +import com.twitter.finagle.tracing.Trace +import com.twitter.home_mixer.model.HomeFeatures.CandidateSourceIdFeature +import com.twitter.home_mixer.model.HomeFeatures.PositionFeature +import com.twitter.home_mixer.model.HomeFeatures.SuggestTypeFeature +import com.twitter.joinkey.context.RequestJoinKeyContext +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.functional_component.decorator.urt.builder.metadata.BaseClientEventDetailsBuilder +import com.twitter.product_mixer.core.model.common.UniversalNoun +import com.twitter.product_mixer.core.model.marshalling.response.urt.metadata.ClientEventDetails +import com.twitter.product_mixer.core.model.marshalling.response.urt.metadata.TimelinesDetails +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.suggests.controller_data.Home +import com.twitter.suggests.controller_data.TweetTypeGenerator +import com.twitter.suggests.controller_data.home_tweets.v1.{thriftscala => v1ht} +import com.twitter.suggests.controller_data.home_tweets.{thriftscala => ht} +import com.twitter.suggests.controller_data.thriftscala.ControllerData +import com.twitter.suggests.controller_data.v2.thriftscala.{ControllerData => ControllerDataV2} + +object HomeClientEventDetailsBuilder { + implicit val ByteSerializer: Serializer[ControllerData, Array[Byte]] = + BinaryScalaCodec(ControllerData) + + val ControllerDataSerializer: Serializer[ControllerData, String] = + Serializer.connect[ControllerData, Array[Byte], Base64String, String] + + /** + * define getRequestJoinId as a method(def) rather than a val because each new request + * needs to call the context to update the id. + */ + private def getRequestJoinId(): Option[Long] = + RequestJoinKeyContext.current.flatMap(_.requestJoinId) +} + +case class HomeClientEventDetailsBuilder[-Query <: PipelineQuery, -Candidate <: UniversalNoun[Any]]( +) extends BaseClientEventDetailsBuilder[Query, Candidate] + with TweetTypeGenerator[FeatureMap] { + + import HomeClientEventDetailsBuilder._ + + override def apply( + query: Query, + candidate: Candidate, + candidateFeatures: FeatureMap + ): Option[ClientEventDetails] = { + + val tweetTypesBitmaps = mkTweetTypesBitmaps( + Home.TweetTypeIdxMap, + HomeTweetTypePredicates.PredicateMap, + candidateFeatures) + + val tweetTypesListBytes = mkItemTypesBitmapsV2( + Home.TweetTypeIdxMap, + HomeTweetTypePredicates.PredicateMap, + candidateFeatures) + + val candidateSourceId = + candidateFeatures.getOrElse(CandidateSourceIdFeature, None).map(_.value.toByte) + + val homeTweetsControllerDataV1 = v1ht.HomeTweetsControllerData( + tweetTypesBitmap = tweetTypesBitmaps.getOrElse(0, 0L), + tweetTypesBitmapContinued1 = tweetTypesBitmaps.get(1), + candidateTweetSourceId = candidateSourceId, + traceId = Some(Trace.id.traceId.toLong), + injectedPosition = candidateFeatures.getOrElse(PositionFeature, None), + tweetTypesListBytes = Some(tweetTypesListBytes), + requestJoinId = getRequestJoinId(), + ) + + val serializedControllerData = ControllerDataSerializer( + ControllerData.V2( + ControllerDataV2.HomeTweets(ht.HomeTweetsControllerData.V1(homeTweetsControllerDataV1)))) + + val clientEventDetails = ClientEventDetails( + conversationDetails = None, + timelinesDetails = Some( + TimelinesDetails( + injectionType = candidateFeatures.getOrElse(SuggestTypeFeature, None).map(_.name), + controllerData = Some(serializedControllerData), + sourceData = None)), + articleDetails = None, + liveEventDetails = None, + commerceDetails = None + ) + + Some(clientEventDetails) + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/HomeConversationServiceCandidateDecorator.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/HomeConversationServiceCandidateDecorator.scala new file mode 100644 index 0000000000..9a94f1c3b4 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/HomeConversationServiceCandidateDecorator.scala @@ -0,0 +1,49 @@ +package com.twitter.home_mixer.functional_component.decorator + +import com.twitter.home_mixer.functional_component.decorator.builder.HomeConversationModuleMetadataBuilder +import com.twitter.home_mixer.model.HomeFeatures.ConversationModuleFocalTweetIdFeature +import com.twitter.product_mixer.component_library.decorator.urt.UrtItemCandidateDecorator +import com.twitter.product_mixer.component_library.decorator.urt.UrtMultipleModulesDecorator +import com.twitter.product_mixer.component_library.decorator.urt.builder.item.tweet.TweetCandidateUrtItemBuilder +import com.twitter.product_mixer.component_library.decorator.urt.builder.metadata.ClientEventInfoBuilder +import com.twitter.product_mixer.component_library.decorator.urt.builder.timeline_module.StaticModuleDisplayTypeBuilder +import com.twitter.product_mixer.component_library.decorator.urt.builder.timeline_module.TimelineModuleBuilder +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.model.marshalling.response.urt.EntryNamespace +import com.twitter.product_mixer.core.model.marshalling.response.urt.timeline_module.VerticalConversation +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.timelines.injection.scribe.InjectionScribeUtil +import com.twitter.timelineservice.suggests.{thriftscala => st} + +object HomeConversationServiceCandidateDecorator { + + private val ConversationModuleNamespace = EntryNamespace("home-conversation") + + def apply( + homeFeedbackActionInfoBuilder: HomeFeedbackActionInfoBuilder + ): Some[UrtMultipleModulesDecorator[PipelineQuery, TweetCandidate, Long]] = { + val suggestType = st.SuggestType.RankedOrganicTweet + val component = InjectionScribeUtil.scribeComponent(suggestType).get + val clientEventInfoBuilder = ClientEventInfoBuilder(component) + val tweetItemBuilder = TweetCandidateUrtItemBuilder( + clientEventInfoBuilder = clientEventInfoBuilder, + timelinesScoreInfoBuilder = Some(HomeTimelinesScoreInfoBuilder), + feedbackActionInfoBuilder = Some(homeFeedbackActionInfoBuilder) + ) + + val moduleBuilder = TimelineModuleBuilder( + entryNamespace = ConversationModuleNamespace, + clientEventInfoBuilder = clientEventInfoBuilder, + displayTypeBuilder = StaticModuleDisplayTypeBuilder(VerticalConversation), + metadataBuilder = Some(HomeConversationModuleMetadataBuilder()) + ) + + Some( + UrtMultipleModulesDecorator( + urtItemCandidateDecorator = UrtItemCandidateDecorator(tweetItemBuilder), + moduleBuilder = moduleBuilder, + groupByKey = (_, _, candidateFeatures) => + candidateFeatures.getOrElse(ConversationModuleFocalTweetIdFeature, None) + )) + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/HomeFeedbackActionInfoBuilder.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/HomeFeedbackActionInfoBuilder.scala new file mode 100644 index 0000000000..032874ead4 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/HomeFeedbackActionInfoBuilder.scala @@ -0,0 +1,54 @@ +package com.twitter.home_mixer.functional_component.decorator + +import com.twitter.home_mixer.model.HomeFeatures.SuggestTypeFeature +import com.twitter.home_mixer.model.request.FollowingProduct +import com.twitter.home_mixer.model.request.ForYouProduct +import com.twitter.home_mixer.param.HomeGlobalParams.EnableNahFeedbackInfoParam +import com.twitter.home_mixer.util.CandidatesUtil +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.functional_component.decorator.urt.builder.metadata.BaseFeedbackActionInfoBuilder +import com.twitter.product_mixer.core.model.marshalling.response.urt.metadata.FeedbackActionInfo +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.timelines.service.{thriftscala => t} +import com.twitter.timelines.util.FeedbackMetadataSerializer + +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class HomeFeedbackActionInfoBuilder @Inject() ( + notInterestedTopicFeedbackActionBuilder: NotInterestedTopicFeedbackActionBuilder, + dontLikeFeedbackActionBuilder: DontLikeFeedbackActionBuilder) + extends BaseFeedbackActionInfoBuilder[PipelineQuery, TweetCandidate] { + + override def apply( + query: PipelineQuery, + candidate: TweetCandidate, + candidateFeatures: FeatureMap + ): Option[FeedbackActionInfo] = { + val supportedProduct = query.product match { + case FollowingProduct => query.params(EnableNahFeedbackInfoParam) + case ForYouProduct => true + case _ => false + } + val isAuthoredByViewer = CandidatesUtil.isAuthoredByViewer(query, candidateFeatures) + + if (supportedProduct && !isAuthoredByViewer) { + val feedbackActions = Seq( + notInterestedTopicFeedbackActionBuilder(candidateFeatures), + dontLikeFeedbackActionBuilder(query, candidate, candidateFeatures) + ).flatten + val feedbackMetadata = FeedbackMetadataSerializer.serialize( + t.FeedbackMetadata(injectionType = candidateFeatures.getOrElse(SuggestTypeFeature, None))) + + Some( + FeedbackActionInfo( + feedbackActions = feedbackActions, + feedbackMetadata = Some(feedbackMetadata), + displayContext = None, + clientEventInfo = None + )) + } else None + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/HomeQueryTypePredicates.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/HomeQueryTypePredicates.scala new file mode 100644 index 0000000000..a795432695 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/HomeQueryTypePredicates.scala @@ -0,0 +1,18 @@ +package com.twitter.home_mixer.functional_component.decorator + +import com.twitter.home_mixer.model.HomeFeatures._ +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap + +object HomeQueryTypePredicates { + private[this] val QueryPredicates: Seq[(String, FeatureMap => Boolean)] = Seq( + ("request", _ => true), + ("get_initial", _.getOrElse(GetInitialFeature, false)), + ("get_newer", _.getOrElse(GetNewerFeature, false)), + ("get_older", _.getOrElse(GetOlderFeature, false)), + ("pull_to_refresh", _.getOrElse(PullToRefreshFeature, false)), + ("request_context_launch", _.getOrElse(IsLaunchRequestFeature, false)), + ("request_context_foreground", _.getOrElse(IsForegroundRequestFeature, false)) + ) + + val PredicateMap = QueryPredicates.toMap +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/HomeTimelinesScoreInfoBuilder.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/HomeTimelinesScoreInfoBuilder.scala new file mode 100644 index 0000000000..bc072ec346 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/HomeTimelinesScoreInfoBuilder.scala @@ -0,0 +1,26 @@ +package com.twitter.home_mixer.functional_component.decorator + +import com.twitter.home_mixer.model.HomeFeatures.ScoreFeature +import com.twitter.home_mixer.param.HomeGlobalParams.EnableSendScoresToClient +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.functional_component.decorator.urt.builder.item.tweet.BaseTimelinesScoreInfoBuilder +import com.twitter.product_mixer.core.model.marshalling.response.urt.item.tweet.TimelinesScoreInfo +import com.twitter.product_mixer.core.pipeline.PipelineQuery + +object HomeTimelinesScoreInfoBuilder + extends BaseTimelinesScoreInfoBuilder[PipelineQuery, TweetCandidate] { + + private val UndefinedTweetScore = -1.0 + + override def apply( + query: PipelineQuery, + candidate: TweetCandidate, + candidateFeatures: FeatureMap + ): Option[TimelinesScoreInfo] = { + if (query.params(EnableSendScoresToClient)) { + val score = candidateFeatures.getOrElse(ScoreFeature, None).getOrElse(UndefinedTweetScore) + Some(TimelinesScoreInfo(score)) + } else None + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/HomeTweetSocialContextBuilder.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/HomeTweetSocialContextBuilder.scala new file mode 100644 index 0000000000..4f3e03f6d2 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/HomeTweetSocialContextBuilder.scala @@ -0,0 +1,44 @@ +package com.twitter.home_mixer.functional_component.decorator + +import com.twitter.home_mixer.model.HomeFeatures.ConversationModuleFocalTweetIdFeature +import com.twitter.home_mixer.model.HomeFeatures.ConversationModuleIdFeature +import com.twitter.home_mixer.param.HomeGlobalParams.EnableSocialContextParam +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.functional_component.decorator.urt.builder.social_context.BaseSocialContextBuilder +import com.twitter.product_mixer.core.model.marshalling.response.urt.metadata.SocialContext +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +case class HomeTweetSocialContextBuilder @Inject() ( + likedBySocialContextBuilder: LikedBySocialContextBuilder, + followedBySocialContextBuilder: FollowedBySocialContextBuilder, + topicSocialContextBuilder: TopicSocialContextBuilder, + extendedReplySocialContextBuilder: ExtendedReplySocialContextBuilder, + receivedReplySocialContextBuilder: ReceivedReplySocialContextBuilder) + extends BaseSocialContextBuilder[PipelineQuery, TweetCandidate] { + + def apply( + query: PipelineQuery, + candidate: TweetCandidate, + features: FeatureMap + ): Option[SocialContext] = { + if (query.params(EnableSocialContextParam)) { + features.getOrElse(ConversationModuleFocalTweetIdFeature, None) match { + case None => + likedBySocialContextBuilder(query, candidate, features) + .orElse(followedBySocialContextBuilder(query, candidate, features)) + .orElse(topicSocialContextBuilder(query, candidate, features)) + case Some(_) => + val conversationId = features.getOrElse(ConversationModuleIdFeature, None) + // Only hydrate the social context into the root tweet in a conversation module + if (conversationId.contains(candidate.id)) { + extendedReplySocialContextBuilder(query, candidate, features) + .orElse(receivedReplySocialContextBuilder(query, candidate, features)) + } else None + } + } else None + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/HomeTweetTypePredicates.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/HomeTweetTypePredicates.scala new file mode 100644 index 0000000000..546cd13e98 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/HomeTweetTypePredicates.scala @@ -0,0 +1,250 @@ +package com.twitter.home_mixer.functional_component.decorator + +import com.twitter.conversions.DurationOps._ +import com.twitter.home_mixer.model.HomeFeatures._ +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.timelinemixer.injection.model.candidate.SemanticCoreFeatures +import com.twitter.tweetypie.{thriftscala => tpt} + +object HomeTweetTypePredicates { + + /** + * IMPORTANT: Please avoid logging tweet types that are tied to sensitive + * internal author information / labels (e.g. blink labels, abuse labels, or geo-location). + */ + private[this] val CandidatePredicates: Seq[(String, FeatureMap => Boolean)] = Seq( + ("with_candidate", _ => true), + ("retweet", _.getOrElse(IsRetweetFeature, false)), + ("reply", _.getOrElse(InReplyToTweetIdFeature, None).nonEmpty), + ("image", _.getOrElse(EarlybirdFeature, None).exists(_.hasImage)), + ("video", _.getOrElse(EarlybirdFeature, None).exists(_.hasVideo)), + ("link", _.getOrElse(EarlybirdFeature, None).exists(_.hasVisibleLink)), + ("quote", _.getOrElse(EarlybirdFeature, None).exists(_.hasQuote.contains(true))), + ("like_social_context", _.getOrElse(NonSelfFavoritedByUserIdsFeature, Seq.empty).nonEmpty), + ("protected", _.getOrElse(EarlybirdFeature, None).exists(_.isProtected)), + ( + "has_exclusive_conversation_author_id", + _.getOrElse(ExclusiveConversationAuthorIdFeature, None).nonEmpty), + ("is_eligible_for_connect_boost", _.getOrElse(AuthorIsEligibleForConnectBoostFeature, false)), + ("hashtag", _.getOrElse(EarlybirdFeature, None).exists(_.numHashtags > 0)), + ("has_scheduled_space", _.getOrElse(AudioSpaceMetaDataFeature, None).exists(_.isScheduled)), + ("has_recorded_space", _.getOrElse(AudioSpaceMetaDataFeature, None).exists(_.isRecorded)), + ("is_read_from_cache", _.getOrElse(IsReadFromCacheFeature, false)), + ( + "is_self_thread_tweet", + _.getOrElse(ConversationFeature, None).exists(_.isSelfThreadTweet.contains(true))), + ("get_initial", _.getOrElse(GetInitialFeature, false)), + ("get_newer", _.getOrElse(GetNewerFeature, false)), + ("get_middle", _.getOrElse(GetMiddleFeature, false)), + ("get_older", _.getOrElse(GetOlderFeature, false)), + ("pull_to_refresh", _.getOrElse(PullToRefreshFeature, false)), + ("polling", _.getOrElse(PollingFeature, false)), + ("tls_size_20_plus", _ => false), + ("near_empty", _ => false), + ("ranked_request", _ => false), + ("mutual_follow", _.getOrElse(EarlybirdFeature, None).exists(_.fromMutualFollow)), + ("has_ticketed_space", _.getOrElse(AudioSpaceMetaDataFeature, None).exists(_.hasTickets)), + ("in_utis_top5", _.getOrElse(PositionFeature, None).exists(_ < 5)), + ("is_utis_pos0", _.getOrElse(PositionFeature, None).exists(_ == 0)), + ("is_utis_pos1", _.getOrElse(PositionFeature, None).exists(_ == 1)), + ("is_utis_pos2", _.getOrElse(PositionFeature, None).exists(_ == 2)), + ("is_utis_pos3", _.getOrElse(PositionFeature, None).exists(_ == 3)), + ("is_utis_pos4", _.getOrElse(PositionFeature, None).exists(_ == 4)), + ( + "is_signup_request", + candidate => candidate.getOrElse(AccountAgeFeature, None).exists(_.untilNow < 30.minutes)), + ("empty_request", _ => false), + ("served_size_less_than_5", _.getOrElse(ServedSizeFeature, None).exists(_ < 5)), + ("served_size_less_than_10", _.getOrElse(ServedSizeFeature, None).exists(_ < 10)), + ("served_size_less_than_20", _.getOrElse(ServedSizeFeature, None).exists(_ < 20)), + ("served_size_less_than_50", _.getOrElse(ServedSizeFeature, None).exists(_ < 50)), + ( + "served_size_between_50_and_100", + _.getOrElse(ServedSizeFeature, None).exists(size => size >= 50 && size < 100)), + ("authored_by_contextual_user", _.getOrElse(AuthoredByContextualUserFeature, false)), + ("has_ancestors", _.getOrElse(AncestorsFeature, Seq.empty).nonEmpty), + ("full_scoring_succeeded", _.getOrElse(FullScoringSucceededFeature, false)), + ( + "account_age_less_than_30_minutes", + _.getOrElse(AccountAgeFeature, None).exists(_.untilNow < 30.minutes)), + ( + "account_age_less_than_1_day", + _.getOrElse(AccountAgeFeature, None).exists(_.untilNow < 1.day)), + ( + "account_age_less_than_7_days", + _.getOrElse(AccountAgeFeature, None).exists(_.untilNow < 7.days)), + ( + "directed_at_user_is_in_first_degree", + _.getOrElse(EarlybirdFeature, None).exists(_.directedAtUserIdIsInFirstDegree.contains(true))), + ("root_user_is_in_first_degree", _ => false), + ( + "has_semantic_core_annotation", + _.getOrElse(EarlybirdFeature, None).exists(_.semanticCoreAnnotations.nonEmpty)), + ("is_request_context_foreground", _.getOrElse(IsForegroundRequestFeature, false)), + ( + "part_of_utt", + _.getOrElse(EarlybirdFeature, None) + .exists(_.semanticCoreAnnotations.exists(_.exists(annotation => + annotation.domainId == SemanticCoreFeatures.UnifiedTwitterTaxonomy)))), + ("is_random_tweet", _.getOrElse(IsRandomTweetFeature, false)), + ("has_random_tweet_in_response", _.getOrElse(HasRandomTweetFeature, false)), + ("is_random_tweet_above_in_utis", _.getOrElse(IsRandomTweetAboveFeature, false)), + ("is_request_context_launch", _.getOrElse(IsLaunchRequestFeature, false)), + ("viewer_is_employee", _ => false), + ("viewer_is_timelines_employee", _ => false), + ("viewer_follows_any_topics", _.getOrElse(UserFollowedTopicsCountFeature, None).exists(_ > 0)), + ( + "has_ancestor_authored_by_viewer", + candidate => + candidate + .getOrElse(AncestorsFeature, Seq.empty).exists(ancestor => + candidate.getOrElse(ViewerIdFeature, 0L) == ancestor.userId)), + ("ancestor", _.getOrElse(IsAncestorCandidateFeature, false)), + ( + "root_ancestor", + candidate => + candidate.getOrElse(IsAncestorCandidateFeature, false) && candidate + .getOrElse(InReplyToTweetIdFeature, None).isEmpty), + ( + "deep_reply", + candidate => + candidate.getOrElse(InReplyToTweetIdFeature, None).nonEmpty && candidate + .getOrElse(AncestorsFeature, Seq.empty).size > 2), + ( + "has_simcluster_embeddings", + _.getOrElse( + SimclustersTweetTopKClustersWithScoresFeature, + Map.empty[String, Double]).nonEmpty), + ( + "tweet_age_less_than_15_seconds", + _.getOrElse(OriginalTweetCreationTimeFromSnowflakeFeature, None) + .exists(_.untilNow <= 15.seconds)), + ("is_followed_topic_tweet", _ => false), + ("is_recommended_topic_tweet", _ => false), + ("is_topic_tweet", _ => false), + ("preferred_language_matches_tweet_language", _ => false), + ( + "device_language_matches_tweet_language", + candidate => + candidate.getOrElse(TweetLanguageFeature, None) == + candidate.getOrElse(DeviceLanguageFeature, None)), + ("question", _.getOrElse(EarlybirdFeature, None).exists(_.hasQuestion.contains(true))), + ("in_network", _.getOrElse(FromInNetworkSourceFeature, true)), + ("viewer_follows_original_author", _ => false), + ("has_account_follow_prompt", _ => false), + ("has_relevance_prompt", _ => false), + ("has_topic_annotation_haug_prompt", _ => false), + ("has_topic_annotation_random_precision_prompt", _ => false), + ("has_topic_annotation_prompt", _ => false), + ( + "has_political_annotation", + _.getOrElse(EarlybirdFeature, None).exists( + _.semanticCoreAnnotations.exists( + _.exists(annotation => + SemanticCoreFeatures.PoliticalDomains.contains(annotation.domainId) || + (annotation.domainId == SemanticCoreFeatures.UnifiedTwitterTaxonomy && + annotation.entityId == SemanticCoreFeatures.UttPoliticsEntityId))))), + ( + "is_dont_at_me_by_invitation", + _.getOrElse(EarlybirdFeature, None).exists( + _.conversationControl.exists(_.isInstanceOf[tpt.ConversationControl.ByInvitation]))), + ( + "is_dont_at_me_community", + _.getOrElse(EarlybirdFeature, None) + .exists(_.conversationControl.exists(_.isInstanceOf[tpt.ConversationControl.Community]))), + ("has_zero_score", _.getOrElse(ScoreFeature, None).exists(_ == 0.0)), + ("is_viewer_not_invited_to_reply", _ => false), + ("is_viewer_invited_to_reply", _ => false), + ("has_gte_10_favs", _.getOrElse(EarlybirdFeature, None).exists(_.favCountV2.exists(_ >= 10))), + ("has_gte_100_favs", _.getOrElse(EarlybirdFeature, None).exists(_.favCountV2.exists(_ >= 100))), + ("has_gte_1k_favs", _.getOrElse(EarlybirdFeature, None).exists(_.favCountV2.exists(_ >= 1000))), + ( + "has_gte_10k_favs", + _.getOrElse(EarlybirdFeature, None).exists(_.favCountV2.exists(_ >= 1000))), + ( + "has_gte_100k_favs", + _.getOrElse(EarlybirdFeature, None).exists(_.favCountV2.exists(_ >= 100000))), + ("above_neighbor_is_topic_tweet", _ => false), + ("is_topic_tweet_with_neighbor_below", _ => false), + ("has_audio_space", _.getOrElse(AudioSpaceMetaDataFeature, None).exists(_.hasSpace)), + ("has_live_audio_space", _.getOrElse(AudioSpaceMetaDataFeature, None).exists(_.isLive)), + ( + "has_gte_10_retweets", + _.getOrElse(EarlybirdFeature, None).exists(_.retweetCountV2.exists(_ >= 10))), + ( + "has_gte_100_retweets", + _.getOrElse(EarlybirdFeature, None).exists(_.retweetCountV2.exists(_ >= 100))), + ( + "has_gte_1k_retweets", + _.getOrElse(EarlybirdFeature, None).exists(_.retweetCountV2.exists(_ >= 1000))), + ( + "has_us_political_annotation", + _.getOrElse(EarlybirdFeature, None) + .exists(_.semanticCoreAnnotations.exists(_.exists(annotation => + annotation.domainId == SemanticCoreFeatures.UnifiedTwitterTaxonomy && + annotation.entityId == SemanticCoreFeatures.usPoliticalTweetEntityId && + annotation.groupId == SemanticCoreFeatures.UsPoliticalTweetAnnotationGroupIds.BalancedV0)))), + ( + "has_toxicity_score_above_threshold", + _.getOrElse(EarlybirdFeature, None).exists(_.toxicityScore.exists(_ > 0.91))), + ( + "text_only", + candidate => + candidate.getOrElse(HasDisplayedTextFeature, false) && + !(candidate.getOrElse(EarlybirdFeature, None).exists(_.hasImage) || + candidate.getOrElse(EarlybirdFeature, None).exists(_.hasVideo) || + candidate.getOrElse(EarlybirdFeature, None).exists(_.hasCard))), + ( + "image_only", + candidate => + candidate.getOrElse(EarlybirdFeature, None).exists(_.hasImage) && + !candidate.getOrElse(HasDisplayedTextFeature, false)), + ("has_1_image", _.getOrElse(NumImagesFeature, None).exists(_ == 1)), + ("has_2_images", _.getOrElse(NumImagesFeature, None).exists(_ == 2)), + ("has_3_images", _.getOrElse(NumImagesFeature, None).exists(_ == 3)), + ("has_4_images", _.getOrElse(NumImagesFeature, None).exists(_ == 4)), + ("has_card", _.getOrElse(EarlybirdFeature, None).exists(_.hasCard)), + ("3_or_more_consecutive_not_in_network", _ => false), + ("2_or_more_consecutive_not_in_network", _ => false), + ("5_out_of_7_not_in_network", _ => false), + ("7_out_of_7_not_in_network", _ => false), + ("5_out_of_5_not_in_network", _ => false), + ("user_follow_count_gte_50", _.getOrElse(UserFollowingCountFeature, None).exists(_ > 50)), + ("has_liked_by_social_context", _ => false), + ("has_followed_by_social_context", _ => false), + ("has_topic_social_context", _ => false), + ("timeline_entry_has_banner", _ => false), + ("served_in_conversation_module", _.getOrElse(ServedInConversationModuleFeature, false)), + ( + "conversation_module_has_2_displayed_tweets", + _.getOrElse(ConversationModule2DisplayedTweetsFeature, false)), + ("conversation_module_has_gap", _.getOrElse(ConversationModuleHasGapFeature, false)), + ("served_in_recap_tweet_candidate_module_injection", _ => false), + ("served_in_threaded_conversation_module", _ => false), + ( + "author_is_elon", + candidate => + candidate + .getOrElse(AuthorIdFeature, None).contains(candidate.getOrElse(DDGStatsElonFeature, 0L))), + ( + "author_is_power_user", + candidate => + candidate + .getOrElse(AuthorIdFeature, None) + .exists(candidate.getOrElse(DDGStatsVitsFeature, Set.empty[Long]).contains)), + ( + "author_is_democrat", + candidate => + candidate + .getOrElse(AuthorIdFeature, None) + .exists(candidate.getOrElse(DDGStatsDemocratsFeature, Set.empty[Long]).contains)), + ( + "author_is_republican", + candidate => + candidate + .getOrElse(AuthorIdFeature, None) + .exists(candidate.getOrElse(DDGStatsRepublicansFeature, Set.empty[Long]).contains)), + ) + + val PredicateMap = CandidatePredicates.toMap +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/LikedBySocialContextBuilder.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/LikedBySocialContextBuilder.scala new file mode 100644 index 0000000000..e2e6f0a232 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/LikedBySocialContextBuilder.scala @@ -0,0 +1,54 @@ +package com.twitter.home_mixer.functional_component.decorator + +import com.twitter.home_mixer.model.HomeFeatures.PerspectiveFilteredLikedByUserIdsFeature +import com.twitter.home_mixer.model.HomeFeatures.SGSValidLikedByUserIdsFeature +import com.twitter.home_mixer.product.following.model.HomeMixerExternalStrings +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.functional_component.decorator.urt.builder.social_context.BaseSocialContextBuilder +import com.twitter.product_mixer.core.model.marshalling.response.urt.metadata.LikeGeneralContextType +import com.twitter.product_mixer.core.model.marshalling.response.urt.metadata.SocialContext +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.product_mixer.core.product.guice.scope.ProductScoped +import com.twitter.stringcenter.client.StringCenter +import javax.inject.Inject +import javax.inject.Provider +import javax.inject.Singleton + +@Singleton +case class LikedBySocialContextBuilder @Inject() ( + externalStrings: HomeMixerExternalStrings, + @ProductScoped stringCenterProvider: Provider[StringCenter]) + extends BaseSocialContextBuilder[PipelineQuery, TweetCandidate] { + + private val stringCenter = stringCenterProvider.get() + + private val engagerSocialContextBuilder = EngagerSocialContextBuilder( + contextType = LikeGeneralContextType, + stringCenter = stringCenter, + oneUserString = externalStrings.socialContextOneUserLikedString, + twoUsersString = externalStrings.socialContextTwoUsersLikedString, + moreUsersString = externalStrings.socialContextMoreUsersLikedString, + timelineTitle = externalStrings.socialContextLikedByTimelineTitle + ) + + def apply( + query: PipelineQuery, + candidate: TweetCandidate, + candidateFeatures: FeatureMap + ): Option[SocialContext] = { + + // Liked by users are valid only if they pass both the SGS and Perspective filters. + val validLikedByUserIds = + candidateFeatures + .getOrElse(SGSValidLikedByUserIdsFeature, Nil) + .filter( + candidateFeatures.getOrElse(PerspectiveFilteredLikedByUserIdsFeature, Nil).toSet.contains) + + engagerSocialContextBuilder( + socialContextIds = validLikedByUserIds, + query = query, + candidateFeatures = candidateFeatures + ) + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/ListConversationServiceCandidateDecorator.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/ListConversationServiceCandidateDecorator.scala new file mode 100644 index 0000000000..a8df05029a --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/ListConversationServiceCandidateDecorator.scala @@ -0,0 +1,47 @@ +package com.twitter.home_mixer.functional_component.decorator + +import com.twitter.home_mixer.functional_component.decorator.builder.HomeConversationModuleMetadataBuilder +import com.twitter.home_mixer.functional_component.decorator.builder.ListClientEventDetailsBuilder +import com.twitter.home_mixer.model.HomeFeatures.ConversationModuleFocalTweetIdFeature +import com.twitter.product_mixer.component_library.decorator.urt.UrtItemCandidateDecorator +import com.twitter.product_mixer.component_library.decorator.urt.UrtMultipleModulesDecorator +import com.twitter.product_mixer.component_library.decorator.urt.builder.item.tweet.TweetCandidateUrtItemBuilder +import com.twitter.product_mixer.component_library.decorator.urt.builder.metadata.ClientEventInfoBuilder +import com.twitter.product_mixer.component_library.decorator.urt.builder.timeline_module.StaticModuleDisplayTypeBuilder +import com.twitter.product_mixer.component_library.decorator.urt.builder.timeline_module.TimelineModuleBuilder +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.model.marshalling.response.urt.EntryNamespace +import com.twitter.product_mixer.core.model.marshalling.response.urt.timeline_module.VerticalConversation +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.timelines.injection.scribe.InjectionScribeUtil +import com.twitter.timelineservice.suggests.{thriftscala => st} + +object ListConversationServiceCandidateDecorator { + + private val ConversationModuleNamespace = EntryNamespace("list-conversation") + + def apply(): Some[UrtMultipleModulesDecorator[PipelineQuery, TweetCandidate, Long]] = { + val suggestType = st.SuggestType.OrganicListTweet + val component = InjectionScribeUtil.scribeComponent(suggestType).get + val clientEventInfoBuilder = + ClientEventInfoBuilder(component, Some(ListClientEventDetailsBuilder)) + val tweetItemBuilder = TweetCandidateUrtItemBuilder( + clientEventInfoBuilder = clientEventInfoBuilder + ) + + val moduleBuilder = TimelineModuleBuilder( + entryNamespace = ConversationModuleNamespace, + clientEventInfoBuilder = clientEventInfoBuilder, + displayTypeBuilder = StaticModuleDisplayTypeBuilder(VerticalConversation), + metadataBuilder = Some(HomeConversationModuleMetadataBuilder()) + ) + + Some( + UrtMultipleModulesDecorator( + urtItemCandidateDecorator = UrtItemCandidateDecorator(tweetItemBuilder), + moduleBuilder = moduleBuilder, + groupByKey = (_, _, candidateFeatures) => + candidateFeatures.getOrElse(ConversationModuleFocalTweetIdFeature, None) + )) + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/MuteUserChildFeedbackActionBuilder.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/MuteUserChildFeedbackActionBuilder.scala new file mode 100644 index 0000000000..1c8d24ef17 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/MuteUserChildFeedbackActionBuilder.scala @@ -0,0 +1,55 @@ +package com.twitter.home_mixer.functional_component.decorator + +import com.twitter.home_mixer.model.HomeFeatures.AuthorIdFeature +import com.twitter.home_mixer.model.HomeFeatures.IsRetweetFeature +import com.twitter.home_mixer.model.HomeFeatures.ScreenNamesFeature +import com.twitter.home_mixer.model.HomeFeatures.SourceUserIdFeature +import com.twitter.home_mixer.product.following.model.HomeMixerExternalStrings +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.model.marshalling.response.urt.icon +import com.twitter.product_mixer.core.model.marshalling.response.urt.metadata.ChildFeedbackAction +import com.twitter.product_mixer.core.model.marshalling.response.urt.metadata.RichBehavior +import com.twitter.product_mixer.core.model.marshalling.response.urt.metadata.RichFeedbackBehaviorToggleMuteUser +import com.twitter.product_mixer.core.product.guice.scope.ProductScoped +import com.twitter.stringcenter.client.StringCenter + +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +case class MuteUserChildFeedbackActionBuilder @Inject() ( + @ProductScoped stringCenter: StringCenter, + externalStrings: HomeMixerExternalStrings) { + + def apply( + candidateFeatures: FeatureMap + ): Option[ChildFeedbackAction] = { + val userIdOpt = + if (candidateFeatures.getOrElse(IsRetweetFeature, false)) + candidateFeatures.getOrElse(SourceUserIdFeature, None) + else candidateFeatures.getOrElse(AuthorIdFeature, None) + + userIdOpt.flatMap { userId => + val screenNamesMap = candidateFeatures.getOrElse(ScreenNamesFeature, Map.empty[Long, String]) + val userScreenNameOpt = screenNamesMap.get(userId) + userScreenNameOpt.map { userScreenName => + val prompt = stringCenter.prepare( + externalStrings.muteUserString, + Map("username" -> userScreenName) + ) + ChildFeedbackAction( + feedbackType = RichBehavior, + prompt = Some(prompt), + confirmation = None, + feedbackUrl = None, + hasUndoAction = Some(true), + confirmationDisplayType = None, + clientEventInfo = None, + icon = Some(icon.SpeakerOff), + richBehavior = Some(RichFeedbackBehaviorToggleMuteUser(userId)), + subprompt = None + ) + } + } + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/NotInterestedTopicFeedbackActionBuilder.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/NotInterestedTopicFeedbackActionBuilder.scala new file mode 100644 index 0000000000..7c8aebb07b --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/NotInterestedTopicFeedbackActionBuilder.scala @@ -0,0 +1,71 @@ +package com.twitter.home_mixer.functional_component.decorator + +import com.twitter.home_mixer.model.HomeFeatures.InNetworkFeature +import com.twitter.home_mixer.model.HomeFeatures.PerspectiveFilteredLikedByUserIdsFeature +import com.twitter.home_mixer.model.HomeFeatures.SGSValidFollowedByUserIdsFeature +import com.twitter.home_mixer.model.HomeFeatures.SGSValidLikedByUserIdsFeature +import com.twitter.home_mixer.model.HomeFeatures.TopicContextFunctionalityTypeFeature +import com.twitter.home_mixer.model.HomeFeatures.TopicIdSocialContextFeature +import com.twitter.home_mixer.product.following.model.HomeMixerExternalStrings +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.model.marshalling.response.urt.metadata.FeedbackAction +import com.twitter.product_mixer.core.model.marshalling.response.urt.metadata.RecWithEducationTopicContextFunctionalityType +import com.twitter.product_mixer.core.model.marshalling.response.urt.metadata.RecommendationTopicContextFunctionalityType +import com.twitter.product_mixer.core.model.marshalling.response.urt.metadata.RichBehavior +import com.twitter.product_mixer.core.model.marshalling.response.urt.metadata.RichFeedbackBehaviorMarkNotInterestedTopic +import com.twitter.product_mixer.core.product.guice.scope.ProductScoped +import com.twitter.stringcenter.client.StringCenter + +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +case class NotInterestedTopicFeedbackActionBuilder @Inject() ( + @ProductScoped stringCenter: StringCenter, + externalStrings: HomeMixerExternalStrings) { + + def apply( + candidateFeatures: FeatureMap + ): Option[FeedbackAction] = { + val isOutOfNetwork = !candidateFeatures.getOrElse(InNetworkFeature, true) + val validFollowedByUserIds = + candidateFeatures.getOrElse(SGSValidFollowedByUserIdsFeature, Nil) + val validLikedByUserIds = + candidateFeatures + .getOrElse(SGSValidLikedByUserIdsFeature, Nil) + .filter( + candidateFeatures.getOrElse(PerspectiveFilteredLikedByUserIdsFeature, Nil).toSet.contains) + + if (isOutOfNetwork && validLikedByUserIds.isEmpty && validFollowedByUserIds.isEmpty) { + val topicIdSocialContext = candidateFeatures.getOrElse(TopicIdSocialContextFeature, None) + val topicContextFunctionalityType = + candidateFeatures.getOrElse(TopicContextFunctionalityTypeFeature, None) + + (topicIdSocialContext, topicContextFunctionalityType) match { + case (Some(topicId), Some(topicContextFunctionalityType)) + if topicContextFunctionalityType == RecommendationTopicContextFunctionalityType || + topicContextFunctionalityType == RecWithEducationTopicContextFunctionalityType => + Some( + FeedbackAction( + feedbackType = RichBehavior, + prompt = None, + confirmation = None, + childFeedbackActions = None, + feedbackUrl = None, + hasUndoAction = Some(true), + confirmationDisplayType = None, + clientEventInfo = None, + icon = None, + richBehavior = + Some(RichFeedbackBehaviorMarkNotInterestedTopic(topicId = topicId.toString)), + subprompt = None, + encodedFeedbackRequest = None + ) + ) + case _ => None + } + } else { + None + } + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/NotRelevantChildFeedbackActionBuilder.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/NotRelevantChildFeedbackActionBuilder.scala new file mode 100644 index 0000000000..1e5536e0b7 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/NotRelevantChildFeedbackActionBuilder.scala @@ -0,0 +1,55 @@ +package com.twitter.home_mixer.functional_component.decorator + +import com.twitter.home_mixer.model.HomeFeatures.SuggestTypeFeature +import com.twitter.home_mixer.product.following.model.HomeMixerExternalStrings +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.model.marshalling.response.urt.metadata.ChildFeedbackAction +import com.twitter.product_mixer.core.model.marshalling.response.urt.metadata.NotRelevant +import com.twitter.product_mixer.core.product.guice.scope.ProductScoped +import com.twitter.stringcenter.client.StringCenter +import com.twitter.timelines.common.{thriftscala => tlc} +import com.twitter.timelineservice.model.FeedbackInfo +import com.twitter.timelineservice.model.FeedbackMetadata +import com.twitter.timelineservice.{thriftscala => tlst} + +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +case class NotRelevantChildFeedbackActionBuilder @Inject() ( + @ProductScoped stringCenter: StringCenter, + externalStrings: HomeMixerExternalStrings) { + + def apply( + candidate: TweetCandidate, + candidateFeatures: FeatureMap + ): Option[ChildFeedbackAction] = { + val prompt = stringCenter.prepare(externalStrings.notRelevantString) + val confirmation = stringCenter.prepare(externalStrings.notRelevantConfirmationString) + val feedbackMetadata = FeedbackMetadata( + engagementType = None, + entityIds = Seq(tlc.FeedbackEntity.TweetId(candidate.id)), + ttl = Some(FeedbackUtil.FeedbackTtl)) + val feedbackUrl = FeedbackInfo.feedbackUrl( + feedbackType = tlst.FeedbackType.NotRelevant, + feedbackMetadata = feedbackMetadata, + injectionType = candidateFeatures.getOrElse(SuggestTypeFeature, None) + ) + + Some( + ChildFeedbackAction( + feedbackType = NotRelevant, + prompt = Some(prompt), + confirmation = Some(confirmation), + feedbackUrl = Some(feedbackUrl), + hasUndoAction = Some(true), + confirmationDisplayType = None, + clientEventInfo = None, + icon = None, + richBehavior = None, + subprompt = None + ) + ) + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/ReceivedReplySocialContextBuilder.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/ReceivedReplySocialContextBuilder.scala new file mode 100644 index 0000000000..a04e9b4aa6 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/ReceivedReplySocialContextBuilder.scala @@ -0,0 +1,76 @@ +package com.twitter.home_mixer.functional_component.decorator + +import com.twitter.home_mixer.model.HomeFeatures.AuthorIdFeature +import com.twitter.home_mixer.model.HomeFeatures.FocalTweetInNetworkFeature +import com.twitter.home_mixer.model.HomeFeatures.InNetworkFeature +import com.twitter.home_mixer.model.HomeFeatures.RealNamesFeature +import com.twitter.home_mixer.product.following.model.HomeMixerExternalStrings +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.functional_component.decorator.urt.builder.social_context.BaseSocialContextBuilder +import com.twitter.product_mixer.core.model.marshalling.response.urt.metadata.SocialContext +import com.twitter.product_mixer.core.model.marshalling.response.urt.metadata._ +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.product_mixer.core.product.guice.scope.ProductScoped +import com.twitter.stringcenter.client.StringCenter +import javax.inject.Inject +import javax.inject.Provider +import javax.inject.Singleton + +/** + * Use '@A received a reply' as social context when the root Tweet is in network and the focal tweet is OON. + * + * This function should only be called for the root Tweet of convo modules. This is enforced by + * [[HomeTweetSocialContextBuilder]]. + */ +@Singleton +case class ReceivedReplySocialContextBuilder @Inject() ( + externalStrings: HomeMixerExternalStrings, + @ProductScoped stringCenterProvider: Provider[StringCenter]) + extends BaseSocialContextBuilder[PipelineQuery, TweetCandidate] { + + private val stringCenter = stringCenterProvider.get() + private val receivedReplyString = externalStrings.socialContextReceivedReply + + def apply( + query: PipelineQuery, + candidate: TweetCandidate, + candidateFeatures: FeatureMap + ): Option[SocialContext] = { + + // If these values are missing default to not showing a received a reply banner + val inNetwork = candidateFeatures.getOrElse(InNetworkFeature, false) + val inNetworkFocalTweet = + candidateFeatures.getOrElse(FocalTweetInNetworkFeature, None).getOrElse(true) + + if (inNetwork && !inNetworkFocalTweet) { + + val authorIdOpt = candidateFeatures.getOrElse(AuthorIdFeature, None) + val realNames = candidateFeatures.getOrElse(RealNamesFeature, Map.empty[Long, String]) + val authorNameOpt = authorIdOpt.flatMap(realNames.get) + + (authorIdOpt, authorNameOpt) match { + case (Some(authorId), Some(authorName)) => + Some( + GeneralContext( + contextType = ConversationGeneralContextType, + text = stringCenter + .prepare(receivedReplyString, placeholders = Map("user1" -> authorName)), + url = None, + contextImageUrls = None, + landingUrl = Some( + Url( + urlType = DeepLink, + url = "", + urtEndpointOptions = None + ) + ) + ) + ) + case _ => None + } + } else { + None + } + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/ReportTweetChildFeedbackActionBuilder.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/ReportTweetChildFeedbackActionBuilder.scala new file mode 100644 index 0000000000..3938a476d0 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/ReportTweetChildFeedbackActionBuilder.scala @@ -0,0 +1,38 @@ +package com.twitter.home_mixer.functional_component.decorator + +import com.twitter.home_mixer.product.following.model.HomeMixerExternalStrings +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.model.marshalling.response.urt.icon +import com.twitter.product_mixer.core.model.marshalling.response.urt.metadata.ChildFeedbackAction +import com.twitter.product_mixer.core.model.marshalling.response.urt.metadata.RichBehavior +import com.twitter.product_mixer.core.model.marshalling.response.urt.metadata.RichFeedbackBehaviorReportTweet +import com.twitter.product_mixer.core.product.guice.scope.ProductScoped +import com.twitter.stringcenter.client.StringCenter + +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +case class ReportTweetChildFeedbackActionBuilder @Inject() ( + @ProductScoped stringCenter: StringCenter, + externalStrings: HomeMixerExternalStrings) { + + def apply( + candidate: TweetCandidate + ): Option[ChildFeedbackAction] = { + Some( + ChildFeedbackAction( + feedbackType = RichBehavior, + prompt = Some(stringCenter.prepare(externalStrings.reportTweetString)), + confirmation = None, + feedbackUrl = None, + hasUndoAction = Some(true), + confirmationDisplayType = None, + clientEventInfo = None, + icon = Some(icon.Flag), + richBehavior = Some(RichFeedbackBehaviorReportTweet(candidate.id)), + subprompt = None + ) + ) + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/RetweeterChildFeedbackActionBuilder.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/RetweeterChildFeedbackActionBuilder.scala new file mode 100644 index 0000000000..f2ec8d6f56 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/RetweeterChildFeedbackActionBuilder.scala @@ -0,0 +1,39 @@ +package com.twitter.home_mixer.functional_component.decorator + +import com.twitter.home_mixer.model.HomeFeatures.AuthorIdFeature +import com.twitter.home_mixer.model.HomeFeatures.IsRetweetFeature +import com.twitter.home_mixer.model.HomeFeatures.ScreenNamesFeature +import com.twitter.home_mixer.model.HomeFeatures.SuggestTypeFeature +import com.twitter.home_mixer.product.following.model.HomeMixerExternalStrings +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.model.marshalling.response.urt.metadata.ChildFeedbackAction +import com.twitter.product_mixer.core.product.guice.scope.ProductScoped +import com.twitter.stringcenter.client.StringCenter +import com.twitter.timelines.service.{thriftscala => t} + +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +case class RetweeterChildFeedbackActionBuilder @Inject() ( + @ProductScoped stringCenter: StringCenter, + externalStrings: HomeMixerExternalStrings) { + + def apply(candidateFeatures: FeatureMap): Option[ChildFeedbackAction] = { + val isRetweet = candidateFeatures.getOrElse(IsRetweetFeature, false) + + if (isRetweet) { + candidateFeatures.getOrElse(AuthorIdFeature, None).flatMap { retweeterId => + FeedbackUtil.buildUserSeeFewerChildFeedbackAction( + userId = retweeterId, + namesByUserId = candidateFeatures.getOrElse(ScreenNamesFeature, Map.empty[Long, String]), + promptExternalString = externalStrings.showFewerRetweetsString, + confirmationExternalString = externalStrings.showFewerRetweetsConfirmationString, + engagementType = t.FeedbackEngagementType.Retweet, + stringCenter = stringCenter, + injectionType = candidateFeatures.getOrElse(SuggestTypeFeature, None) + ) + } + } else None + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/TopicSocialContextBuilder.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/TopicSocialContextBuilder.scala new file mode 100644 index 0000000000..f9a8c48e6c --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/TopicSocialContextBuilder.scala @@ -0,0 +1,42 @@ +package com.twitter.home_mixer.functional_component.decorator + +import com.twitter.home_mixer.model.HomeFeatures.InNetworkFeature +import com.twitter.home_mixer.model.HomeFeatures.TopicContextFunctionalityTypeFeature +import com.twitter.home_mixer.model.HomeFeatures.TopicIdSocialContextFeature +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.functional_component.decorator.urt.builder.social_context.BaseSocialContextBuilder +import com.twitter.product_mixer.core.model.marshalling.response.urt.metadata.SocialContext +import com.twitter.product_mixer.core.model.marshalling.response.urt.metadata.TopicContext +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +case class TopicSocialContextBuilder @Inject() () + extends BaseSocialContextBuilder[PipelineQuery, TweetCandidate] { + + def apply( + query: PipelineQuery, + candidate: TweetCandidate, + candidateFeatures: FeatureMap + ): Option[SocialContext] = { + val inNetwork = candidateFeatures.getOrElse(InNetworkFeature, true) + if (!inNetwork) { + val topicIdSocialContextOpt = candidateFeatures.getOrElse(TopicIdSocialContextFeature, None) + val topicContextFunctionalityTypeOpt = + candidateFeatures.getOrElse(TopicContextFunctionalityTypeFeature, None) + (topicIdSocialContextOpt, topicContextFunctionalityTypeOpt) match { + case (Some(topicId), Some(topicContextFunctionalityType)) => + Some( + TopicContext( + topicId = topicId.toString, + functionalityType = Some(topicContextFunctionalityType) + )) + case _ => None + } + } else { + None + } + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/UnfollowUserChildFeedbackActionBuilder.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/UnfollowUserChildFeedbackActionBuilder.scala new file mode 100644 index 0000000000..8bb700b440 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/UnfollowUserChildFeedbackActionBuilder.scala @@ -0,0 +1,57 @@ +package com.twitter.home_mixer.functional_component.decorator + +import com.twitter.home_mixer.model.HomeFeatures.AuthorIdFeature +import com.twitter.home_mixer.model.HomeFeatures.InNetworkFeature +import com.twitter.home_mixer.model.HomeFeatures.ScreenNamesFeature +import com.twitter.home_mixer.product.following.model.HomeMixerExternalStrings +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.model.marshalling.response.urt.icon +import com.twitter.product_mixer.core.model.marshalling.response.urt.metadata.ChildFeedbackAction +import com.twitter.product_mixer.core.model.marshalling.response.urt.metadata.RichBehavior +import com.twitter.product_mixer.core.model.marshalling.response.urt.metadata.RichFeedbackBehaviorToggleFollowUser +import com.twitter.product_mixer.core.product.guice.scope.ProductScoped +import com.twitter.stringcenter.client.StringCenter + +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +case class UnfollowUserChildFeedbackActionBuilder @Inject() ( + @ProductScoped stringCenter: StringCenter, + externalStrings: HomeMixerExternalStrings) { + + def apply(candidateFeatures: FeatureMap): Option[ChildFeedbackAction] = { + val isInNetwork = candidateFeatures.getOrElse(InNetworkFeature, false) + val userIdOpt = candidateFeatures.getOrElse(AuthorIdFeature, None) + + if (isInNetwork) { + userIdOpt.flatMap { userId => + val screenNamesMap = + candidateFeatures.getOrElse(ScreenNamesFeature, Map.empty[Long, String]) + val userScreenNameOpt = screenNamesMap.get(userId) + userScreenNameOpt.map { userScreenName => + val prompt = stringCenter.prepare( + externalStrings.unfollowUserString, + Map("username" -> userScreenName) + ) + val confirmation = stringCenter.prepare( + externalStrings.unfollowUserConfirmationString, + Map("username" -> userScreenName) + ) + ChildFeedbackAction( + feedbackType = RichBehavior, + prompt = Some(prompt), + confirmation = Some(confirmation), + feedbackUrl = None, + hasUndoAction = Some(true), + confirmationDisplayType = None, + clientEventInfo = None, + icon = Some(icon.Unfollow), + richBehavior = Some(RichFeedbackBehaviorToggleFollowUser(userId)), + subprompt = None + ) + } + } + } else None + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/YouMightLikeSocialContextBuilder.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/YouMightLikeSocialContextBuilder.scala new file mode 100644 index 0000000000..e131a67f0c --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/YouMightLikeSocialContextBuilder.scala @@ -0,0 +1,50 @@ +package com.twitter.home_mixer.functional_component.decorator + +import com.twitter.home_mixer.model.HomeFeatures.InNetworkFeature +import com.twitter.home_mixer.model.HomeFeatures.IsRetweetFeature +import com.twitter.home_mixer.product.following.model.HomeMixerExternalStrings +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.functional_component.decorator.urt.builder.social_context.BaseSocialContextBuilder +import com.twitter.product_mixer.core.model.marshalling.response.urt.metadata.SocialContext +import com.twitter.product_mixer.core.model.marshalling.response.urt.metadata._ +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.product_mixer.core.product.guice.scope.ProductScoped +import com.twitter.stringcenter.client.StringCenter +import javax.inject.Inject +import javax.inject.Provider +import javax.inject.Singleton + +/** + * Renders a fixed 'You Might Like' string above all OON Tweets. + */ +@Singleton +case class YouMightLikeSocialContextBuilder @Inject() ( + externalStrings: HomeMixerExternalStrings, + @ProductScoped stringCenterProvider: Provider[StringCenter]) + extends BaseSocialContextBuilder[PipelineQuery, TweetCandidate] { + + private val stringCenter = stringCenterProvider.get() + private val youMightLikeString = externalStrings.socialContextYouMightLikeString + + def apply( + query: PipelineQuery, + candidate: TweetCandidate, + candidateFeatures: FeatureMap + ): Option[SocialContext] = { + val isInNetwork = candidateFeatures.getOrElse(InNetworkFeature, true) + val isRetweet = candidateFeatures.getOrElse(IsRetweetFeature, false) + if (!isInNetwork && !isRetweet) { + Some( + GeneralContext( + contextType = SparkleGeneralContextType, + text = stringCenter.prepare(youMightLikeString), + url = None, + contextImageUrls = None, + landingUrl = None + )) + } else { + None + } + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/builder/BUILD.bazel b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/builder/BUILD.bazel new file mode 100644 index 0000000000..66dd7fc383 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/builder/BUILD.bazel @@ -0,0 +1,23 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + strict_deps = True, + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/com/twitter/bijection:scrooge", + "finagle/finagle-core/src/main", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/model", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/model/request", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/product/following/model", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/candidate", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/decorator/urt/builder", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline", + "src/scala/com/twitter/suggests/controller_data", + "src/thrift/com/twitter/suggests/controller_data:controller_data-scala", + "src/thrift/com/twitter/timelinescorer/common/scoredtweetcandidate:thrift-scala", + "src/thrift/com/twitter/timelineservice/server/internal:thrift-scala", + "src/thrift/com/twitter/timelineservice/server/suggests/logging:thrift-scala", + "timelines/src/main/scala/com/twitter/timelines/injection/scribe", + ], +) diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/builder/HomeClientEventInfoBuilder.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/builder/HomeClientEventInfoBuilder.scala new file mode 100644 index 0000000000..f79b0d9319 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/builder/HomeClientEventInfoBuilder.scala @@ -0,0 +1,44 @@ +package com.twitter.home_mixer.functional_component.decorator.builder + +import com.twitter.home_mixer.model.HomeFeatures.EntityTokenFeature +import com.twitter.home_mixer.model.HomeFeatures.SuggestTypeFeature +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.functional_component.decorator.urt.builder.metadata.BaseClientEventDetailsBuilder +import com.twitter.product_mixer.core.functional_component.decorator.urt.builder.metadata.BaseClientEventInfoBuilder +import com.twitter.product_mixer.core.model.common.UniversalNoun +import com.twitter.product_mixer.core.model.marshalling.response.urt.metadata.ClientEventInfo +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.timelines.injection.scribe.InjectionScribeUtil + +/** + * Sets the [[ClientEventInfo]] with the `component` field set to the Suggest Type assigned to each candidate + */ +case class HomeClientEventInfoBuilder[Query <: PipelineQuery, Candidate <: UniversalNoun[Any]]( + detailsBuilder: Option[BaseClientEventDetailsBuilder[Query, Candidate]] = None) + extends BaseClientEventInfoBuilder[Query, Candidate] { + + override def apply( + query: Query, + candidate: Candidate, + candidateFeatures: FeatureMap, + element: Option[String] + ): Option[ClientEventInfo] = { + val suggestType = candidateFeatures + .getOrElse(SuggestTypeFeature, None) + .getOrElse(throw new UnsupportedOperationException(s"No SuggestType was set")) + + Some( + ClientEventInfo( + component = InjectionScribeUtil.scribeComponent(suggestType), + element = element, + details = detailsBuilder.flatMap(_.apply(query, candidate, candidateFeatures)), + action = None, + /** + * A backend entity encoded by the Client Entities Encoding Library. + * Placeholder string for now + */ + entityToken = candidateFeatures.getOrElse(EntityTokenFeature, None) + ) + ) + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/builder/HomeConversationModuleMetadataBuilder.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/builder/HomeConversationModuleMetadataBuilder.scala new file mode 100644 index 0000000000..dc6d513278 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/builder/HomeConversationModuleMetadataBuilder.scala @@ -0,0 +1,30 @@ +package com.twitter.home_mixer.functional_component.decorator.builder + +import com.twitter.home_mixer.model.HomeFeatures.AncestorsFeature +import com.twitter.product_mixer.component_library.model.candidate.BaseTweetCandidate +import com.twitter.product_mixer.core.functional_component.decorator.urt.builder.timeline_module.BaseModuleMetadataBuilder +import com.twitter.product_mixer.core.model.common.CandidateWithFeatures +import com.twitter.product_mixer.core.model.marshalling.response.urt.timeline_module.ModuleConversationMetadata +import com.twitter.product_mixer.core.model.marshalling.response.urt.timeline_module.ModuleMetadata +import com.twitter.product_mixer.core.pipeline.PipelineQuery + +case class HomeConversationModuleMetadataBuilder[ + -Query <: PipelineQuery, + -Candidate <: BaseTweetCandidate +]() extends BaseModuleMetadataBuilder[Query, Candidate] { + + override def apply( + query: Query, + candidates: Seq[CandidateWithFeatures[Candidate]] + ): ModuleMetadata = ModuleMetadata( + adsMetadata = None, + conversationMetadata = Some( + ModuleConversationMetadata( + allTweetIds = Some((candidates.last.candidate.id +: + candidates.last.features.getOrElse(AncestorsFeature, Seq.empty).map(_.tweetId)).reverse), + socialContext = None, + enableDeduplication = Some(true) + )), + gridCarouselMetadata = None + ) +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/builder/ListClientEventDetailsBuilder.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/builder/ListClientEventDetailsBuilder.scala new file mode 100644 index 0000000000..4c7927cab1 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/builder/ListClientEventDetailsBuilder.scala @@ -0,0 +1,33 @@ +package com.twitter.home_mixer.functional_component.decorator.builder + +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.functional_component.decorator.urt.builder.metadata.BaseClientEventDetailsBuilder +import com.twitter.product_mixer.core.model.common.UniversalNoun +import com.twitter.product_mixer.core.model.marshalling.response.urt.metadata.ClientEventDetails +import com.twitter.product_mixer.core.model.marshalling.response.urt.metadata.TimelinesDetails +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.timelineservice.suggests.{thriftscala => st} + +object ListClientEventDetailsBuilder + extends BaseClientEventDetailsBuilder[PipelineQuery, UniversalNoun[Any]] { + + override def apply( + query: PipelineQuery, + candidate: UniversalNoun[Any], + candidateFeatures: FeatureMap + ): Option[ClientEventDetails] = { + val clientEventDetails = ClientEventDetails( + conversationDetails = None, + timelinesDetails = Some( + TimelinesDetails( + injectionType = Some(st.SuggestType.OrganicListTweet.name), + controllerData = None, + sourceData = None)), + articleDetails = None, + liveEventDetails = None, + commerceDetails = None + ) + + Some(clientEventDetails) + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/urt/builder/AddEntriesWithReplaceAndShowAlertAndShowCoverInstructionBuilder.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/urt/builder/AddEntriesWithReplaceAndShowAlertAndShowCoverInstructionBuilder.scala new file mode 100644 index 0000000000..dc3080eb5b --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/urt/builder/AddEntriesWithReplaceAndShowAlertAndShowCoverInstructionBuilder.scala @@ -0,0 +1,30 @@ +package com.twitter.home_mixer.functional_component.decorator.urt.builder + +import com.twitter.product_mixer.component_library.premarshaller.urt.builder.AlwaysInclude +import com.twitter.product_mixer.component_library.premarshaller.urt.builder.IncludeInstruction +import com.twitter.product_mixer.component_library.premarshaller.urt.builder.UrtInstructionBuilder +import com.twitter.product_mixer.core.model.marshalling.response.urt.AddEntriesTimelineInstruction +import com.twitter.product_mixer.core.model.marshalling.response.urt.Cover +import com.twitter.product_mixer.core.model.marshalling.response.urt.ShowAlert +import com.twitter.product_mixer.core.model.marshalling.response.urt.TimelineEntry +import com.twitter.product_mixer.core.pipeline.PipelineQuery + +case class AddEntriesWithReplaceAndShowAlertAndCoverInstructionBuilder[Query <: PipelineQuery]( + override val includeInstruction: IncludeInstruction[Query] = AlwaysInclude) + extends UrtInstructionBuilder[Query, AddEntriesTimelineInstruction] { + + override def build( + query: Query, + entries: Seq[TimelineEntry] + ): Seq[AddEntriesTimelineInstruction] = { + if (includeInstruction(query, entries)) { + val entriesToAdd = entries + .filterNot(_.isInstanceOf[ShowAlert]) + .filterNot(_.isInstanceOf[Cover]) + .filter(_.entryIdToReplace.isEmpty) + if (entriesToAdd.nonEmpty) Seq(AddEntriesTimelineInstruction(entriesToAdd)) + else Seq.empty + } else + Seq.empty + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/urt/builder/BUILD.bazel b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/urt/builder/BUILD.bazel new file mode 100644 index 0000000000..e3714868ad --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/urt/builder/BUILD.bazel @@ -0,0 +1,12 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + strict_deps = True, + tags = ["bazel-compatible"], + dependencies = [ + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/premarshaller/urt/builder", + "src/thrift/com/twitter/timelines/service:thrift-scala", + "src/thrift/com/twitter/timelineservice/server/internal:thrift-scala", + ], +) diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/urt/builder/HomeWhoToFollowFeedbackActionInfoBuilder.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/urt/builder/HomeWhoToFollowFeedbackActionInfoBuilder.scala new file mode 100644 index 0000000000..fd108dc499 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/urt/builder/HomeWhoToFollowFeedbackActionInfoBuilder.scala @@ -0,0 +1,51 @@ +package com.twitter.home_mixer.functional_component.decorator.urt.builder + +import com.twitter.product_mixer.component_library.decorator.urt.builder.metadata.WhoToFollowFeedbackActionInfoBuilder +import com.twitter.product_mixer.component_library.model.candidate.UserCandidate +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.functional_component.decorator.urt.builder.metadata.BaseFeedbackActionInfoBuilder +import com.twitter.product_mixer.core.product.guice.scope.ProductScoped +import com.twitter.stringcenter.client.ExternalStringRegistry +import com.twitter.stringcenter.client.StringCenter +import javax.inject.Inject +import javax.inject.Provider +import javax.inject.Singleton +import com.twitter.product_mixer.core.model.marshalling.response.urt.metadata.FeedbackActionInfo +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.timelines.service.{thriftscala => tl} +import com.twitter.timelines.util.FeedbackRequestSerializer +import com.twitter.timelineservice.suggests.thriftscala.SuggestType +import com.twitter.timelineservice.thriftscala.FeedbackType + +object HomeWhoToFollowFeedbackActionInfoBuilder { + private val FeedbackMetadata = tl.FeedbackMetadata( + injectionType = Some(SuggestType.WhoToFollow), + engagementType = None, + entityIds = Seq.empty, + ttlMs = None + ) + private val FeedbackRequest = + tl.DefaultFeedbackRequest2(FeedbackType.SeeFewer, FeedbackMetadata) + private val EncodedFeedbackRequest = + FeedbackRequestSerializer.serialize(tl.FeedbackRequest.DefaultFeedbackRequest2(FeedbackRequest)) +} + +@Singleton +case class HomeWhoToFollowFeedbackActionInfoBuilder @Inject() ( + @ProductScoped externalStringRegistryProvider: Provider[ExternalStringRegistry], + @ProductScoped stringCenterProvider: Provider[StringCenter]) + extends BaseFeedbackActionInfoBuilder[PipelineQuery, UserCandidate] { + + private val whoToFollowFeedbackActionInfoBuilder = WhoToFollowFeedbackActionInfoBuilder( + externalStringRegistry = externalStringRegistryProvider.get(), + stringCenter = stringCenterProvider.get(), + encodedFeedbackRequest = Some(HomeWhoToFollowFeedbackActionInfoBuilder.EncodedFeedbackRequest) + ) + + override def apply( + query: PipelineQuery, + candidate: UserCandidate, + candidateFeatures: FeatureMap + ): Option[FeedbackActionInfo] = + whoToFollowFeedbackActionInfoBuilder.apply(query, candidate, candidateFeatures) +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/AncestorFeatureHydrator.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/AncestorFeatureHydrator.scala new file mode 100644 index 0000000000..8d1a0862dc --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/AncestorFeatureHydrator.scala @@ -0,0 +1,56 @@ +package com.twitter.home_mixer.functional_component.feature_hydrator + +import com.twitter.home_mixer.model.HomeFeatures.AncestorsFeature +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.feature.Feature +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.feature.featuremap.FeatureMapBuilder +import com.twitter.product_mixer.core.functional_component.feature_hydrator.CandidateFeatureHydrator +import com.twitter.product_mixer.core.model.common.identifier.FeatureHydratorIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.stitch.Stitch +import com.twitter.tweetconvosvc.tweet_ancestor.{thriftscala => ta} +import com.twitter.tweetconvosvc.{thriftscala => tcs} +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class AncestorFeatureHydrator @Inject() ( + conversationServiceClient: tcs.ConversationService.MethodPerEndpoint) + extends CandidateFeatureHydrator[PipelineQuery, TweetCandidate] { + + override val identifier: FeatureHydratorIdentifier = FeatureHydratorIdentifier("Ancestor") + + override val features: Set[Feature[_, _]] = Set(AncestorsFeature) + + override def apply( + query: PipelineQuery, + candidate: TweetCandidate, + existingFeatures: FeatureMap + ): Stitch[FeatureMap] = { + val ancestorsRequest = tcs.GetAncestorsRequest(Seq(candidate.id)) + + Stitch.callFuture(conversationServiceClient.getAncestors(ancestorsRequest)).map { + getAncestorsResponse => + val ancestors = getAncestorsResponse.ancestors.headOption + .collect { + case tcs.TweetAncestorsResult.TweetAncestors(ancestorsResult) + if ancestorsResult.nonEmpty => + ancestorsResult.head.ancestors ++ getTruncatedRootTweet(ancestorsResult.head) + }.getOrElse(Seq.empty) + + FeatureMapBuilder().add(AncestorsFeature, ancestors).build() + } + } + + private def getTruncatedRootTweet( + ancestors: ta.TweetAncestors, + ): Option[ta.TweetAncestor] = { + ancestors.conversationRootAuthorId.collect { + case rootAuthorId + if ancestors.state == ta.ReplyState.Partial && + ancestors.ancestors.last.tweetId != ancestors.conversationId => + ta.TweetAncestor(ancestors.conversationId, rootAuthorId) + } + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/AuthorFeatureHydrator.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/AuthorFeatureHydrator.scala new file mode 100644 index 0000000000..a01269c2ca --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/AuthorFeatureHydrator.scala @@ -0,0 +1,95 @@ +package com.twitter.home_mixer.functional_component.feature_hydrator + +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.home_mixer.functional_component.feature_hydrator.adapters.author_features.AuthorFeaturesAdapter +import com.twitter.home_mixer.model.HomeFeatures.AuthorIdFeature +import com.twitter.home_mixer.param.HomeMixerInjectionNames.AuthorFeatureRepository +import com.twitter.home_mixer.util.ObservedKeyValueResultHandler +import com.twitter.ml.api.DataRecord +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.feature.datarecord.DataRecordInAFeature +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.feature.Feature +import com.twitter.product_mixer.core.feature.FeatureWithDefaultOnFailure +import com.twitter.product_mixer.core.feature.featuremap.FeatureMapBuilder +import com.twitter.product_mixer.core.functional_component.feature_hydrator.BulkCandidateFeatureHydrator +import com.twitter.product_mixer.core.model.common.CandidateWithFeatures +import com.twitter.product_mixer.core.model.common.identifier.FeatureHydratorIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.servo.repository.KeyValueResult +import com.twitter.servo.repository.KeyValueRepository +import com.twitter.stitch.Stitch +import com.twitter.timelines.author_features.v1.{thriftjava => af} +import com.twitter.util.Future +import com.twitter.util.Try + +import javax.inject.Inject +import javax.inject.Named +import javax.inject.Singleton +import scala.collection.JavaConverters._ + +object AuthorFeature + extends DataRecordInAFeature[TweetCandidate] + with FeatureWithDefaultOnFailure[TweetCandidate, DataRecord] { + override def defaultValue: DataRecord = new DataRecord() +} + +@Singleton +class AuthorFeatureHydrator @Inject() ( + @Named(AuthorFeatureRepository) client: KeyValueRepository[Seq[Long], Long, af.AuthorFeatures], + override val statsReceiver: StatsReceiver) + extends BulkCandidateFeatureHydrator[PipelineQuery, TweetCandidate] + with ObservedKeyValueResultHandler { + + override val identifier: FeatureHydratorIdentifier = + FeatureHydratorIdentifier("AuthorFeature") + + override val features: Set[Feature[_, _]] = Set(AuthorFeature) + + override val statScope: String = identifier.toString + + override def apply( + query: PipelineQuery, + candidates: Seq[CandidateWithFeatures[TweetCandidate]] + ): Stitch[Seq[FeatureMap]] = { + Stitch.callFuture { + val possiblyAuthorIds = extractKeys(candidates) + val authorIds = possiblyAuthorIds.flatten + + val response: Future[KeyValueResult[Long, af.AuthorFeatures]] = + if (authorIds.isEmpty) { + Future.value(KeyValueResult.empty) + } else { + client(authorIds) + } + + response.map { result => + possiblyAuthorIds.map { possiblyAuthorId => + val value = observedGet(key = possiblyAuthorId, keyValueResult = result) + val transformedValue = postTransformer(value) + + FeatureMapBuilder() + .add(AuthorFeature, transformedValue) + .build() + } + } + } + } + + private def postTransformer(authorFeatures: Try[Option[af.AuthorFeatures]]): Try[DataRecord] = { + authorFeatures.map { features => + AuthorFeaturesAdapter.adaptToDataRecords(features).asScala.head + } + } + + private def extractKeys( + candidates: Seq[CandidateWithFeatures[TweetCandidate]] + ): Seq[Option[Long]] = { + candidates.map { candidate => + candidate.features + .getTry(AuthorIdFeature) + .toOption + .flatten + } + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/BUILD.bazel b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/BUILD.bazel new file mode 100644 index 0000000000..bc7d5247fb --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/BUILD.bazel @@ -0,0 +1,103 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + strict_deps = True, + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/com/twitter/storehaus:core", + "configapi/configapi-core/src/main/scala/com/twitter/timelines/configapi", + "configapi/configapi-decider", + "finatra/inject/inject-core/src/main/scala", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/adapters/author_features", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/adapters/content", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/adapters/earlybird", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/adapters/inferred_topic", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/adapters/twhin_embeddings", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/marshaller/timelines", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/model", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/model/request", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/param", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/param", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/service", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/util", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/util/earlybird", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/util/tweetypie", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/util/tweetypie/content", + "joinkey/src/main/scala/com/twitter/joinkey/context", + "joinkey/src/main/thrift/com/twitter/joinkey/context:joinkey-context-scala", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/candidate_source/timeline_ranker", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/candidate_source/timelines_impression_store", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/candidate_source/topics", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/feature_hydrator/candidate/tweet_is_nsfw", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/feature_hydrator/candidate/tweet_visibility_reason", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/candidate", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/feature/datarecord", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/feature/featuremap/datarecord", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/candidate_source/strato", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/feature_hydrator", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/util", + "representation-scorer/server/src/main/scala/com/twitter/representationscorer/common", + "representation-scorer/server/src/main/thrift:thrift-scala", + "servo/repo/src/main/scala", + "snowflake/src/main/scala/com/twitter/snowflake/id", + "src/java/com/twitter/ml/api/constant", + "src/java/com/twitter/search/common/util/lang", + "src/scala/com/twitter/ml/api/util", + "src/scala/com/twitter/timelines/prediction/adapters/real_graph", + "src/scala/com/twitter/timelines/prediction/adapters/realtime_interaction_graph", + "src/scala/com/twitter/timelines/prediction/adapters/twistly", + "src/scala/com/twitter/timelines/prediction/adapters/two_hop_features", + "src/scala/com/twitter/timelines/prediction/common/util", + "src/scala/com/twitter/timelines/prediction/features/common", + "src/scala/com/twitter/timelines/prediction/features/realtime_interaction_graph", + "src/scala/com/twitter/timelines/prediction/features/recap", + "src/scala/com/twitter/timelines/prediction/features/time_features", + "src/thrift/com/twitter/gizmoduck:thrift-scala", + "src/thrift/com/twitter/ml/api:data-java", + "src/thrift/com/twitter/ml/api:embedding-java", + "src/thrift/com/twitter/onboarding/relevance/features:features-java", + "src/thrift/com/twitter/recos/user_tweet_entity_graph:user_tweet_entity_graph-scala", + "src/thrift/com/twitter/search:earlybird-scala", + "src/thrift/com/twitter/search/common:constants-java", + "src/thrift/com/twitter/socialgraph:thrift-scala", + "src/thrift/com/twitter/spam/rtf:safety-result-scala", + "src/thrift/com/twitter/timelineranker:thrift-scala", + "src/thrift/com/twitter/timelines/author_features:thrift-java", + "src/thrift/com/twitter/timelines/conversation_features:conversation_features-scala", + "src/thrift/com/twitter/timelines/impression:thrift-scala", + "src/thrift/com/twitter/timelines/impression_bloom_filter:thrift-scala", + "src/thrift/com/twitter/timelines/real_graph:real_graph-scala", + "src/thrift/com/twitter/timelinescorer/common/scoredtweetcandidate:thrift-scala", + "src/thrift/com/twitter/topic_recos:topic_recos-thrift-java", + "src/thrift/com/twitter/tweetypie:service-scala", + "src/thrift/com/twitter/tweetypie:tweet-scala", + "src/thrift/com/twitter/user_session_store:thrift-java", + "src/thrift/com/twitter/wtf/candidate:wtf-candidate-scala", + "src/thrift/com/twitter/wtf/real_time_interaction_graph:wtf-real_time_interaction_graph-thrift-java", + "stitch/stitch-core", + "stitch/stitch-gizmoduck", + "stitch/stitch-socialgraph", + "stitch/stitch-timelineservice", + "stitch/stitch-tweetypie", + "strato/config/columns/topic-signals/tsp", + "strato/config/columns/topic-signals/tsp:tsp-strato-client", + "timelinemixer/common/src/main/scala/com/twitter/timelinemixer/clients/feedback", + "timelinemixer/common/src/main/scala/com/twitter/timelinemixer/clients/manhattan", + "timelinemixer/common/src/main/scala/com/twitter/timelinemixer/clients/persistence", + "timelines/src/main/scala/com/twitter/timelines/clients/strato/twistly", + "timelines/src/main/scala/com/twitter/timelines/clients/user_tweet_entity_graph", + "timelines/src/main/scala/com/twitter/timelines/impressionstore/impressionbloomfilter", + "timelines/src/main/scala/com/twitter/timelines/impressionstore/store", + "timelineservice/common/src/main/scala/com/twitter/timelineservice/model", + "topic-social-proof/server/src/main/thrift:thrift-scala", + "topiclisting/topiclisting-core/src/main/scala/com/twitter/topiclisting", + "tweetconvosvc/thrift/src/main/thrift:thrift-scala", + "twitter-config/yaml", + "user_session_store/src/main/scala/com/twitter/user_session_store", + "util/util-core", + ], + exports = [ + "src/thrift/com/twitter/recos/user_tweet_entity_graph:user_tweet_entity_graph-scala", + "timelinemixer/common/src/main/scala/com/twitter/timelinemixer/clients/manhattan", + ], +) diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/DismissInfoQueryFeatureHydrator.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/DismissInfoQueryFeatureHydrator.scala new file mode 100644 index 0000000000..976cd1e30b --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/DismissInfoQueryFeatureHydrator.scala @@ -0,0 +1,45 @@ +package com.twitter.home_mixer.functional_component.feature_hydrator + +import com.twitter.home_mixer.model.HomeFeatures.DismissInfoFeature +import com.twitter.home_mixer.service.HomeMixerAlertConfig +import com.twitter.timelinemixer.clients.manhattan.InjectionHistoryClient +import com.twitter.product_mixer.core.feature.Feature +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.feature.featuremap.FeatureMapBuilder +import com.twitter.product_mixer.core.functional_component.feature_hydrator.QueryFeatureHydrator +import com.twitter.product_mixer.core.model.common.identifier.FeatureHydratorIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.stitch.Stitch +import com.twitter.timelinemixer.clients.manhattan.DismissInfo +import com.twitter.timelineservice.suggests.thriftscala.SuggestType +import javax.inject.Inject +import javax.inject.Singleton + +object DismissInfoQueryFeatureHydrator { + val DismissInfoSuggestTypes = Seq(SuggestType.WhoToFollow) +} + +@Singleton +case class DismissInfoQueryFeatureHydrator @Inject() ( + dismissInfoClient: InjectionHistoryClient) + extends QueryFeatureHydrator[PipelineQuery] { + + override val identifier: FeatureHydratorIdentifier = FeatureHydratorIdentifier("DismissInfo") + + override val features: Set[Feature[_, _]] = Set(DismissInfoFeature) + + override def hydrate(query: PipelineQuery): Stitch[FeatureMap] = + Stitch.callFuture { + dismissInfoClient + .readDismissInfoEntries( + query.getRequiredUserId, + DismissInfoQueryFeatureHydrator.DismissInfoSuggestTypes).map { response => + val dismissInfoMap = response.mapValues(DismissInfo.fromThrift) + FeatureMapBuilder().add(DismissInfoFeature, dismissInfoMap).build() + } + } + + override val alerts = Seq( + HomeMixerAlertConfig.BusinessHours.defaultSuccessRateAlert(99.8, 50, 60, 60) + ) +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/EarlybirdFeatureHydrator.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/EarlybirdFeatureHydrator.scala new file mode 100644 index 0000000000..639c3ac37b --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/EarlybirdFeatureHydrator.scala @@ -0,0 +1,129 @@ +package com.twitter.home_mixer.functional_component.feature_hydrator + +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.home_mixer.functional_component.feature_hydrator.adapters.earlybird.EarlybirdAdapter +import com.twitter.home_mixer.model.HomeFeatures.DeviceLanguageFeature +import com.twitter.home_mixer.model.HomeFeatures.EarlybirdFeature +import com.twitter.home_mixer.model.HomeFeatures.IsRetweetFeature +import com.twitter.home_mixer.model.HomeFeatures.TweetUrlsFeature +import com.twitter.home_mixer.model.HomeFeatures.UserScreenNameFeature +import com.twitter.home_mixer.param.HomeMixerInjectionNames.EarlybirdRepository +import com.twitter.home_mixer.util.ObservedKeyValueResultHandler +import com.twitter.home_mixer.util.earlybird.EarlybirdResponseUtil +import com.twitter.ml.api.DataRecord +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.feature.Feature +import com.twitter.product_mixer.core.feature.FeatureWithDefaultOnFailure +import com.twitter.product_mixer.core.feature.datarecord.DataRecordInAFeature +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.feature.featuremap.FeatureMapBuilder +import com.twitter.product_mixer.core.functional_component.feature_hydrator.BulkCandidateFeatureHydrator +import com.twitter.product_mixer.core.model.common.CandidateWithFeatures +import com.twitter.product_mixer.core.model.common.identifier.FeatureHydratorIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.search.earlybird.{thriftscala => eb} +import com.twitter.servo.keyvalue.KeyValueResult +import com.twitter.servo.repository.KeyValueRepository +import com.twitter.stitch.Stitch +import com.twitter.util.Return +import javax.inject.Inject +import javax.inject.Named +import javax.inject.Singleton +import scala.collection.JavaConverters._ + +object EarlybirdDataRecordFeature + extends DataRecordInAFeature[TweetCandidate] + with FeatureWithDefaultOnFailure[TweetCandidate, DataRecord] { + override def defaultValue: DataRecord = new DataRecord() +} + +@Singleton +class EarlybirdFeatureHydrator @Inject() ( + @Named(EarlybirdRepository) client: KeyValueRepository[ + (Seq[Long], Long), + Long, + eb.ThriftSearchResult + ], + override val statsReceiver: StatsReceiver) + extends BulkCandidateFeatureHydrator[PipelineQuery, TweetCandidate] + with ObservedKeyValueResultHandler { + + override val identifier: FeatureHydratorIdentifier = FeatureHydratorIdentifier("Earlybird") + + override val features: Set[Feature[_, _]] = + Set(EarlybirdDataRecordFeature, EarlybirdFeature, TweetUrlsFeature) + + override val statScope: String = identifier.toString + + private val scopedStatsReceiver = statsReceiver.scope(statScope) + private val originalKeyFoundCounter = scopedStatsReceiver.counter("originalKey/found") + private val originalKeyLossCounter = scopedStatsReceiver.counter("originalKey/loss") + + private val ebFeaturesNotExistPredicate: CandidateWithFeatures[TweetCandidate] => Boolean = + candidate => candidate.features.getOrElse(EarlybirdFeature, None).isEmpty + + override def apply( + query: PipelineQuery, + candidates: Seq[CandidateWithFeatures[TweetCandidate]] + ): Stitch[Seq[FeatureMap]] = { + val candidatesToHydrate = candidates.filter { candidate => + val isEmpty = ebFeaturesNotExistPredicate(candidate) + if (isEmpty) originalKeyLossCounter.incr() else originalKeyFoundCounter.incr() + isEmpty + } + Stitch + .callFuture(client((candidatesToHydrate.map(_.candidate.id), query.getRequiredUserId))) + .map(handleResponse(query, candidates, _)) + } + + private def handleResponse( + query: PipelineQuery, + candidates: Seq[CandidateWithFeatures[TweetCandidate]], + results: KeyValueResult[Long, eb.ThriftSearchResult] + ): Seq[FeatureMap] = { + val queryFeatureMap = query.features.getOrElse(FeatureMap.empty) + val userLanguages = queryFeatureMap.getOrElse(UserLanguagesFeature, Seq.empty) + val uiLanguageCode = queryFeatureMap.getOrElse(DeviceLanguageFeature, None) + val screenName = queryFeatureMap.getOrElse(UserScreenNameFeature, None) + + val searchResults = candidates + .filter(ebFeaturesNotExistPredicate).map { candidate => + observedGet(Some(candidate.candidate.id), results) + }.collect { + case Return(Some(value)) => value + } + + val tweetIdToEbFeatures = EarlybirdResponseUtil.getOONTweetThriftFeaturesByTweetId( + searcherUserId = query.getRequiredUserId, + screenName = screenName, + userLanguages = userLanguages, + uiLanguageCode = uiLanguageCode, + searchResults = searchResults + ) + + candidates.map { candidate => + val hydratedEbFeatures = tweetIdToEbFeatures.get(candidate.candidate.id) + val earlybirdFeatures = + if (hydratedEbFeatures.nonEmpty) hydratedEbFeatures + else candidate.features.getOrElse(EarlybirdFeature, None) + + val candidateIsRetweet = candidate.features.getOrElse(IsRetweetFeature, false) + val sourceTweetEbFeatures = + candidate.features.getOrElse(SourceTweetEarlybirdFeature, None) + + val originalTweetEbFeatures = + if (candidateIsRetweet && sourceTweetEbFeatures.nonEmpty) + sourceTweetEbFeatures + else earlybirdFeatures + + val earlybirdDataRecord = + EarlybirdAdapter.adaptToDataRecords(originalTweetEbFeatures).asScala.head + + FeatureMapBuilder() + .add(EarlybirdFeature, earlybirdFeatures) + .add(EarlybirdDataRecordFeature, earlybirdDataRecord) + .add(TweetUrlsFeature, earlybirdFeatures.flatMap(_.urlsList).getOrElse(Seq.empty)) + .build() + } + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/FeedbackHistoryQueryFeatureHydrator.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/FeedbackHistoryQueryFeatureHydrator.scala new file mode 100644 index 0000000000..aae96fe003 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/FeedbackHistoryQueryFeatureHydrator.scala @@ -0,0 +1,38 @@ +package com.twitter.home_mixer.functional_component.feature_hydrator + +import com.twitter.home_mixer.model.HomeFeatures.FeedbackHistoryFeature +import com.twitter.home_mixer.param.HomeGlobalParams.EnableFeedbackFatigueParam +import com.twitter.product_mixer.core.feature.Feature +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.feature.featuremap.FeatureMapBuilder +import com.twitter.product_mixer.core.functional_component.feature_hydrator.QueryFeatureHydrator +import com.twitter.product_mixer.core.model.common.Conditionally +import com.twitter.product_mixer.core.model.common.identifier.FeatureHydratorIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.stitch.Stitch +import com.twitter.timelinemixer.clients.feedback.FeedbackHistoryManhattanClient +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +case class FeedbackHistoryQueryFeatureHydrator @Inject() ( + feedbackHistoryClient: FeedbackHistoryManhattanClient) + extends QueryFeatureHydrator[PipelineQuery] + with Conditionally[PipelineQuery] { + + override val identifier: FeatureHydratorIdentifier = FeatureHydratorIdentifier("FeedbackHistory") + + override val features: Set[Feature[_, _]] = Set(FeedbackHistoryFeature) + + override def onlyIf(query: PipelineQuery): Boolean = + query.params(EnableFeedbackFatigueParam) + + override def hydrate( + query: PipelineQuery + ): Stitch[FeatureMap] = + Stitch + .callFuture(feedbackHistoryClient.get(query.getRequiredUserId)) + .map { feedbackHistory => + FeatureMapBuilder().add(FeedbackHistoryFeature, feedbackHistory).build() + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/FocalTweetFeatureHydrator.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/FocalTweetFeatureHydrator.scala new file mode 100644 index 0000000000..79500a7b42 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/FocalTweetFeatureHydrator.scala @@ -0,0 +1,84 @@ +package com.twitter.home_mixer.functional_component.feature_hydrator + +import com.twitter.home_mixer.model.HomeFeatures.AuthorIdFeature +import com.twitter.home_mixer.model.HomeFeatures.ConversationModuleFocalTweetIdFeature +import com.twitter.home_mixer.model.HomeFeatures.ConversationModuleIdFeature +import com.twitter.home_mixer.model.HomeFeatures.FocalTweetAuthorIdFeature +import com.twitter.home_mixer.model.HomeFeatures.FocalTweetInNetworkFeature +import com.twitter.home_mixer.model.HomeFeatures.FocalTweetRealNamesFeature +import com.twitter.home_mixer.model.HomeFeatures.FocalTweetScreenNamesFeature +import com.twitter.home_mixer.model.HomeFeatures.InNetworkFeature +import com.twitter.home_mixer.model.HomeFeatures.RealNamesFeature +import com.twitter.home_mixer.model.HomeFeatures.ScreenNamesFeature +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.feature.Feature +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.feature.featuremap.FeatureMapBuilder +import com.twitter.product_mixer.core.functional_component.feature_hydrator.BulkCandidateFeatureHydrator +import com.twitter.product_mixer.core.model.common.CandidateWithFeatures +import com.twitter.product_mixer.core.model.common.identifier.FeatureHydratorIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.stitch.Stitch +import javax.inject.Inject +import javax.inject.Singleton + +/** + * Social context for convo modules is hydrated on the root Tweet but needs info about the focal + * Tweet (e.g. author) to render the banner. This hydrator copies focal Tweet data into the root. + */ +@Singleton +class FocalTweetFeatureHydrator @Inject() () + extends BulkCandidateFeatureHydrator[PipelineQuery, TweetCandidate] { + + override val identifier: FeatureHydratorIdentifier = FeatureHydratorIdentifier("FocalTweet") + + override val features: Set[Feature[_, _]] = Set( + FocalTweetAuthorIdFeature, + FocalTweetInNetworkFeature, + FocalTweetRealNamesFeature, + FocalTweetScreenNamesFeature + ) + + private val DefaultFeatureMap = FeatureMapBuilder() + .add(FocalTweetAuthorIdFeature, None) + .add(FocalTweetInNetworkFeature, None) + .add(FocalTweetRealNamesFeature, None) + .add(FocalTweetScreenNamesFeature, None) + .build() + + override def apply( + query: PipelineQuery, + candidates: Seq[CandidateWithFeatures[TweetCandidate]] + ): Stitch[Seq[FeatureMap]] = { + // Build a map of all the focal tweets to their corresponding features + val focalTweetIdToFeatureMap = candidates.flatMap { candidate => + val focalTweetId = candidate.features.getOrElse(ConversationModuleFocalTweetIdFeature, None) + if (focalTweetId.contains(candidate.candidate.id)) { + Some(candidate.candidate.id -> candidate.features) + } else None + }.toMap + + val updatedFeatureMap = candidates.map { candidate => + val focalTweetId = candidate.features.getOrElse(ConversationModuleFocalTweetIdFeature, None) + val conversationId = candidate.features.getOrElse(ConversationModuleIdFeature, None) + + // Check if the candidate is a root tweet and ensure its focal tweet's features are available + if (conversationId.contains(candidate.candidate.id) + && focalTweetId.exists(focalTweetIdToFeatureMap.contains)) { + val featureMap = focalTweetIdToFeatureMap.get(focalTweetId.get).get + FeatureMapBuilder() + .add(FocalTweetAuthorIdFeature, featureMap.getOrElse(AuthorIdFeature, None)) + .add(FocalTweetInNetworkFeature, Some(featureMap.getOrElse(InNetworkFeature, true))) + .add( + FocalTweetRealNamesFeature, + Some(featureMap.getOrElse(RealNamesFeature, Map.empty[Long, String]))) + .add( + FocalTweetScreenNamesFeature, + Some(featureMap.getOrElse(ScreenNamesFeature, Map.empty[Long, String]))) + .build() + } else DefaultFeatureMap + } + + Stitch.value(updatedFeatureMap) + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/FollowedTopicsQueryFeatureHydrator.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/FollowedTopicsQueryFeatureHydrator.scala new file mode 100644 index 0000000000..5e9bae5c91 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/FollowedTopicsQueryFeatureHydrator.scala @@ -0,0 +1,41 @@ +package com.twitter.home_mixer.functional_component.feature_hydrator + +import com.twitter.conversions.DurationOps._ +import com.twitter.home_mixer.model.HomeFeatures.UserFollowedTopicsCountFeature +import com.twitter.home_mixer.service.HomeMixerAlertConfig +import com.twitter.product_mixer.component_library.candidate_source.topics.FollowedTopicsCandidateSource +import com.twitter.product_mixer.core.feature.Feature +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.feature.featuremap.FeatureMapBuilder +import com.twitter.product_mixer.core.functional_component.candidate_source.strato.StratoKeyView +import com.twitter.product_mixer.core.functional_component.feature_hydrator.QueryFeatureHydrator +import com.twitter.product_mixer.core.model.common.identifier.FeatureHydratorIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.stitch.Stitch +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +case class FollowedTopicsQueryFeatureHydrator @Inject() ( + followedTopicsCandidateSource: FollowedTopicsCandidateSource) + extends QueryFeatureHydrator[PipelineQuery] { + + override val identifier: FeatureHydratorIdentifier = FeatureHydratorIdentifier("FollowedTopics") + + override val features: Set[Feature[_, _]] = Set(UserFollowedTopicsCountFeature) + + override def hydrate(query: PipelineQuery): Stitch[FeatureMap] = { + val request: StratoKeyView[Long, Unit] = StratoKeyView(query.getRequiredUserId, Unit) + followedTopicsCandidateSource(request) + .map { topics => + FeatureMapBuilder().add(UserFollowedTopicsCountFeature, Some(topics.size)).build() + }.handle { + case _ => FeatureMapBuilder().add(UserFollowedTopicsCountFeature, None).build() + } + } + + override val alerts = Seq( + HomeMixerAlertConfig.BusinessHours.defaultSuccessRateAlert(99.9), + HomeMixerAlertConfig.BusinessHours.defaultLatencyAlert(1500.millis) + ) +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/GizmoduckAuthorSafetyFeatureHydrator.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/GizmoduckAuthorSafetyFeatureHydrator.scala new file mode 100644 index 0000000000..3b13140d18 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/GizmoduckAuthorSafetyFeatureHydrator.scala @@ -0,0 +1,58 @@ +package com.twitter.home_mixer.functional_component.feature_hydrator + +import com.twitter.gizmoduck.{thriftscala => gt} +import com.twitter.home_mixer.model.HomeFeatures.AuthorIdFeature +import com.twitter.home_mixer.model.HomeFeatures.AuthorIsBlueVerifiedFeature +import com.twitter.home_mixer.param.HomeGlobalParams.EnableGizmoduckAuthorSafetyFeatureHydratorParam +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.feature.Feature +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.feature.featuremap.FeatureMapBuilder +import com.twitter.product_mixer.core.functional_component.feature_hydrator.CandidateFeatureHydrator +import com.twitter.product_mixer.core.model.common.Conditionally +import com.twitter.product_mixer.core.model.common.identifier.FeatureHydratorIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.stitch.Stitch +import com.twitter.stitch.gizmoduck.Gizmoduck +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class GizmoduckAuthorSafetyFeatureHydrator @Inject() (gizmoduck: Gizmoduck) + extends CandidateFeatureHydrator[PipelineQuery, TweetCandidate] + with Conditionally[PipelineQuery] { + + override val identifier: FeatureHydratorIdentifier = + FeatureHydratorIdentifier("GizmoduckAuthorSafety") + + override val features: Set[Feature[_, _]] = Set(AuthorIsBlueVerifiedFeature) + + override def onlyIf(query: PipelineQuery): Boolean = + query.params(EnableGizmoduckAuthorSafetyFeatureHydratorParam) + + private val queryFields: Set[gt.QueryFields] = Set(gt.QueryFields.Safety) + + override def apply( + query: PipelineQuery, + candidate: TweetCandidate, + existingFeatures: FeatureMap + ): Stitch[FeatureMap] = { + val authorIdOption = existingFeatures.getOrElse(AuthorIdFeature, None) + + val blueVerifiedStitch = authorIdOption + .map { authorId => + gizmoduck + .getUserById( + userId = authorId, + queryFields = queryFields + ) + .map { _.safety.flatMap(_.isBlueVerified).getOrElse(false) } + }.getOrElse(Stitch.False) + + blueVerifiedStitch.map { isBlueVerified => + FeatureMapBuilder() + .add(AuthorIsBlueVerifiedFeature, isBlueVerified) + .build() + } + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/GizmoduckUserQueryFeatureHydrator.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/GizmoduckUserQueryFeatureHydrator.scala new file mode 100644 index 0000000000..4317119005 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/GizmoduckUserQueryFeatureHydrator.scala @@ -0,0 +1,50 @@ +package com.twitter.home_mixer.functional_component.feature_hydrator + +import com.twitter.gizmoduck.{thriftscala => gt} +import com.twitter.home_mixer.model.HomeFeatures.UserFollowingCountFeature +import com.twitter.home_mixer.model.HomeFeatures.UserScreenNameFeature +import com.twitter.home_mixer.model.HomeFeatures.UserTypeFeature +import com.twitter.home_mixer.service.HomeMixerAlertConfig +import com.twitter.product_mixer.core.feature.Feature +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.feature.featuremap.FeatureMapBuilder +import com.twitter.product_mixer.core.functional_component.feature_hydrator.QueryFeatureHydrator +import com.twitter.product_mixer.core.model.common.identifier.FeatureHydratorIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.stitch.Stitch +import com.twitter.stitch.gizmoduck.Gizmoduck +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +case class GizmoduckUserQueryFeatureHydrator @Inject() (gizmoduck: Gizmoduck) + extends QueryFeatureHydrator[PipelineQuery] { + + override val identifier: FeatureHydratorIdentifier = FeatureHydratorIdentifier("GizmoduckUser") + + override val features: Set[Feature[_, _]] = + Set(UserFollowingCountFeature, UserTypeFeature, UserScreenNameFeature) + + private val queryFields: Set[gt.QueryFields] = + Set(gt.QueryFields.Counts, gt.QueryFields.Safety, gt.QueryFields.Profile) + + override def hydrate(query: PipelineQuery): Stitch[FeatureMap] = { + val userId = query.getRequiredUserId + gizmoduck + .getUserById( + userId = userId, + queryFields = queryFields, + context = gt.LookupContext(forUserId = Some(userId), includeSoftUsers = true)) + .map { user => + FeatureMapBuilder() + .add(UserFollowingCountFeature, user.counts.map(_.following.toInt)) + .add(UserTypeFeature, Some(user.userType)) + .add(UserScreenNameFeature, user.profile.map(_.screenName)) + .build() + } + } + + override val alerts = Seq( + HomeMixerAlertConfig.BusinessHours.defaultSuccessRateAlert(99.7) + ) +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/GraphTwoHopFeatureHydrator.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/GraphTwoHopFeatureHydrator.scala new file mode 100644 index 0000000000..7a3ed69d2a --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/GraphTwoHopFeatureHydrator.scala @@ -0,0 +1,105 @@ +package com.twitter.home_mixer.functional_component.feature_hydrator + +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.graph_feature_service.{thriftscala => gfs} +import com.twitter.home_mixer.model.HomeFeatures.FollowedByUserIdsFeature +import com.twitter.home_mixer.model.HomeFeatures.InNetworkFeature +import com.twitter.home_mixer.model.HomeFeatures.IsExtendedReplyFeature +import com.twitter.home_mixer.model.HomeFeatures.IsRetweetFeature +import com.twitter.home_mixer.param.HomeMixerInjectionNames.GraphTwoHopRepository +import com.twitter.home_mixer.util.CandidatesUtil +import com.twitter.home_mixer.util.ObservedKeyValueResultHandler +import com.twitter.home_mixer.util.ReplyRetweetUtil +import com.twitter.ml.api.DataRecord +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.feature.Feature +import com.twitter.product_mixer.core.feature.FeatureWithDefaultOnFailure +import com.twitter.product_mixer.core.feature.datarecord.DataRecordInAFeature +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.feature.featuremap.FeatureMapBuilder +import com.twitter.product_mixer.core.functional_component.feature_hydrator.BulkCandidateFeatureHydrator +import com.twitter.product_mixer.core.model.common.CandidateWithFeatures +import com.twitter.product_mixer.core.model.common.identifier.FeatureHydratorIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.servo.repository.KeyValueRepository +import com.twitter.stitch.Stitch +import com.twitter.timelines.prediction.adapters.two_hop_features.TwoHopFeaturesAdapter +import com.twitter.util.Try +import javax.inject.Inject +import javax.inject.Named +import javax.inject.Singleton +import scala.collection.JavaConverters._ + +object GraphTwoHopFeature + extends DataRecordInAFeature[TweetCandidate] + with FeatureWithDefaultOnFailure[TweetCandidate, DataRecord] { + override def defaultValue: DataRecord = new DataRecord() +} + +@Singleton +class GraphTwoHopFeatureHydrator @Inject() ( + @Named(GraphTwoHopRepository) client: KeyValueRepository[(Seq[Long], Long), Long, Seq[ + gfs.IntersectionValue + ]], + override val statsReceiver: StatsReceiver) + extends BulkCandidateFeatureHydrator[PipelineQuery, TweetCandidate] + with ObservedKeyValueResultHandler { + + override val identifier: FeatureHydratorIdentifier = FeatureHydratorIdentifier("GraphTwoHop") + + override val features: Set[Feature[_, _]] = Set(GraphTwoHopFeature, FollowedByUserIdsFeature) + + override val statScope: String = identifier.toString + + private val twoHopFeaturesAdapter = new TwoHopFeaturesAdapter + + private val FollowFeatureType = gfs.FeatureType(gfs.EdgeType.Following, gfs.EdgeType.FollowedBy) + + override def apply( + query: PipelineQuery, + candidates: Seq[CandidateWithFeatures[TweetCandidate]] + ): Stitch[Seq[FeatureMap]] = { + // Apply filters to in network candidates for ExtendedReplyAncestors and retweets. + // ExtendedReplyAncestors should also be in candidates. No filter for oon. + val (inNetworkCandidates, oonCandidates) = candidates.partition { candidate => + candidate.features.getOrElse(InNetworkFeature, false) + } + + val inNetworkReplyToAncestorTweet = + ReplyRetweetUtil.replyToAncestorTweetCandidatesMap(inNetworkCandidates) + + val inNetworkExtendedReplyAncestors = inNetworkCandidates + .filter(_.features.getOrElse(IsExtendedReplyFeature, false)).flatMap { inNetworkCandidate => + inNetworkReplyToAncestorTweet.get(inNetworkCandidate.candidate.id) + }.flatten + + val inNetworkCandidatesToHydrate = inNetworkExtendedReplyAncestors ++ + inNetworkCandidates.filter(_.features.getOrElse(IsRetweetFeature, false)) + + val candidatesToHydrate = (inNetworkCandidatesToHydrate ++ oonCandidates) + .flatMap(candidate => CandidatesUtil.getOriginalAuthorId(candidate.features)).distinct + + val response = Stitch.callFuture(client((candidatesToHydrate, query.getRequiredUserId))) + + response.map { result => + candidates.map { candidate => + val originalAuthorId = CandidatesUtil.getOriginalAuthorId(candidate.features) + + val value = observedGet(key = originalAuthorId, keyValueResult = result) + val transformedValue = postTransformer(value) + val followedByUserIds = value.toOption.flatMap(getFollowedByUserIds(_)).getOrElse(Seq.empty) + + FeatureMapBuilder() + .add(GraphTwoHopFeature, transformedValue) + .add(FollowedByUserIdsFeature, followedByUserIds) + .build() + } + } + } + + private def getFollowedByUserIds(input: Option[Seq[gfs.IntersectionValue]]): Option[Seq[Long]] = + input.map(_.filter(_.featureType == FollowFeatureType).flatMap(_.intersectionIds).flatten) + + private def postTransformer(input: Try[Option[Seq[gfs.IntersectionValue]]]): Try[DataRecord] = + input.map(twoHopFeaturesAdapter.adaptToDataRecords(_).asScala.head) +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/ImpressionBloomFilterQueryFeatureHydrator.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/ImpressionBloomFilterQueryFeatureHydrator.scala new file mode 100644 index 0000000000..19b1ab557e --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/ImpressionBloomFilterQueryFeatureHydrator.scala @@ -0,0 +1,57 @@ +package com.twitter.home_mixer.functional_component.feature_hydrator + +import com.twitter.conversions.DurationOps._ +import com.twitter.home_mixer.model.HomeFeatures.ImpressionBloomFilterFeature +import com.twitter.home_mixer.model.request.HasSeenTweetIds +import com.twitter.home_mixer.service.HomeMixerAlertConfig +import com.twitter.product_mixer.core.feature.Feature +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.feature.featuremap.FeatureMapBuilder +import com.twitter.product_mixer.core.functional_component.feature_hydrator.QueryFeatureHydrator +import com.twitter.product_mixer.core.model.common.identifier.FeatureHydratorIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.stitch.Stitch +import com.twitter.timelines.impressionbloomfilter.{thriftscala => t} +import com.twitter.timelines.impressionstore.impressionbloomfilter.ImpressionBloomFilter +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +case class ImpressionBloomFilterQueryFeatureHydrator[ + Query <: PipelineQuery with HasSeenTweetIds] @Inject() ( + bloomFilter: ImpressionBloomFilter) + extends QueryFeatureHydrator[Query] { + + override val identifier: FeatureHydratorIdentifier = FeatureHydratorIdentifier( + "ImpressionBloomFilter") + + private val ImpressionBloomFilterTTL = 7.day + private val ImpressionBloomFilterFalsePositiveRate = 0.002 + + override val features: Set[Feature[_, _]] = Set(ImpressionBloomFilterFeature) + + private val SurfaceArea = t.SurfaceArea.HomeTimeline + + override def hydrate(query: Query): Stitch[FeatureMap] = { + val userId = query.getRequiredUserId + bloomFilter.getBloomFilterSeq(userId, SurfaceArea).map { bloomFilterSeq => + val updatedBloomFilterSeq = + if (query.seenTweetIds.forall(_.isEmpty)) bloomFilterSeq + else { + bloomFilter.addElements( + userId = userId, + surfaceArea = SurfaceArea, + tweetIds = query.seenTweetIds.get, + bloomFilterEntrySeq = bloomFilterSeq, + timeToLive = ImpressionBloomFilterTTL, + falsePositiveRate = ImpressionBloomFilterFalsePositiveRate + ) + } + FeatureMapBuilder().add(ImpressionBloomFilterFeature, updatedBloomFilterSeq).build() + } + } + + override val alerts = Seq( + HomeMixerAlertConfig.BusinessHours.defaultSuccessRateAlert(99.8) + ) +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/LastNonPollingTimeQueryFeatureHydrator.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/LastNonPollingTimeQueryFeatureHydrator.scala new file mode 100644 index 0000000000..9f3049049a --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/LastNonPollingTimeQueryFeatureHydrator.scala @@ -0,0 +1,68 @@ +package com.twitter.home_mixer.functional_component.feature_hydrator + +import com.twitter.home_mixer.model.HomeFeatures.FollowingLastNonPollingTimeFeature +import com.twitter.home_mixer.model.HomeFeatures.LastNonPollingTimeFeature +import com.twitter.home_mixer.model.HomeFeatures.NonPollingTimesFeature +import com.twitter.home_mixer.service.HomeMixerAlertConfig +import com.twitter.product_mixer.core.feature.Feature +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.feature.featuremap.FeatureMapBuilder +import com.twitter.product_mixer.core.functional_component.feature_hydrator.QueryFeatureHydrator +import com.twitter.product_mixer.core.model.common.identifier.FeatureHydratorIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.stitch.Stitch +import com.twitter.user_session_store.ReadRequest +import com.twitter.user_session_store.ReadWriteUserSessionStore +import com.twitter.user_session_store.UserSessionDataset +import com.twitter.user_session_store.UserSessionDataset.UserSessionDataset +import com.twitter.util.Time + +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +case class LastNonPollingTimeQueryFeatureHydrator @Inject() ( + userSessionStore: ReadWriteUserSessionStore) + extends QueryFeatureHydrator[PipelineQuery] { + + override val identifier: FeatureHydratorIdentifier = + FeatureHydratorIdentifier("LastNonPollingTime") + + override val features: Set[Feature[_, _]] = Set( + FollowingLastNonPollingTimeFeature, + LastNonPollingTimeFeature, + NonPollingTimesFeature + ) + + private val datasets: Set[UserSessionDataset] = Set(UserSessionDataset.NonPollingTimes) + + override def hydrate(query: PipelineQuery): Stitch[FeatureMap] = { + userSessionStore + .read(ReadRequest(query.getRequiredUserId, datasets)) + .map { userSession => + val nonPollingTimestamps = userSession.flatMap(_.nonPollingTimestamps) + + val lastNonPollingTime = nonPollingTimestamps + .flatMap(_.nonPollingTimestampsMs.headOption) + .map(Time.fromMilliseconds) + + val followingLastNonPollingTime = nonPollingTimestamps + .flatMap(_.mostRecentHomeLatestNonPollingTimestampMs) + .map(Time.fromMilliseconds) + + val nonPollingTimes = nonPollingTimestamps + .map(_.nonPollingTimestampsMs) + .getOrElse(Seq.empty) + + FeatureMapBuilder() + .add(FollowingLastNonPollingTimeFeature, followingLastNonPollingTime) + .add(LastNonPollingTimeFeature, lastNonPollingTime) + .add(NonPollingTimesFeature, nonPollingTimes) + .build() + } + } + + override val alerts = Seq( + HomeMixerAlertConfig.BusinessHours.defaultSuccessRateAlert(99.9) + ) +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/ListMembersQueryFeatureHydrator.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/ListMembersQueryFeatureHydrator.scala new file mode 100644 index 0000000000..a6d3324d30 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/ListMembersQueryFeatureHydrator.scala @@ -0,0 +1,42 @@ +package com.twitter.home_mixer.functional_component.feature_hydrator + +import com.twitter.home_mixer.model.request.HasListId +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.feature.featuremap.FeatureMapBuilder +import com.twitter.product_mixer.core.feature.Feature +import com.twitter.product_mixer.core.feature.FeatureWithDefaultOnFailure +import com.twitter.product_mixer.core.functional_component.feature_hydrator.QueryFeatureHydrator +import com.twitter.product_mixer.core.model.common.identifier.FeatureHydratorIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.socialgraph.{thriftscala => sg} +import com.twitter.stitch.Stitch +import com.twitter.stitch.socialgraph.SocialGraph + +import javax.inject.Inject +import javax.inject.Singleton + +case object ListMembersFeature extends FeatureWithDefaultOnFailure[PipelineQuery, Seq[Long]] { + override val defaultValue: Seq[Long] = Seq.empty +} + +@Singleton +class ListMembersQueryFeatureHydrator @Inject() (socialGraph: SocialGraph) + extends QueryFeatureHydrator[PipelineQuery with HasListId] { + + override val identifier: FeatureHydratorIdentifier = FeatureHydratorIdentifier("ListMembers") + + override val features: Set[Feature[_, _]] = Set(ListMembersFeature) + + private val MaxRecentMembers = 10 + + override def hydrate(query: PipelineQuery with HasListId): Stitch[FeatureMap] = { + val request = sg.IdsRequest( + relationships = Seq(sg + .SrcRelationship(query.listId, sg.RelationshipType.ListHasMember, hasRelationship = true)), + pageRequest = Some(sg.PageRequest(selectAll = Some(true), count = Some(MaxRecentMembers))) + ) + socialGraph.ids(request).map(_.ids).map { listMembers => + FeatureMapBuilder().add(ListMembersFeature, listMembers).build() + } + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/MetricCenterUserCountingFeatureHydrator.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/MetricCenterUserCountingFeatureHydrator.scala new file mode 100644 index 0000000000..de0b195e04 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/MetricCenterUserCountingFeatureHydrator.scala @@ -0,0 +1,81 @@ +package com.twitter.home_mixer +package functional_component.feature_hydrator + +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.home_mixer.model.HomeFeatures.AuthorIdFeature +import com.twitter.home_mixer.param.HomeMixerInjectionNames.MetricCenterUserCountingFeatureRepository +import com.twitter.home_mixer.util.ObservedKeyValueResultHandler +import com.twitter.onboarding.relevance.features.{thriftjava => rf} +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.feature.Feature +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.feature.featuremap.FeatureMapBuilder +import com.twitter.product_mixer.core.functional_component.feature_hydrator.BulkCandidateFeatureHydrator +import com.twitter.product_mixer.core.model.common.CandidateWithFeatures +import com.twitter.product_mixer.core.model.common.identifier.FeatureHydratorIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.servo.keyvalue.KeyValueResult +import com.twitter.servo.repository.KeyValueRepository +import com.twitter.stitch.Stitch +import com.twitter.util.Future + +import javax.inject.Inject +import javax.inject.Named +import javax.inject.Singleton + +object MetricCenterUserCountingFeature + extends Feature[TweetCandidate, Option[rf.MCUserCountingFeatures]] + +@Singleton +class MetricCenterUserCountingFeatureHydrator @Inject() ( + @Named(MetricCenterUserCountingFeatureRepository) client: KeyValueRepository[Seq[ + Long + ], Long, rf.MCUserCountingFeatures], + override val statsReceiver: StatsReceiver) + extends BulkCandidateFeatureHydrator[PipelineQuery, TweetCandidate] + with ObservedKeyValueResultHandler { + + override val identifier: FeatureHydratorIdentifier = + FeatureHydratorIdentifier("MetricCenterUserCounting") + + override val features: Set[Feature[_, _]] = Set(MetricCenterUserCountingFeature) + + override val statScope: String = identifier.toString + + override def apply( + query: PipelineQuery, + candidates: Seq[CandidateWithFeatures[TweetCandidate]] + ): Stitch[Seq[FeatureMap]] = { + Stitch.callFuture { + val possiblyAuthorIds = extractKeys(candidates) + val userIds = possiblyAuthorIds.flatten + + val response: Future[KeyValueResult[Long, rf.MCUserCountingFeatures]] = if (userIds.isEmpty) { + Future.value(KeyValueResult.empty) + } else { + client(userIds) + } + + response.map { result => + possiblyAuthorIds.map { possiblyAuthorId => + val value = observedGet(key = possiblyAuthorId, keyValueResult = result) + + FeatureMapBuilder() + .add(MetricCenterUserCountingFeature, value) + .build() + } + } + } + } + + private def extractKeys( + candidates: Seq[CandidateWithFeatures[TweetCandidate]] + ): Seq[Option[Long]] = { + candidates.map { candidate => + candidate.features + .getTry(AuthorIdFeature) + .toOption + .flatten + } + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/NamesFeatureHydrator.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/NamesFeatureHydrator.scala new file mode 100644 index 0000000000..ede075e5c5 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/NamesFeatureHydrator.scala @@ -0,0 +1,97 @@ +package com.twitter.home_mixer.functional_component.feature_hydrator + +import com.twitter.gizmoduck.{thriftscala => gt} +import com.twitter.home_mixer.model.HomeFeatures.AuthorIdFeature +import com.twitter.home_mixer.model.HomeFeatures.FavoritedByUserIdsFeature +import com.twitter.home_mixer.model.HomeFeatures.FollowedByUserIdsFeature +import com.twitter.home_mixer.model.HomeFeatures.RealNamesFeature +import com.twitter.home_mixer.model.HomeFeatures.ScreenNamesFeature +import com.twitter.home_mixer.model.HomeFeatures.SourceUserIdFeature +import com.twitter.home_mixer.model.request.FollowingProduct +import com.twitter.home_mixer.param.HomeGlobalParams.EnableNahFeedbackInfoParam +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.feature.Feature +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.feature.featuremap.FeatureMapBuilder +import com.twitter.product_mixer.core.functional_component.feature_hydrator.BulkCandidateFeatureHydrator +import com.twitter.product_mixer.core.model.common.CandidateWithFeatures +import com.twitter.product_mixer.core.model.common.Conditionally +import com.twitter.product_mixer.core.model.common.identifier.FeatureHydratorIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.stitch.Stitch +import com.twitter.stitch.gizmoduck.Gizmoduck +import com.twitter.util.Return +import javax.inject.Inject +import javax.inject.Singleton + +protected case class ProfileNames(screenName: String, realName: String) + +@Singleton +class NamesFeatureHydrator @Inject() (gizmoduck: Gizmoduck) + extends BulkCandidateFeatureHydrator[PipelineQuery, TweetCandidate] + with Conditionally[PipelineQuery] { + + override val identifier: FeatureHydratorIdentifier = FeatureHydratorIdentifier("Names") + + override val features: Set[Feature[_, _]] = Set(ScreenNamesFeature, RealNamesFeature) + + override def onlyIf(query: PipelineQuery): Boolean = query.product match { + case FollowingProduct => query.params(EnableNahFeedbackInfoParam) + case _ => true + } + + private val queryFields: Set[gt.QueryFields] = Set(gt.QueryFields.Profile) + + /** + * The UI currently only ever displays the first 2 names in social context lines + * E.g. "User and 3 others like" or "UserA and UserB liked" + */ + private val MaxCountUsers = 2 + + override def apply( + query: PipelineQuery, + candidates: Seq[CandidateWithFeatures[TweetCandidate]] + ): Stitch[Seq[FeatureMap]] = { + + val candidateUserIdsMap = candidates.map { candidate => + candidate.candidate.id -> + (candidate.features.getOrElse(FavoritedByUserIdsFeature, Nil).take(MaxCountUsers) ++ + candidate.features.getOrElse(FollowedByUserIdsFeature, Nil).take(MaxCountUsers) ++ + candidate.features.getOrElse(AuthorIdFeature, None) ++ + candidate.features.getOrElse(SourceUserIdFeature, None)).distinct + }.toMap + + val distinctUserIds = candidateUserIdsMap.values.flatten.toSeq.distinct + + Stitch + .collectToTry(distinctUserIds.map(userId => gizmoduck.getUserById(userId, queryFields))) + .map { allUsers => + val idToProfileNamesMap = allUsers.flatMap { + case Return(allUser) => + allUser.profile + .map(profile => allUser.id -> ProfileNames(profile.screenName, profile.name)) + case _ => None + }.toMap + + val validUserIds = idToProfileNamesMap.keySet + + candidates.map { candidate => + val combinedMap = candidateUserIdsMap + .getOrElse(candidate.candidate.id, Nil) + .flatMap { + case userId if validUserIds.contains(userId) => + idToProfileNamesMap.get(userId).map(profileNames => userId -> profileNames) + case _ => None + } + + val perCandidateRealNameMap = combinedMap.map { case (k, v) => k -> v.realName }.toMap + val perCandidateScreenNameMap = combinedMap.map { case (k, v) => k -> v.screenName }.toMap + + FeatureMapBuilder() + .add(ScreenNamesFeature, perCandidateScreenNameMap) + .add(RealNamesFeature, perCandidateRealNameMap) + .build() + } + } + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/PersistenceStoreQueryFeatureHydrator.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/PersistenceStoreQueryFeatureHydrator.scala new file mode 100644 index 0000000000..10f42865c3 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/PersistenceStoreQueryFeatureHydrator.scala @@ -0,0 +1,95 @@ +package com.twitter.home_mixer.functional_component.feature_hydrator + +import com.twitter.conversions.DurationOps._ +import com.twitter.common_internal.analytics.twitter_client_user_agent_parser.UserAgent +import com.twitter.home_mixer.model.HomeFeatures.PersistenceEntriesFeature +import com.twitter.home_mixer.model.HomeFeatures.ServedTweetIdsFeature +import com.twitter.home_mixer.model.HomeFeatures.WhoToFollowExcludedUserIdsFeature +import com.twitter.home_mixer.model.request.FollowingProduct +import com.twitter.home_mixer.model.request.ForYouProduct +import com.twitter.home_mixer.service.HomeMixerAlertConfig +import com.twitter.product_mixer.core.feature.Feature +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.feature.featuremap.FeatureMapBuilder +import com.twitter.product_mixer.core.functional_component.feature_hydrator.QueryFeatureHydrator +import com.twitter.product_mixer.core.model.common.identifier.FeatureHydratorIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.stitch.Stitch +import com.twitter.timelinemixer.clients.persistence.TimelineResponseBatchesClient +import com.twitter.timelinemixer.clients.persistence.TimelineResponseV3 +import com.twitter.timelines.util.client_info.ClientPlatform +import com.twitter.timelineservice.model.TimelineQuery +import com.twitter.timelineservice.model.core.TimelineKind +import com.twitter.timelineservice.model.rich.EntityIdType +import com.twitter.util.Time +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +case class PersistenceStoreQueryFeatureHydrator @Inject() ( + timelineResponseBatchesClient: TimelineResponseBatchesClient[TimelineResponseV3]) + extends QueryFeatureHydrator[PipelineQuery] { + + override val identifier: FeatureHydratorIdentifier = FeatureHydratorIdentifier("PersistenceStore") + + private val WhoToFollowExcludedUserIdsLimit = 1000 + private val ServedTweetIdsDuration = 1.hour + private val ServedTweetIdsLimit = 100 + + override val features: Set[Feature[_, _]] = + Set(ServedTweetIdsFeature, PersistenceEntriesFeature, WhoToFollowExcludedUserIdsFeature) + + private val supportedClients = Seq( + ClientPlatform.IPhone, + ClientPlatform.IPad, + ClientPlatform.Mac, + ClientPlatform.Android, + ClientPlatform.Web, + ClientPlatform.RWeb, + ClientPlatform.TweetDeckGryphon + ) + + override def hydrate(query: PipelineQuery): Stitch[FeatureMap] = { + val timelineKind = query.product match { + case FollowingProduct => TimelineKind.homeLatest + case ForYouProduct => TimelineKind.home + case other => throw new UnsupportedOperationException(s"Unknown product: $other") + } + val timelineQuery = TimelineQuery(id = query.getRequiredUserId, kind = timelineKind) + + Stitch.callFuture { + timelineResponseBatchesClient + .get(query = timelineQuery, clientPlatforms = supportedClients) + .map { timelineResponses => + // Note that the WTF entries are not being scoped by ClientPlatform + val whoToFollowUserIds = timelineResponses + .flatMap { timelineResponse => + timelineResponse.entries + .filter(_.entityIdType == EntityIdType.WhoToFollow) + .flatMap(_.itemIds.toSeq.flatMap(_.flatMap(_.userId))) + }.take(WhoToFollowExcludedUserIdsLimit) + + val clientPlatform = ClientPlatform.fromQueryOptions( + clientAppId = query.clientContext.appId, + userAgent = query.clientContext.userAgent.flatMap(UserAgent.fromString)) + + val servedTweetIds = timelineResponses + .filter(_.clientPlatform == clientPlatform) + .filter(_.servedTime >= Time.now - ServedTweetIdsDuration) + .sortBy(-_.servedTime.inMilliseconds) + .flatMap( + _.entries.flatMap(_.tweetIds(includeSourceTweets = true)).take(ServedTweetIdsLimit)) + + FeatureMapBuilder() + .add(ServedTweetIdsFeature, servedTweetIds) + .add(PersistenceEntriesFeature, timelineResponses) + .add(WhoToFollowExcludedUserIdsFeature, whoToFollowUserIds) + .build() + } + } + } + + override val alerts = Seq( + HomeMixerAlertConfig.BusinessHours.defaultSuccessRateAlert(99.7, 50, 60, 60) + ) +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/PerspectiveFilteredSocialContextFeatureHydrator.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/PerspectiveFilteredSocialContextFeatureHydrator.scala new file mode 100644 index 0000000000..1c8dedc652 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/PerspectiveFilteredSocialContextFeatureHydrator.scala @@ -0,0 +1,71 @@ +package com.twitter.home_mixer.functional_component.feature_hydrator + +import com.twitter.home_mixer.model.HomeFeatures.FavoritedByUserIdsFeature +import com.twitter.home_mixer.model.HomeFeatures.PerspectiveFilteredLikedByUserIdsFeature +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.feature.Feature +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.feature.featuremap.FeatureMapBuilder +import com.twitter.product_mixer.core.functional_component.feature_hydrator.BulkCandidateFeatureHydrator +import com.twitter.product_mixer.core.model.common.CandidateWithFeatures +import com.twitter.product_mixer.core.model.common.identifier.FeatureHydratorIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.stitch.Stitch +import com.twitter.stitch.timelineservice.TimelineService +import com.twitter.stitch.timelineservice.TimelineService.GetPerspectives +import com.twitter.timelineservice.thriftscala.PerspectiveType +import com.twitter.timelineservice.thriftscala.PerspectiveType.Favorited +import javax.inject.Inject +import javax.inject.Singleton + +/** + * Filter out unlike edges from liked-by tweets + * Useful if the likes come from a cache and because UTEG does not fully remove unlike edges. + */ +@Singleton +class PerspectiveFilteredSocialContextFeatureHydrator @Inject() (timelineService: TimelineService) + extends BulkCandidateFeatureHydrator[PipelineQuery, TweetCandidate] { + + override val identifier: FeatureHydratorIdentifier = + FeatureHydratorIdentifier("PerspectiveFilteredSocialContext") + + override val features: Set[Feature[_, _]] = Set(PerspectiveFilteredLikedByUserIdsFeature) + + private val MaxCountUsers = 10 + private val favoritePerspectiveSet: Set[PerspectiveType] = Set(Favorited) + + override def apply( + query: PipelineQuery, + candidates: Seq[CandidateWithFeatures[TweetCandidate]] + ): Stitch[Seq[FeatureMap]] = { + val engagingUserIdtoTweetId = candidates.flatMap { candidate => + candidate.features + .get(FavoritedByUserIdsFeature).take(MaxCountUsers) + .map(favoritedBy => favoritedBy -> candidate.candidate.id) + } + + val queries = engagingUserIdtoTweetId.map { + case (userId, tweetId) => + GetPerspectives.Query(userId = userId, tweetId = tweetId, types = favoritePerspectiveSet) + } + + Stitch.collect(queries.map(timelineService.getPerspective)).map { perspectiveResults => + val validUserIdTweetIds: Set[(Long, Long)] = + queries + .zip(perspectiveResults) + .collect { case (query, perspective) if perspective.favorited => query } + .map(query => (query.userId, query.tweetId)) + .toSet + + candidates.map { candidate => + val perspectiveFilteredFavoritedByUserIds: Seq[Long] = candidate.features + .get(FavoritedByUserIdsFeature).take(MaxCountUsers) + .filter { userId => validUserIdTweetIds.contains((userId, candidate.candidate.id)) } + + FeatureMapBuilder() + .add(PerspectiveFilteredLikedByUserIdsFeature, perspectiveFilteredFavoritedByUserIds) + .build() + } + } + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/RealGraphInNetworkScoresQueryFeatureHydrator.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/RealGraphInNetworkScoresQueryFeatureHydrator.scala new file mode 100644 index 0000000000..f044906255 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/RealGraphInNetworkScoresQueryFeatureHydrator.scala @@ -0,0 +1,42 @@ +package com.twitter.home_mixer.functional_component.feature_hydrator + +import com.twitter.home_mixer.model.HomeFeatures.RealGraphInNetworkScoresFeature +import com.twitter.home_mixer.param.HomeMixerInjectionNames.RealGraphInNetworkScores +import com.twitter.product_mixer.core.feature.Feature +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.feature.featuremap.FeatureMapBuilder +import com.twitter.product_mixer.core.functional_component.feature_hydrator.QueryFeatureHydrator +import com.twitter.product_mixer.core.model.common.identifier.FeatureHydratorIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.stitch.Stitch +import com.twitter.storehaus.ReadableStore +import com.twitter.wtf.candidate.{thriftscala => wtf} +import javax.inject.Inject +import javax.inject.Named +import javax.inject.Singleton + +@Singleton +case class RealGraphInNetworkScoresQueryFeatureHydrator @Inject() ( + @Named(RealGraphInNetworkScores) store: ReadableStore[Long, Seq[wtf.Candidate]]) + extends QueryFeatureHydrator[PipelineQuery] { + + override val identifier: FeatureHydratorIdentifier = + FeatureHydratorIdentifier("RealGraphInNetworkScores") + + override val features: Set[Feature[_, _]] = Set(RealGraphInNetworkScoresFeature) + + private val RealGraphCandidateCount = 1000 + + override def hydrate(query: PipelineQuery): Stitch[FeatureMap] = { + Stitch.callFuture(store.get(query.getRequiredUserId)).map { realGraphFollowedUsers => + val realGraphScoresFeatures = realGraphFollowedUsers + .getOrElse(Seq.empty) + .sortBy(-_.score) + .map(candidate => candidate.userId -> candidate.score) + .take(RealGraphCandidateCount) + .toMap + + FeatureMapBuilder().add(RealGraphInNetworkScoresFeature, realGraphScoresFeatures).build() + } + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/RealGraphQueryFeatureHydrator.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/RealGraphQueryFeatureHydrator.scala new file mode 100644 index 0000000000..4eb735714b --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/RealGraphQueryFeatureHydrator.scala @@ -0,0 +1,48 @@ +package com.twitter.home_mixer.functional_component.feature_hydrator + +import com.twitter.home_mixer.param.HomeMixerInjectionNames.RealGraphFeatureRepository +import com.twitter.product_mixer.core.feature.Feature +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.feature.featuremap.FeatureMapBuilder +import com.twitter.product_mixer.core.functional_component.feature_hydrator.QueryFeatureHydrator +import com.twitter.product_mixer.core.model.common.identifier.FeatureHydratorIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.servo.repository.Repository +import com.twitter.timelines.real_graph.{thriftscala => rg} +import com.twitter.stitch.Stitch +import com.twitter.timelines.model.UserId +import com.twitter.timelines.real_graph.v1.thriftscala.RealGraphEdgeFeatures +import com.twitter.user_session_store.{thriftscala => uss} + +import javax.inject.Inject +import javax.inject.Named +import javax.inject.Singleton + +object RealGraphFeatures extends Feature[PipelineQuery, Option[Map[UserId, RealGraphEdgeFeatures]]] + +@Singleton +class RealGraphQueryFeatureHydrator @Inject() ( + @Named(RealGraphFeatureRepository) repository: Repository[Long, Option[uss.UserSession]]) + extends QueryFeatureHydrator[PipelineQuery] { + + override val identifier: FeatureHydratorIdentifier = + FeatureHydratorIdentifier("RealGraphFeatures") + + override val features: Set[Feature[_, _]] = Set(RealGraphFeatures) + + override def hydrate(query: PipelineQuery): Stitch[FeatureMap] = { + Stitch.callFuture { + repository(query.getRequiredUserId).map { userSession => + val realGraphFeaturesMap = userSession.flatMap { userSession => + userSession.realGraphFeatures.collect { + case rg.RealGraphFeatures.V1(realGraphFeatures) => + val edgeFeatures = realGraphFeatures.edgeFeatures ++ realGraphFeatures.oonEdgeFeatures + edgeFeatures.map { edge => edge.destId -> edge }.toMap + } + } + + FeatureMapBuilder().add(RealGraphFeatures, realGraphFeaturesMap).build() + } + } + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/RealGraphViewerAuthorFeatureHydrator.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/RealGraphViewerAuthorFeatureHydrator.scala new file mode 100644 index 0000000000..05b434ccf8 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/RealGraphViewerAuthorFeatureHydrator.scala @@ -0,0 +1,123 @@ +package com.twitter.home_mixer.functional_component.feature_hydrator + +import com.twitter.home_mixer.functional_component.feature_hydrator.RealGraphViewerAuthorFeatureHydrator.getCombinedRealGraphFeatures +import com.twitter.home_mixer.model.HomeFeatures.AuthorIdFeature +import com.twitter.home_mixer.model.HomeFeatures.InReplyToUserIdFeature +import com.twitter.home_mixer.util.MissingKeyException +import com.twitter.ml.api.DataRecord +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.feature.Feature +import com.twitter.product_mixer.core.feature.FeatureWithDefaultOnFailure +import com.twitter.product_mixer.core.feature.datarecord.DataRecordInAFeature +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.feature.featuremap.FeatureMapBuilder +import com.twitter.product_mixer.core.functional_component.feature_hydrator.CandidateFeatureHydrator +import com.twitter.product_mixer.core.model.common.identifier.FeatureHydratorIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.stitch.Stitch +import com.twitter.timelines.prediction.adapters.real_graph.RealGraphEdgeFeaturesCombineAdapter +import com.twitter.timelines.prediction.adapters.real_graph.RealGraphFeaturesAdapter +import com.twitter.timelines.real_graph.v1.{thriftscala => v1} +import com.twitter.timelines.real_graph.{thriftscala => rg} +import com.twitter.util.Throw +import javax.inject.Inject +import javax.inject.Singleton +import scala.collection.JavaConverters._ + +object RealGraphViewerAuthorDataRecordFeature + extends DataRecordInAFeature[TweetCandidate] + with FeatureWithDefaultOnFailure[TweetCandidate, DataRecord] { + override def defaultValue: DataRecord = new DataRecord() +} + +object RealGraphViewerAuthorsDataRecordFeature + extends DataRecordInAFeature[TweetCandidate] + with FeatureWithDefaultOnFailure[TweetCandidate, DataRecord] { + override def defaultValue: DataRecord = new DataRecord() +} + +@Singleton +class RealGraphViewerAuthorFeatureHydrator @Inject() () + extends CandidateFeatureHydrator[PipelineQuery, TweetCandidate] { + + override val identifier: FeatureHydratorIdentifier = + FeatureHydratorIdentifier("RealGraphViewerAuthor") + + override val features: Set[Feature[_, _]] = + Set(RealGraphViewerAuthorDataRecordFeature, RealGraphViewerAuthorsDataRecordFeature) + + private val realGraphEdgeFeaturesAdapter = new RealGraphFeaturesAdapter + private val realGraphEdgeFeaturesCombineAdapter = + new RealGraphEdgeFeaturesCombineAdapter(prefix = "authors.realgraph") + + private val MissingKeyFeatureMap = FeatureMapBuilder() + .add(RealGraphViewerAuthorDataRecordFeature, Throw(MissingKeyException)) + .add(RealGraphViewerAuthorsDataRecordFeature, Throw(MissingKeyException)) + .build() + + override def apply( + query: PipelineQuery, + candidate: TweetCandidate, + existingFeatures: FeatureMap + ): Stitch[FeatureMap] = { + val viewerId = query.getRequiredUserId + val realGraphFeatures = query.features + .flatMap(_.getOrElse(RealGraphFeatures, None)) + .getOrElse(Map.empty[Long, v1.RealGraphEdgeFeatures]) + + val result: FeatureMap = existingFeatures.getOrElse(AuthorIdFeature, None) match { + case Some(authorId) => + val realGraphAuthorFeatures = + getRealGraphViewerAuthorFeatures(viewerId, authorId, realGraphFeatures) + val realGraphAuthorDataRecord = realGraphEdgeFeaturesAdapter + .adaptToDataRecords(realGraphAuthorFeatures).asScala.headOption.getOrElse(new DataRecord) + + val combinedRealGraphFeaturesDataRecord = for { + inReplyToAuthorId <- existingFeatures.getOrElse(InReplyToUserIdFeature, None) + } yield { + val combinedRealGraphFeatures = + getCombinedRealGraphFeatures(Seq(authorId, inReplyToAuthorId), realGraphFeatures) + realGraphEdgeFeaturesCombineAdapter + .adaptToDataRecords(Some(combinedRealGraphFeatures)).asScala.headOption + .getOrElse(new DataRecord) + } + + FeatureMapBuilder() + .add(RealGraphViewerAuthorDataRecordFeature, realGraphAuthorDataRecord) + .add( + RealGraphViewerAuthorsDataRecordFeature, + combinedRealGraphFeaturesDataRecord.getOrElse(new DataRecord)) + .build() + case _ => MissingKeyFeatureMap + } + Stitch(result) + } + + private def getRealGraphViewerAuthorFeatures( + viewerId: Long, + authorId: Long, + realGraphEdgeFeaturesMap: Map[Long, v1.RealGraphEdgeFeatures] + ): rg.UserRealGraphFeatures = { + realGraphEdgeFeaturesMap.get(authorId) match { + case Some(realGraphEdgeFeatures) => + rg.UserRealGraphFeatures( + srcId = viewerId, + features = rg.RealGraphFeatures.V1( + v1.RealGraphFeatures(edgeFeatures = Seq(realGraphEdgeFeatures)))) + case _ => + rg.UserRealGraphFeatures( + srcId = viewerId, + features = rg.RealGraphFeatures.V1(v1.RealGraphFeatures(edgeFeatures = Seq.empty))) + } + } +} + +object RealGraphViewerAuthorFeatureHydrator { + def getCombinedRealGraphFeatures( + userIds: Seq[Long], + realGraphEdgeFeaturesMap: Map[Long, v1.RealGraphEdgeFeatures] + ): rg.RealGraphFeatures = { + val edgeFeatures = userIds.flatMap(realGraphEdgeFeaturesMap.get) + rg.RealGraphFeatures.V1(v1.RealGraphFeatures(edgeFeatures = edgeFeatures)) + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/RealGraphViewerRelatedUsersFeatureHydrator.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/RealGraphViewerRelatedUsersFeatureHydrator.scala new file mode 100644 index 0000000000..45897ec057 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/RealGraphViewerRelatedUsersFeatureHydrator.scala @@ -0,0 +1,74 @@ +package com.twitter.home_mixer.functional_component.feature_hydrator + +import com.twitter.home_mixer.model.HomeFeatures.AuthorIdFeature +import com.twitter.home_mixer.model.HomeFeatures.DirectedAtUserIdFeature +import com.twitter.home_mixer.model.HomeFeatures.MentionUserIdFeature +import com.twitter.home_mixer.model.HomeFeatures.SourceUserIdFeature +import com.twitter.home_mixer.util.CandidatesUtil +import com.twitter.ml.api.DataRecord +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.feature.Feature +import com.twitter.product_mixer.core.feature.FeatureWithDefaultOnFailure +import com.twitter.product_mixer.core.feature.datarecord.DataRecordInAFeature +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.feature.featuremap.FeatureMapBuilder +import com.twitter.product_mixer.core.functional_component.feature_hydrator.CandidateFeatureHydrator +import com.twitter.product_mixer.core.model.common.identifier.FeatureHydratorIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.stitch.Stitch +import com.twitter.timelines.prediction.adapters.real_graph.RealGraphEdgeFeaturesCombineAdapter +import com.twitter.timelines.real_graph.v1.{thriftscala => v1} +import javax.inject.Inject +import javax.inject.Singleton +import scala.collection.JavaConverters._ + +object RealGraphViewerRelatedUsersDataRecordFeature + extends DataRecordInAFeature[TweetCandidate] + with FeatureWithDefaultOnFailure[TweetCandidate, DataRecord] { + override def defaultValue: DataRecord = new DataRecord() +} + +@Singleton +class RealGraphViewerRelatedUsersFeatureHydrator @Inject() () + extends CandidateFeatureHydrator[PipelineQuery, TweetCandidate] { + + override val identifier: FeatureHydratorIdentifier = + FeatureHydratorIdentifier("RealGraphViewerRelatedUsers") + + override val features: Set[Feature[_, _]] = Set(RealGraphViewerRelatedUsersDataRecordFeature) + + private val RealGraphEdgeFeaturesCombineAdapter = new RealGraphEdgeFeaturesCombineAdapter + + override def apply( + query: PipelineQuery, + candidate: TweetCandidate, + existingFeatures: FeatureMap + ): Stitch[FeatureMap] = { + val realGraphQueryFeatures = query.features + .flatMap(_.getOrElse(RealGraphFeatures, None)) + .getOrElse(Map.empty[Long, v1.RealGraphEdgeFeatures]) + + val allRelatedUserIds = getRelatedUserIds(existingFeatures) + val realGraphFeatures = + RealGraphViewerAuthorFeatureHydrator.getCombinedRealGraphFeatures( + allRelatedUserIds, + realGraphQueryFeatures) + val realGraphFeaturesDataRecord = RealGraphEdgeFeaturesCombineAdapter + .adaptToDataRecords(Some(realGraphFeatures)).asScala.headOption + .getOrElse(new DataRecord) + + Stitch.value { + FeatureMapBuilder() + .add(RealGraphViewerRelatedUsersDataRecordFeature, realGraphFeaturesDataRecord) + .build() + } + } + + private def getRelatedUserIds(features: FeatureMap): Seq[Long] = { + (CandidatesUtil.getEngagerUserIds(features) ++ + features.getOrElse(AuthorIdFeature, None) ++ + features.getOrElse(MentionUserIdFeature, Seq.empty) ++ + features.getOrElse(SourceUserIdFeature, None) ++ + features.getOrElse(DirectedAtUserIdFeature, None)).distinct + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/RealTimeInteractionGraphEdgeFeatureHydrator.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/RealTimeInteractionGraphEdgeFeatureHydrator.scala new file mode 100644 index 0000000000..8f64353567 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/RealTimeInteractionGraphEdgeFeatureHydrator.scala @@ -0,0 +1,64 @@ +package com.twitter.home_mixer.functional_component.feature_hydrator + +import com.twitter.home_mixer.model.HomeFeatures.AuthorIdFeature +import com.twitter.ml.api.DataRecord +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.feature.datarecord.DataRecordInAFeature +import com.twitter.product_mixer.core.feature.Feature +import com.twitter.product_mixer.core.feature.FeatureWithDefaultOnFailure +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.feature.featuremap.FeatureMapBuilder +import com.twitter.product_mixer.core.functional_component.feature_hydrator.BulkCandidateFeatureHydrator +import com.twitter.product_mixer.core.model.common.CandidateWithFeatures +import com.twitter.product_mixer.core.model.common.identifier.FeatureHydratorIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.stitch.Stitch +import com.twitter.timelines.prediction.adapters.realtime_interaction_graph.RealTimeInteractionGraphFeaturesAdapter +import com.twitter.timelines.prediction.features.realtime_interaction_graph.RealTimeInteractionGraphEdgeFeatures +import com.twitter.util.Time + +import javax.inject.Inject +import javax.inject.Singleton +import scala.collection.JavaConverters._ + +object RealTimeInteractionGraphEdgeFeature + extends DataRecordInAFeature[TweetCandidate] + with FeatureWithDefaultOnFailure[TweetCandidate, DataRecord] { + override def defaultValue: DataRecord = new DataRecord() +} + +@Singleton +class RealTimeInteractionGraphEdgeFeatureHydrator @Inject() () + extends BulkCandidateFeatureHydrator[PipelineQuery, TweetCandidate] { + + override val identifier: FeatureHydratorIdentifier = FeatureHydratorIdentifier( + "RealTimeInteractionGraphEdge") + + override val features: Set[Feature[_, _]] = Set(RealTimeInteractionGraphEdgeFeature) + + private val realTimeInteractionGraphFeaturesAdapter = new RealTimeInteractionGraphFeaturesAdapter + + override def apply( + query: PipelineQuery, + candidates: Seq[CandidateWithFeatures[TweetCandidate]] + ): Stitch[Seq[FeatureMap]] = { + val userVertex = + query.features.flatMap(_.getOrElse(RealTimeInteractionGraphUserVertexQueryFeature, None)) + val realTimeInteractionGraphFeaturesMap = + userVertex.map(RealTimeInteractionGraphEdgeFeatures(_, Time.now)) + + Stitch.value { + candidates.map { candidate => + val feature = candidate.features.getOrElse(AuthorIdFeature, None).flatMap { authorId => + realTimeInteractionGraphFeaturesMap.flatMap(_.get(authorId)) + } + + FeatureMapBuilder() + .add( + RealTimeInteractionGraphEdgeFeature, + realTimeInteractionGraphFeaturesAdapter.adaptToDataRecords(feature).asScala.head) + .build() + } + } + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/RealTimeInteractionGraphUserVertexQueryFeatureHydrator.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/RealTimeInteractionGraphUserVertexQueryFeatureHydrator.scala new file mode 100644 index 0000000000..899f07630e --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/RealTimeInteractionGraphUserVertexQueryFeatureHydrator.scala @@ -0,0 +1,49 @@ +package com.twitter.home_mixer.functional_component.feature_hydrator + +import com.google.inject.name.Named +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.home_mixer.param.HomeMixerInjectionNames.RealTimeInteractionGraphUserVertexCache +import com.twitter.home_mixer.util.ObservedKeyValueResultHandler +import com.twitter.product_mixer.core.feature.Feature +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.feature.featuremap.FeatureMapBuilder +import com.twitter.product_mixer.core.functional_component.feature_hydrator.QueryFeatureHydrator +import com.twitter.product_mixer.core.model.common.identifier.FeatureHydratorIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.servo.cache.ReadCache +import com.twitter.stitch.Stitch +import com.twitter.wtf.real_time_interaction_graph.{thriftscala => ig} + +import javax.inject.Inject +import javax.inject.Singleton + +object RealTimeInteractionGraphUserVertexQueryFeature + extends Feature[PipelineQuery, Option[ig.UserVertex]] + +@Singleton +class RealTimeInteractionGraphUserVertexQueryFeatureHydrator @Inject() ( + @Named(RealTimeInteractionGraphUserVertexCache) client: ReadCache[Long, ig.UserVertex], + override val statsReceiver: StatsReceiver) + extends QueryFeatureHydrator[PipelineQuery] + with ObservedKeyValueResultHandler { + + override val identifier: FeatureHydratorIdentifier = + FeatureHydratorIdentifier("RealTimeInteractionGraphUserVertex") + + override val features: Set[Feature[_, _]] = Set(RealTimeInteractionGraphUserVertexQueryFeature) + + override val statScope: String = identifier.toString + + override def hydrate(query: PipelineQuery): Stitch[FeatureMap] = { + val userId = query.getRequiredUserId + + Stitch.callFuture( + client.get(Seq(userId)).map { results => + val feature = observedGet(key = Some(userId), keyValueResult = results) + FeatureMapBuilder() + .add(RealTimeInteractionGraphUserVertexQueryFeature, feature) + .build() + } + ) + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/ReplyFeatureHydrator.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/ReplyFeatureHydrator.scala new file mode 100644 index 0000000000..5833838f13 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/ReplyFeatureHydrator.scala @@ -0,0 +1,196 @@ +package com.twitter.home_mixer.functional_component.feature_hydrator + +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.home_mixer.model.HomeFeatures._ +import com.twitter.home_mixer.util.ReplyRetweetUtil +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.feature.Feature +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.feature.featuremap.FeatureMapBuilder +import com.twitter.product_mixer.core.functional_component.feature_hydrator.BulkCandidateFeatureHydrator +import com.twitter.product_mixer.core.model.common.CandidateWithFeatures +import com.twitter.product_mixer.core.model.common.identifier.FeatureHydratorIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.search.common.features.thriftscala.ThriftTweetFeatures +import com.twitter.snowflake.id.SnowflakeId +import com.twitter.stitch.Stitch +import com.twitter.timelines.conversation_features.v1.thriftscala.ConversationFeatures +import com.twitter.util.Duration +import com.twitter.util.Time +import javax.inject.Inject +import javax.inject.Singleton + +object InReplyToTweetHydratedEarlybirdFeature + extends Feature[TweetCandidate, Option[ThriftTweetFeatures]] + +/** + * The purpose of this hydrator is to + * 1) hydrate simple features into replies and their ancestor tweets + * 2) keep both the normal replies and ancestor source candidates, but hydrate into the candidates + * features useful for predicting the quality of the replies and source ancestor tweets. + */ +@Singleton +class ReplyFeatureHydrator @Inject() (statsReceiver: StatsReceiver) + extends BulkCandidateFeatureHydrator[PipelineQuery, TweetCandidate] { + + override val identifier: FeatureHydratorIdentifier = FeatureHydratorIdentifier("ReplyTweet") + + override val features: Set[Feature[_, _]] = Set( + ConversationFeature, + InReplyToTweetHydratedEarlybirdFeature + ) + + private val DefaultFeatureMap = FeatureMapBuilder() + .add(ConversationFeature, None) + .add(InReplyToTweetHydratedEarlybirdFeature, None) + .build() + + private val scopedStatsReceiver = statsReceiver.scope(getClass.getSimpleName) + private val hydratedReplyCounter = scopedStatsReceiver.counter("hydratedReply") + private val hydratedAncestorCounter = scopedStatsReceiver.counter("hydratedAncestor") + + override def apply( + query: PipelineQuery, + candidates: Seq[CandidateWithFeatures[TweetCandidate]] + ): Stitch[Seq[FeatureMap]] = { + val replyToInReplyToTweetMap = + ReplyRetweetUtil.replyTweetIdToInReplyToTweetMap(candidates) + val candidatesWithRepliesHydrated = candidates.map { candidate => + replyToInReplyToTweetMap + .get(candidate.candidate.id).map { inReplyToTweet => + hydratedReplyCounter.incr() + hydratedReplyCandidate(candidate, inReplyToTweet) + }.getOrElse((candidate, None, None)) + } + + /** + * Update ancestor tweets with descendant replies and hydrate simple features from one of + * the descendants. + */ + val ancestorTweetToDescendantRepliesMap = + ReplyRetweetUtil.ancestorTweetIdToDescendantRepliesMap(candidates) + val candidatesWithRepliesAndAncestorTweetsHydrated = candidatesWithRepliesHydrated.map { + case ( + maybeAncestorTweetCandidate, + updatedReplyConversationFeatures, + inReplyToTweetEarlyBirdFeature) => + ancestorTweetToDescendantRepliesMap + .get(maybeAncestorTweetCandidate.candidate.id) + .map { descendantReplies => + hydratedAncestorCounter.incr() + val (ancestorTweetCandidate, updatedConversationFeatures): ( + CandidateWithFeatures[TweetCandidate], + Option[ConversationFeatures] + ) = + hydrateAncestorTweetCandidate( + maybeAncestorTweetCandidate, + descendantReplies, + updatedReplyConversationFeatures) + (ancestorTweetCandidate, inReplyToTweetEarlyBirdFeature, updatedConversationFeatures) + } + .getOrElse( + ( + maybeAncestorTweetCandidate, + inReplyToTweetEarlyBirdFeature, + updatedReplyConversationFeatures)) + } + Stitch.value( + candidatesWithRepliesAndAncestorTweetsHydrated.map { + case (candidate, inReplyToTweetEarlyBirdFeature, updatedConversationFeatures) => + FeatureMapBuilder() + .add(ConversationFeature, updatedConversationFeatures) + .add(InReplyToTweetHydratedEarlybirdFeature, inReplyToTweetEarlyBirdFeature) + .build() + case _ => DefaultFeatureMap + } + ) + } + + private def hydratedReplyCandidate( + replyCandidate: CandidateWithFeatures[TweetCandidate], + inReplyToTweetCandidate: CandidateWithFeatures[TweetCandidate] + ): ( + CandidateWithFeatures[TweetCandidate], + Option[ConversationFeatures], + Option[ThriftTweetFeatures] + ) = { + val tweetedAfterInReplyToTweetInSecs = + ( + originalTweetAgeFromSnowflake(inReplyToTweetCandidate), + originalTweetAgeFromSnowflake(replyCandidate)) match { + case (Some(inReplyToTweetAge), Some(replyTweetAge)) => + Some((inReplyToTweetAge - replyTweetAge).inSeconds.toLong) + case _ => None + } + + val existingConversationFeatures = Some( + replyCandidate.features + .getOrElse(ConversationFeature, None).getOrElse(ConversationFeatures())) + + val updatedConversationFeatures = existingConversationFeatures match { + case Some(v1) => + Some( + v1.copy( + tweetedAfterInReplyToTweetInSecs = tweetedAfterInReplyToTweetInSecs, + isSelfReply = Some( + replyCandidate.features.getOrElse( + AuthorIdFeature, + None) == inReplyToTweetCandidate.features.getOrElse(AuthorIdFeature, None)) + ) + ) + case _ => None + } + + // Note: if inReplyToTweet is a retweet, we need to read early bird feature from the merged + // early bird feature field from RetweetSourceTweetFeatureHydrator class. + // But if inReplyToTweet is a reply, we return its early bird feature directly + val inReplyToTweetThriftTweetFeaturesOpt = { + if (inReplyToTweetCandidate.features.getOrElse(IsRetweetFeature, false)) { + inReplyToTweetCandidate.features.getOrElse(SourceTweetEarlybirdFeature, None) + } else { + inReplyToTweetCandidate.features.getOrElse(EarlybirdFeature, None) + } + } + + (replyCandidate, updatedConversationFeatures, inReplyToTweetThriftTweetFeaturesOpt) + } + + private def hydrateAncestorTweetCandidate( + ancestorTweetCandidate: CandidateWithFeatures[TweetCandidate], + descendantReplies: Seq[CandidateWithFeatures[TweetCandidate]], + updatedReplyConversationFeatures: Option[ConversationFeatures] + ): (CandidateWithFeatures[TweetCandidate], Option[ConversationFeatures]) = { + // Ancestor could be a reply. For example, in thread: tweetA -> tweetB -> tweetC, + // tweetB is a reply and ancestor at the same time. Hence, tweetB's conversation feature + // will be updated by hydratedReplyCandidate and hydrateAncestorTweetCandidate functions. + val existingConversationFeatures = + if (updatedReplyConversationFeatures.nonEmpty) + updatedReplyConversationFeatures + else + Some( + ancestorTweetCandidate.features + .getOrElse(ConversationFeature, None).getOrElse(ConversationFeatures())) + + val updatedConversationFeatures = existingConversationFeatures match { + case Some(v1) => + Some( + v1.copy( + hasDescendantReplyCandidate = Some(true), + hasInNetworkDescendantReply = + Some(descendantReplies.exists(_.features.getOrElse(InNetworkFeature, false))) + )) + case _ => None + } + (ancestorTweetCandidate, updatedConversationFeatures) + } + + private def originalTweetAgeFromSnowflake( + candidate: CandidateWithFeatures[TweetCandidate] + ): Option[Duration] = { + SnowflakeId + .timeFromIdOpt( + candidate.features + .getOrElse(SourceTweetIdFeature, None).getOrElse(candidate.candidate.id)) + .map(Time.now - _) + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/RequestQueryFeatureHydrator.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/RequestQueryFeatureHydrator.scala new file mode 100644 index 0000000000..7381f989b1 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/RequestQueryFeatureHydrator.scala @@ -0,0 +1,128 @@ +package com.twitter.home_mixer.functional_component.feature_hydrator + +import com.twitter.config.yaml.YamlMap +import com.twitter.finagle.tracing.Annotation.BinaryAnnotation +import com.twitter.finagle.tracing.ForwardAnnotation +import com.twitter.home_mixer.model.HomeFeatures._ +import com.twitter.home_mixer.model.request.DeviceContext.RequestContext +import com.twitter.home_mixer.model.request.HasDeviceContext +import com.twitter.home_mixer.param.HomeMixerInjectionNames.DDGStatsAuthors +import com.twitter.joinkey.context.RequestJoinKeyContext +import com.twitter.product_mixer.component_library.model.cursor.UrtOrderedCursor +import com.twitter.product_mixer.core.feature.Feature +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.feature.featuremap.FeatureMapBuilder +import com.twitter.product_mixer.core.functional_component.feature_hydrator.QueryFeatureHydrator +import com.twitter.product_mixer.core.model.common.identifier.FeatureHydratorIdentifier +import com.twitter.product_mixer.core.model.marshalling.response.urt.operation.BottomCursor +import com.twitter.product_mixer.core.model.marshalling.response.urt.operation.GapCursor +import com.twitter.product_mixer.core.model.marshalling.response.urt.operation.TopCursor +import com.twitter.product_mixer.core.pipeline.HasPipelineCursor +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.search.common.util.lang.ThriftLanguageUtil +import com.twitter.snowflake.id.SnowflakeId +import com.twitter.stitch.Stitch +import java.util.UUID +import javax.inject.Inject +import javax.inject.Named +import javax.inject.Singleton + +@Singleton +class RequestQueryFeatureHydrator[ + Query <: PipelineQuery with HasPipelineCursor[UrtOrderedCursor] with HasDeviceContext] @Inject() ( + @Named(DDGStatsAuthors) ddgStatsAuthors: YamlMap) + extends QueryFeatureHydrator[Query] { + + override val features: Set[Feature[_, _]] = Set( + AccountAgeFeature, + ClientIdFeature, + DDGStatsDemocratsFeature, + DDGStatsRepublicansFeature, + DDGStatsElonFeature, + DDGStatsVitsFeature, + DeviceLanguageFeature, + GetInitialFeature, + GetMiddleFeature, + GetNewerFeature, + GetOlderFeature, + GuestIdFeature, + HasDarkRequestFeature, + IsForegroundRequestFeature, + IsLaunchRequestFeature, + PollingFeature, + PullToRefreshFeature, + RequestJoinIdFeature, + ServedRequestIdFeature, + ViewerIdFeature + ) + + override val identifier: FeatureHydratorIdentifier = FeatureHydratorIdentifier("Request") + + private val DarkRequestAnnotation = "clnt/has_dark_request" + private val Democrats = "democrats" + private val Republicans = "republicans" + private val Elon = "elon" + private val Vits = "vits" + + // Convert Language code to ISO 639-3 format + private def getLanguageISOFormatByCode(languageCode: String): String = + ThriftLanguageUtil.getLanguageCodeOf(ThriftLanguageUtil.getThriftLanguageOf(languageCode)) + + private def getRequestJoinId(servedRequestId: Long): Option[Long] = + Some(RequestJoinKeyContext.current.flatMap(_.requestJoinId).getOrElse(servedRequestId)) + + private def hasDarkRequest: Option[Boolean] = ForwardAnnotation.current + .getOrElse(Seq[BinaryAnnotation]()) + .find(_.key == DarkRequestAnnotation) + .map(_.value.asInstanceOf[Boolean]) + + override def hydrate(query: Query): Stitch[FeatureMap] = { + val requestContext = query.deviceContext.flatMap(_.requestContextValue) + val servedRequestId = UUID.randomUUID.getMostSignificantBits + + val featureMap = FeatureMapBuilder() + .add(AccountAgeFeature, query.getOptionalUserId.flatMap(SnowflakeId.timeFromIdOpt)) + .add(ClientIdFeature, query.clientContext.appId) + /** + * These author ID lists are used purely for metrics collection. We track how often we are + * serving Tweets from these authors and how often their tweets are being impressed by users. + * This helps us validate in our A/B experimentation platform that we do not ship changes + * that negatively impacts one group over others. + */ + .add(DDGStatsDemocratsFeature, ddgStatsAuthors.longSeq(Democrats).toSet) + .add(DDGStatsRepublicansFeature, ddgStatsAuthors.longSeq(Republicans).toSet) + .add(DDGStatsVitsFeature, ddgStatsAuthors.longSeq(Vits).toSet) + .add(DDGStatsElonFeature, ddgStatsAuthors.longValue(Elon)) + .add(DeviceLanguageFeature, query.getLanguageCode.map(getLanguageISOFormatByCode)) + .add( + GetInitialFeature, + query.pipelineCursor.forall(cursor => cursor.id.isEmpty && cursor.gapBoundaryId.isEmpty)) + .add( + GetMiddleFeature, + query.pipelineCursor.exists(cursor => + cursor.id.isDefined && cursor.gapBoundaryId.isDefined && + cursor.cursorType.contains(GapCursor))) + .add( + GetNewerFeature, + query.pipelineCursor.exists(cursor => + cursor.id.isDefined && cursor.gapBoundaryId.isEmpty && + cursor.cursorType.contains(TopCursor))) + .add( + GetOlderFeature, + query.pipelineCursor.exists(cursor => + cursor.id.isDefined && cursor.gapBoundaryId.isEmpty && + cursor.cursorType.contains(BottomCursor))) + .add(GuestIdFeature, query.clientContext.guestId) + .add(IsForegroundRequestFeature, requestContext.contains(RequestContext.Foreground)) + .add(IsLaunchRequestFeature, requestContext.contains(RequestContext.Launch)) + .add(PollingFeature, query.deviceContext.exists(_.isPolling.contains(true))) + .add(PullToRefreshFeature, requestContext.contains(RequestContext.PullToRefresh)) + .add(ServedRequestIdFeature, Some(servedRequestId)) + .add(RequestJoinIdFeature, getRequestJoinId(servedRequestId)) + .add(HasDarkRequestFeature, hasDarkRequest) + .add(ViewerIdFeature, query.getRequiredUserId) + .build() + + Stitch.value(featureMap) + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/RetweetSourceTweetFeatureHydrator.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/RetweetSourceTweetFeatureHydrator.scala new file mode 100644 index 0000000000..98385b1306 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/RetweetSourceTweetFeatureHydrator.scala @@ -0,0 +1,76 @@ +package com.twitter.home_mixer.functional_component.feature_hydrator + +import com.twitter.home_mixer.model.HomeFeatures._ +import com.twitter.product_mixer.component_library.candidate_source.timeline_ranker.TimelineRankerInNetworkSourceTweetsByTweetIdMapFeature +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.feature.Feature +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.feature.featuremap.FeatureMapBuilder +import com.twitter.product_mixer.core.functional_component.feature_hydrator.BulkCandidateFeatureHydrator +import com.twitter.product_mixer.core.model.common.CandidateWithFeatures +import com.twitter.product_mixer.core.model.common.identifier.FeatureHydratorIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.search.common.features.thriftscala.ThriftTweetFeatures +import com.twitter.stitch.Stitch +import com.twitter.timelineranker.thriftscala.CandidateTweet + +object SourceTweetEarlybirdFeature extends Feature[TweetCandidate, Option[ThriftTweetFeatures]] + +/** + * Feature Hydrator that bulk hydrates source tweets' features to retweet candidates + */ +object RetweetSourceTweetFeatureHydrator + extends BulkCandidateFeatureHydrator[PipelineQuery, TweetCandidate] { + + override val identifier: FeatureHydratorIdentifier = FeatureHydratorIdentifier( + "RetweetSourceTweet") + + override val features: Set[Feature[_, _]] = Set( + SourceTweetEarlybirdFeature, + ) + + private val DefaultFeatureMap = FeatureMapBuilder() + .add(SourceTweetEarlybirdFeature, None) + .build() + + override def apply( + query: PipelineQuery, + candidates: Seq[CandidateWithFeatures[TweetCandidate]] + ): Stitch[Seq[FeatureMap]] = { + val sourceTweetsByTweetId: Option[Map[Long, CandidateTweet]] = { + query.features.map( + _.getOrElse( + TimelineRankerInNetworkSourceTweetsByTweetIdMapFeature, + Map.empty[Long, CandidateTweet])) + } + + /** + * Return DefaultFeatureMap (no-op to candidate) when it is unfeasible to hydrate the + * source tweet's feature to the current candidate: early bird does not return source + * tweets info / candidate is not a retweet / sourceTweetId is not found + */ + Stitch.value { + if (sourceTweetsByTweetId.exists(_.nonEmpty)) { + candidates.map { candidate => + val candidateIsRetweet = candidate.features.getOrElse(IsRetweetFeature, false) + val sourceTweetId = candidate.features.getOrElse(SourceTweetIdFeature, None) + if (!candidateIsRetweet || sourceTweetId.isEmpty) { + DefaultFeatureMap + } else { + val sourceTweet = sourceTweetsByTweetId.flatMap(_.get(sourceTweetId.get)) + if (sourceTweet.nonEmpty) { + val source = sourceTweet.get + FeatureMapBuilder() + .add(SourceTweetEarlybirdFeature, source.features) + .build() + } else { + DefaultFeatureMap + } + } + } + } else { + candidates.map(_ => DefaultFeatureMap) + } + } + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/SGSFollowedUsersQueryFeatureHydrator.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/SGSFollowedUsersQueryFeatureHydrator.scala new file mode 100644 index 0000000000..e2aa73de8e --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/SGSFollowedUsersQueryFeatureHydrator.scala @@ -0,0 +1,46 @@ +package com.twitter.home_mixer.functional_component.feature_hydrator + +import com.twitter.product_mixer.core.feature.Feature +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.feature.featuremap.FeatureMapBuilder +import com.twitter.product_mixer.core.functional_component.feature_hydrator.QueryFeatureHydrator +import com.twitter.product_mixer.core.model.common.identifier.FeatureHydratorIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.socialgraph.{thriftscala => sg} +import com.twitter.stitch.Stitch +import com.twitter.stitch.socialgraph.{SocialGraph => SocialGraphStitchClient} +import javax.inject.Inject +import javax.inject.Singleton + +object SGSFollowedUsersFeature extends Feature[PipelineQuery, Seq[Long]] + +@Singleton +case class SGSFollowedUsersQueryFeatureHydrator @Inject() ( + socialGraphStitchClient: SocialGraphStitchClient) + extends QueryFeatureHydrator[PipelineQuery] { + + override val identifier: FeatureHydratorIdentifier = + FeatureHydratorIdentifier("SGSFollowedUsers") + + override val features: Set[Feature[_, _]] = Set(SGSFollowedUsersFeature) + + private val SocialGraphLimit = 14999 + + override def hydrate(query: PipelineQuery): Stitch[FeatureMap] = { + val userId = query.getRequiredUserId + + val request = sg.IdsRequest( + relationships = Seq( + sg.SrcRelationship(userId, sg.RelationshipType.Following, hasRelationship = true), + sg.SrcRelationship(userId, sg.RelationshipType.Muting, hasRelationship = false) + ), + pageRequest = Some(sg.PageRequest(count = Some(SocialGraphLimit))) + ) + + socialGraphStitchClient + .ids(request).map(_.ids) + .map { followedUsers => + FeatureMapBuilder().add(SGSFollowedUsersFeature, followedUsers).build() + } + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/SGSValidSocialContextFeatureHydrator.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/SGSValidSocialContextFeatureHydrator.scala new file mode 100644 index 0000000000..1ba706dd18 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/SGSValidSocialContextFeatureHydrator.scala @@ -0,0 +1,105 @@ +package com.twitter.home_mixer.functional_component.feature_hydrator + +import com.twitter.home_mixer.model.HomeFeatures.FavoritedByUserIdsFeature +import com.twitter.home_mixer.model.HomeFeatures.FollowedByUserIdsFeature +import com.twitter.home_mixer.model.HomeFeatures.SGSValidFollowedByUserIdsFeature +import com.twitter.home_mixer.model.HomeFeatures.SGSValidLikedByUserIdsFeature +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.feature.Feature +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.feature.featuremap.FeatureMapBuilder +import com.twitter.product_mixer.core.functional_component.feature_hydrator.BulkCandidateFeatureHydrator +import com.twitter.product_mixer.core.model.common.CandidateWithFeatures +import com.twitter.product_mixer.core.model.common.identifier.FeatureHydratorIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.socialgraph.{thriftscala => sg} +import com.twitter.stitch.Stitch +import com.twitter.stitch.socialgraph.SocialGraph +import javax.inject.Inject +import javax.inject.Singleton + +/** + * This hydrator takes liked-by and followed-by user ids and checks via SGS that the viewer is + * following the engager, that the viewer is not blocking the engager, that the engager is not + * blocking the viewer, and that the viewer has not muted the engager. + */ +@Singleton +class SGSValidSocialContextFeatureHydrator @Inject() ( + socialGraph: SocialGraph) + extends BulkCandidateFeatureHydrator[PipelineQuery, TweetCandidate] { + + override val identifier: FeatureHydratorIdentifier = + FeatureHydratorIdentifier("SGSValidSocialContext") + + override val features: Set[Feature[_, _]] = Set( + SGSValidFollowedByUserIdsFeature, + SGSValidLikedByUserIdsFeature + ) + + private val MaxCountUsers = 10 + + override def apply( + query: PipelineQuery, + candidates: Seq[CandidateWithFeatures[TweetCandidate]] + ): Stitch[Seq[FeatureMap]] = { + + val allSocialContextUserIds = + candidates.flatMap { candidate => + candidate.features.getOrElse(FavoritedByUserIdsFeature, Nil).take(MaxCountUsers) ++ + candidate.features.getOrElse(FollowedByUserIdsFeature, Nil).take(MaxCountUsers) + }.distinct + + getValidUserIds(query.getRequiredUserId, allSocialContextUserIds).map { validUserIds => + candidates.map { candidate => + val sgsFilteredLikedByUserIds = + candidate.features + .getOrElse(FavoritedByUserIdsFeature, Nil).take(MaxCountUsers) + .filter(validUserIds.contains) + + val sgsFilteredFollowedByUserIds = + candidate.features + .getOrElse(FollowedByUserIdsFeature, Nil).take(MaxCountUsers) + .filter(validUserIds.contains) + + FeatureMapBuilder() + .add(SGSValidFollowedByUserIdsFeature, sgsFilteredFollowedByUserIds) + .add(SGSValidLikedByUserIdsFeature, sgsFilteredLikedByUserIds) + .build() + } + } + } + + private def getValidUserIds( + viewerId: Long, + socialProofUserIds: Seq[Long] + ): Stitch[Seq[Long]] = { + if (socialProofUserIds.nonEmpty) { + val request = sg.IdsRequest( + relationships = Seq( + sg.SrcRelationship( + viewerId, + sg.RelationshipType.Following, + targets = Some(socialProofUserIds), + hasRelationship = true), + sg.SrcRelationship( + viewerId, + sg.RelationshipType.Blocking, + targets = Some(socialProofUserIds), + hasRelationship = false), + sg.SrcRelationship( + viewerId, + sg.RelationshipType.BlockedBy, + targets = Some(socialProofUserIds), + hasRelationship = false), + sg.SrcRelationship( + viewerId, + sg.RelationshipType.Muting, + targets = Some(socialProofUserIds), + hasRelationship = false) + ), + pageRequest = Some(sg.PageRequest(selectAll = Some(true))) + ) + socialGraph.ids(request).map(_.ids) + } else Stitch.Nil + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/SimClustersEngagementSimilarityFeatureHydrator.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/SimClustersEngagementSimilarityFeatureHydrator.scala new file mode 100644 index 0000000000..0c04bce5af --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/SimClustersEngagementSimilarityFeatureHydrator.scala @@ -0,0 +1,83 @@ +package com.twitter.home_mixer.functional_component.feature_hydrator + +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.home_mixer.product.scored_tweets.param.ScoredTweetsParam +import com.twitter.ml.api.DataRecord +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.feature.Feature +import com.twitter.product_mixer.core.feature.FeatureWithDefaultOnFailure +import com.twitter.product_mixer.core.feature.datarecord.DataRecordInAFeature +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.feature.featuremap.FeatureMapBuilder +import com.twitter.product_mixer.core.functional_component.feature_hydrator.BulkCandidateFeatureHydrator +import com.twitter.product_mixer.core.model.common.CandidateWithFeatures +import com.twitter.product_mixer.core.model.common.Conditionally +import com.twitter.product_mixer.core.model.common.identifier.FeatureHydratorIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.stitch.Stitch +import com.twitter.timelines.clients.strato.twistly.SimClustersRecentEngagementSimilarityClient +import com.twitter.timelines.configapi.decider.BooleanDeciderParam +import com.twitter.timelines.prediction.adapters.twistly.SimClustersRecentEngagementSimilarityFeaturesAdapter +import javax.inject.Inject +import javax.inject.Singleton + +object SimClustersEngagementSimilarityFeature + extends DataRecordInAFeature[PipelineQuery] + with FeatureWithDefaultOnFailure[PipelineQuery, DataRecord] { + override def defaultValue: DataRecord = new DataRecord() +} + +@Singleton +class SimClustersEngagementSimilarityFeatureHydrator @Inject() ( + simClustersEngagementSimilarityClient: SimClustersRecentEngagementSimilarityClient, + statsReceiver: StatsReceiver) + extends BulkCandidateFeatureHydrator[PipelineQuery, TweetCandidate] + with Conditionally[PipelineQuery] { + + override val identifier: FeatureHydratorIdentifier = + FeatureHydratorIdentifier("SimClustersEngagementSimilarity") + + override val features: Set[Feature[_, _]] = Set(SimClustersEngagementSimilarityFeature) + + private val scopedStatsReceiver = statsReceiver.scope(identifier.toString) + + private val hydratedCandidatesSizeStat = scopedStatsReceiver.stat("hydratedCandidatesSize") + + private val simClustersRecentEngagementSimilarityFeaturesAdapter = + new SimClustersRecentEngagementSimilarityFeaturesAdapter + + override def onlyIf(query: PipelineQuery): Boolean = { + val param: BooleanDeciderParam = + ScoredTweetsParam.EnableSimClustersSimilarityFeatureHydrationDeciderParam + query.params.apply(param) + } + + override def apply( + query: PipelineQuery, + candidates: Seq[CandidateWithFeatures[TweetCandidate]] + ): Stitch[Seq[FeatureMap]] = { + val tweetToCandidates = candidates.map(candidate => candidate.candidate.id -> candidate).toMap + val tweetIds = tweetToCandidates.keySet.toSeq + val userId = query.getRequiredUserId + val userTweetEdges = tweetIds.map(tweetId => (userId, tweetId)) + val resultFuture = simClustersEngagementSimilarityClient + .getSimClustersRecentEngagementSimilarityScores(userTweetEdges).map { + simClustersRecentEngagementSimilarityScoresMap => + hydratedCandidatesSizeStat.add(simClustersRecentEngagementSimilarityScoresMap.size) + candidates.map { candidate => + val similarityFeatureOpt = simClustersRecentEngagementSimilarityScoresMap + .get(userId -> candidate.candidate.id).flatten + val dataRecordOpt = similarityFeatureOpt.map { similarityFeature => + simClustersRecentEngagementSimilarityFeaturesAdapter + .adaptToDataRecords(similarityFeature) + .get(0) + } + FeatureMapBuilder() + .add(SimClustersEngagementSimilarityFeature, dataRecordOpt.getOrElse(new DataRecord)) + .build() + } + } + Stitch.callFuture(resultFuture) + } + +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/SocialGraphServiceFeatureHydrator.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/SocialGraphServiceFeatureHydrator.scala new file mode 100644 index 0000000000..729a40e94c --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/SocialGraphServiceFeatureHydrator.scala @@ -0,0 +1,67 @@ +package com.twitter.home_mixer.functional_component.feature_hydrator + +import com.twitter.home_mixer.model.HomeFeatures.AuthorIdFeature +import com.twitter.home_mixer.model.HomeFeatures.InNetworkFeature +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.feature.Feature +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.feature.featuremap.FeatureMapBuilder +import com.twitter.product_mixer.core.functional_component.feature_hydrator.BulkCandidateFeatureHydrator +import com.twitter.product_mixer.core.model.common.CandidateWithFeatures +import com.twitter.product_mixer.core.model.common.identifier.FeatureHydratorIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.socialgraph.{thriftscala => sg} +import com.twitter.stitch.Stitch +import com.twitter.stitch.socialgraph.{SocialGraph => SocialGraphStitchClient} +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class SocialGraphServiceFeatureHydrator @Inject() (socialGraphStitchClient: SocialGraphStitchClient) + extends BulkCandidateFeatureHydrator[PipelineQuery, TweetCandidate] { + + override val identifier: FeatureHydratorIdentifier = + FeatureHydratorIdentifier("SocialGraphService") + + override val features: Set[Feature[_, _]] = Set(InNetworkFeature) + + override def apply( + query: PipelineQuery, + candidates: Seq[CandidateWithFeatures[TweetCandidate]] + ): Stitch[Seq[FeatureMap]] = { + val viewerId = query.getRequiredUserId + + // We use authorId and not sourceAuthorId here so that retweets are defined as in network + val authorIds = candidates.map(_.features.getOrElse(AuthorIdFeature, None).getOrElse(0L)) + val distinctNonSelfAuthorIds = authorIds.filter(_ != viewerId).distinct + + val idsRequest = createIdsRequest( + userId = viewerId, + relationshipTypes = Set(sg.RelationshipType.Following), + targetIds = Some(distinctNonSelfAuthorIds) + ) + + socialGraphStitchClient + .ids(request = idsRequest, requestContext = None) + .map { idResult => + authorIds.map { authorId => + // Users cannot follow themselves but this is in network by definition + val isSelfTweet = authorId == viewerId + val inNetworkAuthorIds = idResult.ids.toSet + val isInNetwork = isSelfTweet || inNetworkAuthorIds.contains(authorId) || authorId == 0L + FeatureMapBuilder().add(InNetworkFeature, isInNetwork).build() + } + } + } + + private def createIdsRequest( + userId: Long, + relationshipTypes: Set[sg.RelationshipType], + targetIds: Option[Seq[Long]] = None + ): sg.IdsRequest = sg.IdsRequest( + relationshipTypes.map { relationshipType => + sg.SrcRelationship(userId, relationshipType, targets = targetIds) + }.toSeq, + Some(sg.PageRequest(selectAll = Some(true))) + ) +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/TSPInferredTopicFeatureHydrator.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/TSPInferredTopicFeatureHydrator.scala new file mode 100644 index 0000000000..2e823ebd82 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/TSPInferredTopicFeatureHydrator.scala @@ -0,0 +1,162 @@ +package com.twitter.home_mixer.functional_component.feature_hydrator + +import com.twitter.contentrecommender.{thriftscala => cr} +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.home_mixer.functional_component.feature_hydrator.adapters.inferred_topic.InferredTopicAdapter +import com.twitter.home_mixer.model.HomeFeatures.CandidateSourceIdFeature +import com.twitter.home_mixer.model.HomeFeatures.TSPMetricTagFeature +import com.twitter.home_mixer.model.HomeFeatures.TopicContextFunctionalityTypeFeature +import com.twitter.home_mixer.model.HomeFeatures.TopicIdSocialContextFeature +import com.twitter.ml.api.DataRecord +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.feature.Feature +import com.twitter.product_mixer.core.feature.FeatureWithDefaultOnFailure +import com.twitter.product_mixer.core.feature.datarecord.DataRecordInAFeature +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.feature.featuremap.FeatureMapBuilder +import com.twitter.product_mixer.core.functional_component.feature_hydrator.BulkCandidateFeatureHydrator +import com.twitter.product_mixer.core.model.common.CandidateWithFeatures +import com.twitter.product_mixer.core.model.common.identifier.FeatureHydratorIdentifier +import com.twitter.product_mixer.core.model.marshalling.response.urt.metadata.BasicTopicContextFunctionalityType +import com.twitter.product_mixer.core.model.marshalling.response.urt.metadata.RecommendationTopicContextFunctionalityType +import com.twitter.product_mixer.core.model.marshalling.response.urt.metadata.TopicContextFunctionalityType +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.stitch.Stitch +import com.twitter.strato.generated.client.topic_signals.tsp.TopicSocialProofClientColumn +import com.twitter.timelineservice.suggests.logging.candidate_tweet_source_id.{thriftscala => sid} +import com.twitter.topiclisting.TopicListingViewerContext +import com.twitter.tsp.{thriftscala => tsp} + +import javax.inject.Inject +import javax.inject.Singleton +import scala.collection.JavaConverters._ + +object TSPInferredTopicFeature extends Feature[TweetCandidate, Map[Long, Double]] +object TSPInferredTopicDataRecordFeature + extends DataRecordInAFeature[TweetCandidate] + with FeatureWithDefaultOnFailure[TweetCandidate, DataRecord] { + override def defaultValue: DataRecord = new DataRecord() +} + +@Singleton +class TSPInferredTopicFeatureHydrator @Inject() ( + topicSocialProofClientColumn: TopicSocialProofClientColumn, + statsReceiver: StatsReceiver, +) extends BulkCandidateFeatureHydrator[PipelineQuery, TweetCandidate] { + + override val identifier: FeatureHydratorIdentifier = FeatureHydratorIdentifier("TSPInferredTopic") + + override val features: Set[Feature[_, _]] = + Set( + TSPInferredTopicFeature, + TSPInferredTopicDataRecordFeature, + TopicIdSocialContextFeature, + TopicContextFunctionalityTypeFeature) + + private val topK = 3 + + private val sourcesToSetSocialProof: Set[sid.CandidateTweetSourceId] = Set( + sid.CandidateTweetSourceId.Simcluster, + sid.CandidateTweetSourceId.CroonTweet + ) + + private val scopedStatsReceiver = statsReceiver.scope(getClass.getSimpleName) + private val keyFoundCounter = scopedStatsReceiver.counter("key/found") + private val keyLossCounter = scopedStatsReceiver.counter("key/loss") + private val requestFailCounter = scopedStatsReceiver.counter("request/fail") + + private val DefaultFeatureMap = FeatureMapBuilder() + .add(TSPInferredTopicFeature, Map.empty[Long, Double]) + .add(TSPInferredTopicDataRecordFeature, new DataRecord()) + .add(TopicIdSocialContextFeature, None) + .add(TopicContextFunctionalityTypeFeature, None) + .build() + + override def apply( + query: PipelineQuery, + candidates: Seq[CandidateWithFeatures[TweetCandidate]] + ): Stitch[Seq[FeatureMap]] = { + val tags = candidates.collect { + case candidate if candidate.features.getTry(TSPMetricTagFeature).isReturn => + candidate.candidate.id -> candidate.features + .getOrElse(TSPMetricTagFeature, Set.empty[tsp.MetricTag]) + }.toMap + + val topicSocialProofRequest = + tsp.TopicSocialProofRequest( + userId = query.getRequiredUserId, + tweetIds = candidates.map(_.candidate.id).toSet, + displayLocation = cr.DisplayLocation.HomeTimeline, + topicListingSetting = tsp.TopicListingSetting.Followable, + context = TopicListingViewerContext.fromClientContext(query.clientContext).toThrift, + bypassModes = None, + // Only CRMixer source has this data. Convert the CRMixer metric tag to tsp metric tag. + tags = if (tags.isEmpty) None else Some(tags) + ) + + topicSocialProofClientColumn.fetcher + .fetch(topicSocialProofRequest) + .map(_.v) + .map { + case Some(response) => + candidates.map { candidate => + val topicWithScores = response.socialProofs.getOrElse(candidate.candidate.id, Seq.empty) + if (topicWithScores.nonEmpty) { + keyFoundCounter.incr() + val (socialProofId, socialProofFunctionalityType) = + if (candidate.features + .getOrElse(CandidateSourceIdFeature, None) + .exists(sourcesToSetSocialProof.contains)) { + getSocialProof(topicWithScores) + } else { + (None, None) + } + val inferredTopicFeatures = convertTopicWithScores(topicWithScores) + val inferredTopicDataRecord = + InferredTopicAdapter.adaptToDataRecords(inferredTopicFeatures).asScala.head + FeatureMapBuilder() + .add(TSPInferredTopicFeature, inferredTopicFeatures) + .add(TSPInferredTopicDataRecordFeature, inferredTopicDataRecord) + .add(TopicIdSocialContextFeature, socialProofId) + .add(TopicContextFunctionalityTypeFeature, socialProofFunctionalityType) + .build() + } else { + keyLossCounter.incr() + DefaultFeatureMap + } + } + case _ => + requestFailCounter.incr() + candidates.map { _ => + DefaultFeatureMap + } + } + } + + private def getSocialProof( + topicWithScores: Seq[tsp.TopicWithScore] + ): (Option[Long], Option[TopicContextFunctionalityType]) = { + val followingTopicId = topicWithScores + .collectFirst { + case tsp.TopicWithScore(topicId, _, _, Some(tsp.TopicFollowType.Following)) => + topicId + } + if (followingTopicId.nonEmpty) { + return (followingTopicId, Some(BasicTopicContextFunctionalityType)) + } + val implicitFollowingId = topicWithScores.collectFirst { + case tsp.TopicWithScore(topicId, _, _, Some(tsp.TopicFollowType.ImplicitFollow)) => + topicId + } + if (implicitFollowingId.nonEmpty) { + return (implicitFollowingId, Some(RecommendationTopicContextFunctionalityType)) + } + (None, None) + } + + private def convertTopicWithScores( + topicWithScores: Seq[tsp.TopicWithScore], + ): Map[Long, Double] = { + topicWithScores.sortBy(-_.score).take(topK).map(a => (a.topicId, a.score)).toMap + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/TimeFeaturesHydrator.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/TimeFeaturesHydrator.scala new file mode 100644 index 0000000000..16661188a2 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/TimeFeaturesHydrator.scala @@ -0,0 +1,251 @@ +package com.twitter.home_mixer.functional_component.feature_hydrator + +import com.twitter.conversions.DurationOps._ +import com.twitter.home_mixer.model.HomeFeatures.EarlybirdFeature +import com.twitter.home_mixer.model.HomeFeatures.NonPollingTimesFeature +import com.twitter.home_mixer.model.HomeFeatures.SourceTweetIdFeature +import com.twitter.ml.api.DataRecord +import com.twitter.ml.api.RichDataRecord +import com.twitter.ml.api.util.FDsl._ +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.feature.Feature +import com.twitter.product_mixer.core.feature.FeatureWithDefaultOnFailure +import com.twitter.product_mixer.core.feature.datarecord.DataRecordInAFeature +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.feature.featuremap.FeatureMapBuilder +import com.twitter.product_mixer.core.functional_component.feature_hydrator.CandidateFeatureHydrator +import com.twitter.product_mixer.core.model.common.identifier.FeatureHydratorIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.search.common.features.{thriftscala => sc} +import com.twitter.snowflake.id.SnowflakeId +import com.twitter.stitch.Stitch +import com.twitter.timelines.prediction.features.time_features.AccountAgeInterval +import com.twitter.timelines.prediction.features.time_features.TimeDataRecordFeatures._ +import com.twitter.timelines.prediction.features.time_features.TimeFeatures +import com.twitter.util.Duration +import scala.collection.Searching._ + +object TimeFeaturesDataRecordFeature + extends DataRecordInAFeature[TweetCandidate] + with FeatureWithDefaultOnFailure[TweetCandidate, DataRecord] { + override def defaultValue: DataRecord = new DataRecord() +} + +object TimeFeaturesHydrator extends CandidateFeatureHydrator[PipelineQuery, TweetCandidate] { + + override val identifier: FeatureHydratorIdentifier = FeatureHydratorIdentifier("TimeFeatures") + + override val features: Set[Feature[_, _]] = Set(TimeFeaturesDataRecordFeature) + + override def apply( + query: PipelineQuery, + candidate: TweetCandidate, + existingFeatures: FeatureMap + ): Stitch[FeatureMap] = { + Stitch.value { + val richDataRecord = new RichDataRecord() + setTimeFeatures(richDataRecord, candidate, existingFeatures, query) + FeatureMapBuilder() + .add(TimeFeaturesDataRecordFeature, richDataRecord.getRecord) + .build() + } + } + + private def setTimeFeatures( + richDataRecord: RichDataRecord, + candidate: TweetCandidate, + existingFeatures: FeatureMap, + query: PipelineQuery, + ): Unit = { + val timeFeaturesOpt = getTimeFeatures(query, candidate, existingFeatures) + timeFeaturesOpt.foreach(timeFeatures => setFeatures(timeFeatures, richDataRecord)) + } + + private[feature_hydrator] def getTimeFeatures( + query: PipelineQuery, + candidate: TweetCandidate, + existingFeatures: FeatureMap, + ): Option[TimeFeatures] = { + for { + requestTimestampMs <- Some(query.queryTime.inMilliseconds) + tweetId <- Some(candidate.id) + viewerId <- query.getOptionalUserId + tweetCreationTimeMs <- timeFromTweetOrUserId(tweetId) + timeSinceTweetCreation = requestTimestampMs - tweetCreationTimeMs + accountAgeDurationOpt = timeFromTweetOrUserId(viewerId).map { viewerAccountCreationTimeMs => + Duration.fromMilliseconds(requestTimestampMs - viewerAccountCreationTimeMs) + } + timeSinceSourceTweetCreation = + existingFeatures + .getOrElse(SourceTweetIdFeature, None) + .flatMap { sourceTweetId => + timeFromTweetOrUserId(sourceTweetId).map { sourceTweetCreationTimeMs => + requestTimestampMs - sourceTweetCreationTimeMs + } + } + .getOrElse(timeSinceTweetCreation) + if (timeSinceTweetCreation > 0 && timeSinceSourceTweetCreation > 0) + } yield { + val timeFeatures = TimeFeatures( + timeSinceTweetCreation = timeSinceTweetCreation, + timeSinceSourceTweetCreation = timeSinceSourceTweetCreation, + timeSinceViewerAccountCreationSecs = accountAgeDurationOpt.map(_.inSeconds), + isDay30NewUser = accountAgeDurationOpt.map(_ < 30.days).getOrElse(false), + isMonth12NewUser = accountAgeDurationOpt.map(_ < 365.days).getOrElse(false), + accountAgeInterval = accountAgeDurationOpt.flatMap(AccountAgeInterval.fromDuration), + isTweetRecycled = false // only set in RecyclableTweetCandidateFilter, but it's not used + ) + + val timeFeaturesWithLastEngagement = addLastEngagementTimeFeatures( + existingFeatures.getOrElse(EarlybirdFeature, None), + timeFeatures, + timeSinceSourceTweetCreation + ).getOrElse(timeFeatures) + + val nonPollingTimestampsMs = + query.features.map(_.getOrElse(NonPollingTimesFeature, Seq.empty)) + val timeFeaturesWithNonPollingOpt = addNonPollingTimeFeatures( + timeFeaturesWithLastEngagement, + requestTimestampMs, + tweetCreationTimeMs, + nonPollingTimestampsMs + ) + timeFeaturesWithNonPollingOpt.getOrElse(timeFeaturesWithLastEngagement) + } + } + + private def timeFromTweetOrUserId(tweetOrUserId: Long): Option[Long] = { + if (SnowflakeId.isSnowflakeId(tweetOrUserId)) + Some(SnowflakeId(tweetOrUserId).time.inMilliseconds) + else None + } + + private def addLastEngagementTimeFeatures( + tweetFeaturesOpt: Option[sc.ThriftTweetFeatures], + timeFeatures: TimeFeatures, + timeSinceSourceTweetCreation: Long + ): Option[TimeFeatures] = { + tweetFeaturesOpt.map { tweetFeatures => + val lastFavSinceCreationHrs = tweetFeatures.lastFavSinceCreationHrs.map(_.toDouble) + val lastRetweetSinceCreationHrs = tweetFeatures.lastRetweetSinceCreationHrs.map(_.toDouble) + val lastReplySinceCreationHrs = tweetFeatures.lastReplySinceCreationHrs.map(_.toDouble) + val lastQuoteSinceCreationHrs = tweetFeatures.lastQuoteSinceCreationHrs.map(_.toDouble) + + timeFeatures.copy( + lastFavSinceCreationHrs = lastFavSinceCreationHrs, + lastRetweetSinceCreationHrs = lastRetweetSinceCreationHrs, + lastReplySinceCreationHrs = lastReplySinceCreationHrs, + lastQuoteSinceCreationHrs = lastQuoteSinceCreationHrs, + timeSinceLastFavoriteHrs = getTimeSinceLastEngagementHrs( + lastFavSinceCreationHrs, + timeSinceSourceTweetCreation + ), + timeSinceLastRetweetHrs = getTimeSinceLastEngagementHrs( + lastRetweetSinceCreationHrs, + timeSinceSourceTweetCreation + ), + timeSinceLastReplyHrs = getTimeSinceLastEngagementHrs( + lastReplySinceCreationHrs, + timeSinceSourceTweetCreation + ), + timeSinceLastQuoteHrs = getTimeSinceLastEngagementHrs( + lastQuoteSinceCreationHrs, + timeSinceSourceTweetCreation + ) + ) + } + } + + private def addNonPollingTimeFeatures( + timeFeatures: TimeFeatures, + requestTimestampMs: Long, + creationTimeMs: Long, + nonPollingTimestampsMs: Option[Seq[Long]] + ): Option[TimeFeatures] = { + for { + nonPollingTimestampsMs <- nonPollingTimestampsMs + lastNonPollingTimestampMs <- nonPollingTimestampsMs.headOption + earliestNonPollingTimestampMs <- nonPollingTimestampsMs.lastOption + } yield { + val timeSinceLastNonPollingRequest = requestTimestampMs - lastNonPollingTimestampMs + val tweetAgeRatio = timeSinceLastNonPollingRequest / math.max( + 1.0, + timeFeatures.timeSinceTweetCreation + ) + /* + * Non-polling timestamps are stored in chronological order. + * The latest timestamps occur first, therefore we need to explicitly search in reverse order. + */ + val nonPollingRequestsSinceTweetCreation = + if (nonPollingTimestampsMs.nonEmpty) { + nonPollingTimestampsMs.search(creationTimeMs)(Ordering[Long].reverse).insertionPoint + } else { + 0 + } + /* + * Calculate the average time between non-polling requests; include + * request time in this calculation as latest timestamp. + */ + val timeBetweenNonPollingRequestsAvg = + (requestTimestampMs - earliestNonPollingTimestampMs) / math + .max(1.0, nonPollingTimestampsMs.size) + val timeFeaturesWithNonPolling = timeFeatures.copy( + timeBetweenNonPollingRequestsAvg = Some(timeBetweenNonPollingRequestsAvg), + timeSinceLastNonPollingRequest = Some(timeSinceLastNonPollingRequest), + nonPollingRequestsSinceTweetCreation = Some(nonPollingRequestsSinceTweetCreation), + tweetAgeRatio = Some(tweetAgeRatio) + ) + timeFeaturesWithNonPolling + } + } + + private[this] def getTimeSinceLastEngagementHrs( + lastEngagementTimeSinceCreationHrsOpt: Option[Double], + timeSinceTweetCreation: Long + ): Option[Double] = { + lastEngagementTimeSinceCreationHrsOpt.map { lastEngagementTimeSinceCreationHrs => + val timeSinceTweetCreationHrs = (timeSinceTweetCreation / (60 * 60 * 1000)).toInt + timeSinceTweetCreationHrs - lastEngagementTimeSinceCreationHrs + } + } + + private def setFeatures(features: TimeFeatures, richDataRecord: RichDataRecord): Unit = { + val record = richDataRecord.getRecord + .setFeatureValue(IS_TWEET_RECYCLED, features.isTweetRecycled) + .setFeatureValue(TIME_SINCE_TWEET_CREATION, features.timeSinceTweetCreation) + .setFeatureValueFromOption( + TIME_SINCE_VIEWER_ACCOUNT_CREATION_SECS, + features.timeSinceViewerAccountCreationSecs) + .setFeatureValue( + USER_ID_IS_SNOWFLAKE_ID, + features.timeSinceViewerAccountCreationSecs.isDefined + ) + .setFeatureValueFromOption(ACCOUNT_AGE_INTERVAL, features.accountAgeInterval.map(_.id.toLong)) + .setFeatureValue(IS_30_DAY_NEW_USER, features.isDay30NewUser) + .setFeatureValue(IS_12_MONTH_NEW_USER, features.isMonth12NewUser) + .setFeatureValueFromOption(LAST_FAVORITE_SINCE_CREATION_HRS, features.lastFavSinceCreationHrs) + .setFeatureValueFromOption( + LAST_RETWEET_SINCE_CREATION_HRS, + features.lastRetweetSinceCreationHrs + ) + .setFeatureValueFromOption(LAST_REPLY_SINCE_CREATION_HRS, features.lastReplySinceCreationHrs) + .setFeatureValueFromOption(LAST_QUOTE_SINCE_CREATION_HRS, features.lastQuoteSinceCreationHrs) + .setFeatureValueFromOption(TIME_SINCE_LAST_FAVORITE_HRS, features.timeSinceLastFavoriteHrs) + .setFeatureValueFromOption(TIME_SINCE_LAST_RETWEET_HRS, features.timeSinceLastRetweetHrs) + .setFeatureValueFromOption(TIME_SINCE_LAST_REPLY_HRS, features.timeSinceLastReplyHrs) + .setFeatureValueFromOption(TIME_SINCE_LAST_QUOTE_HRS, features.timeSinceLastQuoteHrs) + /* + * set features whose values are optional as some users do not have non-polling timestamps + */ + features.timeBetweenNonPollingRequestsAvg.foreach( + record.setFeatureValue(TIME_BETWEEN_NON_POLLING_REQUESTS_AVG, _) + ) + features.timeSinceLastNonPollingRequest.foreach( + record.setFeatureValue(TIME_SINCE_LAST_NON_POLLING_REQUEST, _) + ) + features.nonPollingRequestsSinceTweetCreation.foreach( + record.setFeatureValue(NON_POLLING_REQUESTS_SINCE_TWEET_CREATION, _) + ) + features.tweetAgeRatio.foreach(record.setFeatureValue(TWEET_AGE_RATIO, _)) + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/TimelineServiceTweetsQueryFeatureHydrator.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/TimelineServiceTweetsQueryFeatureHydrator.scala new file mode 100644 index 0000000000..fc9727cb77 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/TimelineServiceTweetsQueryFeatureHydrator.scala @@ -0,0 +1,63 @@ +package com.twitter.home_mixer.functional_component.feature_hydrator + +import com.twitter.home_mixer.marshaller.timelines.DeviceContextMarshaller +import com.twitter.home_mixer.model.request.DeviceContext +import com.twitter.home_mixer.model.request.HasDeviceContext +import com.twitter.home_mixer.service.HomeMixerAlertConfig +import com.twitter.product_mixer.core.feature.Feature +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.feature.featuremap.FeatureMapBuilder +import com.twitter.product_mixer.core.functional_component.feature_hydrator.QueryFeatureHydrator +import com.twitter.product_mixer.core.model.common.identifier.FeatureHydratorIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.stitch.Stitch +import com.twitter.stitch.timelineservice.TimelineService +import com.twitter.timelineservice.{thriftscala => t} +import javax.inject.Inject +import javax.inject.Singleton + +object TimelineServiceTweetsFeature extends Feature[PipelineQuery, Seq[Long]] + +@Singleton +case class TimelineServiceTweetsQueryFeatureHydrator @Inject() ( + timelineService: TimelineService, + deviceContextMarshaller: DeviceContextMarshaller) + extends QueryFeatureHydrator[PipelineQuery with HasDeviceContext] { + + override val identifier: FeatureHydratorIdentifier = + FeatureHydratorIdentifier("TimelineServiceTweets") + + override val features: Set[Feature[_, _]] = Set(TimelineServiceTweetsFeature) + + private val MaxTimelineServiceTweets = 200 + + override def hydrate(query: PipelineQuery with HasDeviceContext): Stitch[FeatureMap] = { + val deviceContext = query.deviceContext.getOrElse(DeviceContext.Empty) + + val timelineQueryOptions = t.TimelineQueryOptions( + contextualUserId = query.clientContext.userId, + deviceContext = Some(deviceContextMarshaller(deviceContext, query.clientContext)) + ) + + val timelineServiceQuery = t.TimelineQuery( + timelineType = t.TimelineType.Home, + timelineId = query.getRequiredUserId, + maxCount = MaxTimelineServiceTweets.toShort, + cursor2 = None, + options = Some(timelineQueryOptions), + timelineId2 = query.clientContext.userId.map(t.TimelineId(t.TimelineType.Home, _, None)), + ) + + timelineService.getTimeline(timelineServiceQuery).map { timeline => + val tweets = timeline.entries.collect { + case t.TimelineEntry.Tweet(tweet) => tweet.statusId + } + + FeatureMapBuilder().add(TimelineServiceTweetsFeature, tweets).build() + } + } + + override val alerts = Seq( + HomeMixerAlertConfig.BusinessHours.defaultSuccessRateAlert(99.7) + ) +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/TweetImpressionsQueryFeatureHydrator.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/TweetImpressionsQueryFeatureHydrator.scala new file mode 100644 index 0000000000..4e585ec453 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/TweetImpressionsQueryFeatureHydrator.scala @@ -0,0 +1,87 @@ +package com.twitter.home_mixer.functional_component.feature_hydrator + +import com.twitter.conversions.DurationOps._ +import com.twitter.home_mixer.model.HomeFeatures.TweetImpressionsFeature +import com.twitter.home_mixer.model.request.HasSeenTweetIds +import com.twitter.home_mixer.service.HomeMixerAlertConfig +import com.twitter.product_mixer.core.feature.Feature +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.feature.featuremap.FeatureMapBuilder +import com.twitter.product_mixer.core.functional_component.feature_hydrator.QueryFeatureHydrator +import com.twitter.product_mixer.core.model.common.identifier.FeatureHydratorIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.stitch.Stitch +import com.twitter.timelines.impression.{thriftscala => t} +import com.twitter.timelines.impressionstore.store.ManhattanTweetImpressionStoreClient +import com.twitter.util.Duration +import com.twitter.util.Time +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +case class TweetImpressionsQueryFeatureHydrator[ + Query <: PipelineQuery with HasSeenTweetIds] @Inject() ( + manhattanTweetImpressionStoreClient: ManhattanTweetImpressionStoreClient) + extends QueryFeatureHydrator[Query] { + + private val TweetImpressionTTL = 1.day + private val TweetImpressionCap = 3000 + + override val identifier: FeatureHydratorIdentifier = FeatureHydratorIdentifier("TweetImpressions") + + override val features: Set[Feature[_, _]] = Set(TweetImpressionsFeature) + + override def hydrate(query: Query): Stitch[FeatureMap] = { + manhattanTweetImpressionStoreClient.get(query.getRequiredUserId).map { entriesOpt => + val entries = entriesOpt.map(_.entries).toSeq.flatten + val updatedImpressions = + if (query.seenTweetIds.forall(_.isEmpty)) entries + else updateTweetImpressions(entries, query.seenTweetIds.get) + + FeatureMapBuilder().add(TweetImpressionsFeature, updatedImpressions).build() + } + } + + override val alerts = Seq( + HomeMixerAlertConfig.BusinessHours.defaultSuccessRateAlert(99.8) + ) + + /** + * 1) Check timestamps and remove expired tweets based on [[TweetImpressionTTL]] + * 2) Filter duplicates between current tweets and those in the impression store (remove older ones) + * 3) Prepend new (Timestamp, Seq[TweetIds]) to the tweets from the impression store + * 4) Truncate older tweets if sum of all tweets across timestamps >= [[TweetImpressionCap]], + */ + private[feature_hydrator] def updateTweetImpressions( + tweetImpressionsFromStore: Seq[t.TweetImpressionsEntry], + seenIdsFromClient: Seq[Long], + currentTime: Long = Time.now.inMilliseconds, + tweetImpressionTTL: Duration = TweetImpressionTTL, + tweetImpressionCap: Int = TweetImpressionCap, + ): Seq[t.TweetImpressionsEntry] = { + val seenIdsFromClientSet = seenIdsFromClient.toSet + val dedupedTweetImpressionsFromStore: Seq[t.TweetImpressionsEntry] = tweetImpressionsFromStore + .collect { + case t.TweetImpressionsEntry(ts, tweetIds) + if Time.fromMilliseconds(ts).untilNow < tweetImpressionTTL => + t.TweetImpressionsEntry(ts, tweetIds.filterNot(seenIdsFromClientSet.contains)) + }.filter { _.tweetIds.nonEmpty } + + val mergedTweetImpressionsEntries = + t.TweetImpressionsEntry(currentTime, seenIdsFromClient) +: dedupedTweetImpressionsFromStore + val initialTweetImpressionsWithCap = (Seq.empty[t.TweetImpressionsEntry], tweetImpressionCap) + + val (truncatedTweetImpressionsEntries: Seq[t.TweetImpressionsEntry], _) = + mergedTweetImpressionsEntries + .foldLeft(initialTweetImpressionsWithCap) { + case ( + (tweetImpressions: Seq[t.TweetImpressionsEntry], remainingCap), + t.TweetImpressionsEntry(ts, tweetIds)) if remainingCap > 0 => + ( + t.TweetImpressionsEntry(ts, tweetIds.take(remainingCap)) +: tweetImpressions, + remainingCap - tweetIds.size) + case (tweetImpressionsWithCap, _) => tweetImpressionsWithCap + } + truncatedTweetImpressionsEntries.reverse + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/TweetMetaDataFeatureHydrator.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/TweetMetaDataFeatureHydrator.scala new file mode 100644 index 0000000000..8c79c38742 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/TweetMetaDataFeatureHydrator.scala @@ -0,0 +1,66 @@ +package com.twitter.home_mixer.functional_component.feature_hydrator + +import com.twitter.home_mixer.model.HomeFeatures.CandidateSourceIdFeature +import com.twitter.home_mixer.util.CandidatesUtil +import com.twitter.ml.api.DataRecord +import com.twitter.ml.api.RichDataRecord +import com.twitter.ml.api.constant.SharedFeatures +import com.twitter.ml.api.util.DataRecordConverters._ +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.feature.Feature +import com.twitter.product_mixer.core.feature.FeatureWithDefaultOnFailure +import com.twitter.product_mixer.core.feature.datarecord.DataRecordInAFeature +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.feature.featuremap.FeatureMapBuilder +import com.twitter.product_mixer.core.functional_component.feature_hydrator.CandidateFeatureHydrator +import com.twitter.product_mixer.core.model.common.identifier.FeatureHydratorIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.stitch.Stitch +import com.twitter.timelines.prediction.features.common.TimelinesSharedFeatures +import java.lang.{Long => JLong} + +object TweetMetaDataDataRecord + extends DataRecordInAFeature[TweetCandidate] + with FeatureWithDefaultOnFailure[TweetCandidate, DataRecord] { + override def defaultValue: DataRecord = new DataRecord() +} + +object TweetMetaDataFeatureHydrator + extends CandidateFeatureHydrator[PipelineQuery, TweetCandidate] { + + override val identifier: FeatureHydratorIdentifier = FeatureHydratorIdentifier("TweetMetaData") + + override def features: Set[Feature[_, _]] = Set(TweetMetaDataDataRecord) + + override def apply( + query: PipelineQuery, + candidate: TweetCandidate, + existingFeatures: FeatureMap + ): Stitch[FeatureMap] = { + val richDataRecord = new RichDataRecord() + + setFeatures(richDataRecord, candidate, existingFeatures) + + Stitch.value { + FeatureMapBuilder() + .add(TweetMetaDataDataRecord, richDataRecord.getRecord) + .build() + } + } + + private def setFeatures( + richDataRecord: RichDataRecord, + candidate: TweetCandidate, + existingFeatures: FeatureMap + ): Unit = { + richDataRecord.setFeatureValue[JLong](SharedFeatures.TWEET_ID, candidate.id) + + richDataRecord.setFeatureValueFromOption( + TimelinesSharedFeatures.ORIGINAL_AUTHOR_ID, + CandidatesUtil.getOriginalAuthorId(existingFeatures)) + + richDataRecord.setFeatureValueFromOption( + TimelinesSharedFeatures.CANDIDATE_TWEET_SOURCE_ID, + existingFeatures.getOrElse(CandidateSourceIdFeature, None).map(_.value.toLong)) + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/TweetypieContentFeatureHydrator.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/TweetypieContentFeatureHydrator.scala new file mode 100644 index 0000000000..8ad3acc46d --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/TweetypieContentFeatureHydrator.scala @@ -0,0 +1,149 @@ +package com.twitter.home_mixer.functional_component.feature_hydrator + +import com.twitter.escherbird.{thriftscala => esb} +import com.twitter.finagle.stats.Stat +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.home_mixer.functional_component.feature_hydrator.adapters.content.ContentFeatureAdapter +import com.twitter.home_mixer.model.HomeFeatures.MediaUnderstandingAnnotationIdsFeature +import com.twitter.home_mixer.model.HomeFeatures.SourceTweetIdFeature +import com.twitter.home_mixer.param.HomeMixerInjectionNames.TweetypieContentRepository +import com.twitter.home_mixer.util.ObservedKeyValueResultHandler +import com.twitter.home_mixer.util.tweetypie.content.FeatureExtractionHelper +import com.twitter.ml.api.DataRecord +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.feature.Feature +import com.twitter.product_mixer.core.feature.FeatureWithDefaultOnFailure +import com.twitter.product_mixer.core.feature.datarecord.DataRecordInAFeature +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.feature.featuremap.FeatureMapBuilder +import com.twitter.product_mixer.core.functional_component.feature_hydrator.BulkCandidateFeatureHydrator +import com.twitter.product_mixer.core.model.common.CandidateWithFeatures +import com.twitter.product_mixer.core.model.common.identifier.FeatureHydratorIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.product_mixer.core.util.OffloadFuturePools +import com.twitter.servo.keyvalue.KeyValueResult +import com.twitter.servo.repository.KeyValueRepository +import com.twitter.stitch.Stitch +import com.twitter.timelines.prediction.common.util.MediaUnderstandingAnnotations +import com.twitter.tweetypie.{thriftscala => tp} +import com.twitter.util.Future +import com.twitter.util.Return +import com.twitter.util.Throw +import com.twitter.util.Try +import javax.inject.Inject +import javax.inject.Named +import javax.inject.Singleton +import scala.collection.JavaConverters._ + +object TweetypieContentDataRecordFeature + extends DataRecordInAFeature[TweetCandidate] + with FeatureWithDefaultOnFailure[TweetCandidate, DataRecord] { + override def defaultValue: DataRecord = new DataRecord() +} + +@Singleton +class TweetypieContentFeatureHydrator @Inject() ( + @Named(TweetypieContentRepository) client: KeyValueRepository[Seq[Long], Long, tp.Tweet], + override val statsReceiver: StatsReceiver) + extends BulkCandidateFeatureHydrator[PipelineQuery, TweetCandidate] + with ObservedKeyValueResultHandler { + + override val identifier: FeatureHydratorIdentifier = FeatureHydratorIdentifier("TweetypieContent") + + override val features: Set[Feature[_, _]] = Set( + MediaUnderstandingAnnotationIdsFeature, + TweetypieContentDataRecordFeature + ) + + override val statScope: String = identifier.toString + + private val bulkRequestLatencyStat = + statsReceiver.scope(statScope).scope("bulkRequest").stat("latency_ms") + private val postTransformerLatencyStat = + statsReceiver.scope(statScope).scope("postTransformer").stat("latency_ms") + private val bulkPostTransformerLatencyStat = + statsReceiver.scope(statScope).scope("bulkPostTransformer").stat("latency_ms") + + override def apply( + query: PipelineQuery, + candidates: Seq[CandidateWithFeatures[TweetCandidate]] + ): Stitch[Seq[FeatureMap]] = { + Stitch.callFuture { + val tweetIdsToHydrate = candidates.map(getCandidateOriginalTweetId).distinct + + val response: Future[KeyValueResult[Long, tp.Tweet]] = + Stat.timeFuture(bulkRequestLatencyStat)( + if (tweetIdsToHydrate.isEmpty) { + Future.value(KeyValueResult.empty) + } else { + client(tweetIdsToHydrate) + } + ) + + response.flatMap { result => + Stat.timeFuture(bulkPostTransformerLatencyStat) { + OffloadFuturePools + .parallelize[CandidateWithFeatures[TweetCandidate], Try[(Seq[Long], DataRecord)]]( + candidates, + parTransformer(result, _), + parallelism = 32, + default = Return((Seq.empty, new DataRecord)) + ).map { + _.map { + case Return(result) => + FeatureMapBuilder() + .add(MediaUnderstandingAnnotationIdsFeature, result._1) + .add(TweetypieContentDataRecordFeature, result._2) + .build() + case Throw(e) => + FeatureMapBuilder() + .add(MediaUnderstandingAnnotationIdsFeature, Throw(e)) + .add(TweetypieContentDataRecordFeature, Throw(e)) + .build() + } + } + } + } + } + } + + private def parTransformer( + result: KeyValueResult[Long, tp.Tweet], + candidate: CandidateWithFeatures[TweetCandidate] + ): Try[(Seq[Long], DataRecord)] = { + val originalTweetId = Some(getCandidateOriginalTweetId(candidate)) + + val value = observedGet(key = originalTweetId, keyValueResult = result) + Stat.time(postTransformerLatencyStat)(postTransformer(value)) + } + + private def postTransformer( + result: Try[Option[tp.Tweet]] + ): Try[(Seq[Long], DataRecord)] = { + result.map { tweet => + val transformedValue = tweet.map(FeatureExtractionHelper.extractFeatures) + val semanticAnnotations = transformedValue + .flatMap { contentFeatures => + contentFeatures.semanticCoreAnnotations.map { + getNonSensitiveHighRecallMediaUnderstandingAnnotationEntityIds + } + }.getOrElse(Seq.empty) + val dataRecord = ContentFeatureAdapter.adaptToDataRecords(transformedValue).asScala.head + (semanticAnnotations, dataRecord) + } + } + + private def getCandidateOriginalTweetId( + candidate: CandidateWithFeatures[TweetCandidate] + ): Long = { + candidate.features + .getOrElse(SourceTweetIdFeature, None).getOrElse(candidate.candidate.id) + } + + private def getNonSensitiveHighRecallMediaUnderstandingAnnotationEntityIds( + semanticCoreAnnotations: Seq[esb.TweetEntityAnnotation] + ): Seq[Long] = + semanticCoreAnnotations + .filter(MediaUnderstandingAnnotations.isEligibleNonSensitiveHighRecallMUAnnotation) + .map(_.entityId) +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/TweetypieFeatureHydrator.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/TweetypieFeatureHydrator.scala new file mode 100644 index 0000000000..09fc62405d --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/TweetypieFeatureHydrator.scala @@ -0,0 +1,156 @@ +package com.twitter.home_mixer.functional_component.feature_hydrator + +import com.twitter.home_mixer.model.HomeFeatures.AuthorIdFeature +import com.twitter.home_mixer.model.HomeFeatures.InReplyToTweetIdFeature +import com.twitter.home_mixer.model.HomeFeatures.IsHydratedFeature +import com.twitter.home_mixer.model.HomeFeatures.IsNsfwFeature +import com.twitter.home_mixer.model.HomeFeatures.IsRetweetFeature +import com.twitter.home_mixer.model.HomeFeatures.QuotedTweetDroppedFeature +import com.twitter.home_mixer.model.HomeFeatures.QuotedTweetIdFeature +import com.twitter.home_mixer.model.HomeFeatures.QuotedUserIdFeature +import com.twitter.home_mixer.model.HomeFeatures.SourceTweetIdFeature +import com.twitter.home_mixer.model.HomeFeatures.SourceUserIdFeature +import com.twitter.home_mixer.model.HomeFeatures.TweetTextFeature +import com.twitter.home_mixer.model.request.FollowingProduct +import com.twitter.home_mixer.model.request.ForYouProduct +import com.twitter.home_mixer.model.request.ListTweetsProduct +import com.twitter.home_mixer.model.request.ScoredTweetsProduct +import com.twitter.home_mixer.util.tweetypie.RequestFields +import com.twitter.product_mixer.component_library.feature_hydrator.candidate.tweet_is_nsfw.IsNsfw +import com.twitter.product_mixer.component_library.feature_hydrator.candidate.tweet_visibility_reason.VisibilityReason +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.feature.Feature +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.feature.featuremap.FeatureMapBuilder +import com.twitter.product_mixer.core.functional_component.feature_hydrator.CandidateFeatureHydrator +import com.twitter.product_mixer.core.model.common.identifier.FeatureHydratorIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.spam.rtf.{thriftscala => rtf} +import com.twitter.stitch.Stitch +import com.twitter.stitch.tweetypie.{TweetyPie => TweetypieStitchClient} +import com.twitter.tweetypie.{thriftscala => tp} +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class TweetypieFeatureHydrator @Inject() (tweetypieStitchClient: TweetypieStitchClient) + extends CandidateFeatureHydrator[PipelineQuery, TweetCandidate] { + + override val identifier: FeatureHydratorIdentifier = FeatureHydratorIdentifier("Tweetypie") + + override val features: Set[Feature[_, _]] = Set( + AuthorIdFeature, + InReplyToTweetIdFeature, + IsHydratedFeature, + IsNsfw, + IsNsfwFeature, + IsRetweetFeature, + QuotedTweetDroppedFeature, + QuotedTweetIdFeature, + QuotedUserIdFeature, + SourceTweetIdFeature, + SourceUserIdFeature, + TweetTextFeature, + VisibilityReason + ) + + private val DefaultFeatureMap = FeatureMapBuilder() + .add(IsHydratedFeature, false) + .add(IsNsfw, None) + .add(IsNsfwFeature, false) + .add(QuotedTweetDroppedFeature, false) + .add(TweetTextFeature, None) + .add(VisibilityReason, None) + .build() + + override def apply( + query: PipelineQuery, + candidate: TweetCandidate, + existingFeatures: FeatureMap + ): Stitch[FeatureMap] = { + val safetyLevel = query.product match { + case FollowingProduct => rtf.SafetyLevel.TimelineHomeLatest + case ForYouProduct => rtf.SafetyLevel.TimelineHome + case ScoredTweetsProduct => rtf.SafetyLevel.TimelineHome + case ListTweetsProduct => rtf.SafetyLevel.TimelineLists + case unknown => throw new UnsupportedOperationException(s"Unknown product: $unknown") + } + + val tweetFieldsOptions = tp.GetTweetFieldsOptions( + tweetIncludes = RequestFields.TweetTPHydrationFields, + includeRetweetedTweet = true, + includeQuotedTweet = true, + visibilityPolicy = tp.TweetVisibilityPolicy.UserVisible, + safetyLevel = Some(safetyLevel), + forUserId = Some(query.getRequiredUserId) + ) + + tweetypieStitchClient.getTweetFields(tweetId = candidate.id, options = tweetFieldsOptions).map { + case tp.GetTweetFieldsResult(_, tp.TweetFieldsResultState.Found(found), quoteOpt, _) => + val coreData = found.tweet.coreData + val isNsfwAdmin = coreData.exists(_.nsfwAdmin) + val isNsfwUser = coreData.exists(_.nsfwUser) + + val quotedTweetDropped = quoteOpt.exists { + case _: tp.TweetFieldsResultState.Filtered => true + case _: tp.TweetFieldsResultState.NotFound => true + case _ => false + } + val quotedTweetIsNsfw = quoteOpt.exists { + case quoteTweet: tp.TweetFieldsResultState.Found => + quoteTweet.found.tweet.coreData.exists(data => data.nsfwAdmin || data.nsfwUser) + case _ => false + } + + val sourceTweetIsNsfw = + found.retweetedTweet.exists(_.coreData.exists(data => data.nsfwAdmin || data.nsfwUser)) + + val tweetText = coreData.map(_.text) + + val tweetAuthorId = coreData.map(_.userId) + val inReplyToTweetId = coreData.flatMap(_.reply.flatMap(_.inReplyToStatusId)) + val retweetedTweetId = found.retweetedTweet.map(_.id) + val quotedTweetId = quoteOpt.flatMap { + case quoteTweet: tp.TweetFieldsResultState.Found => + Some(quoteTweet.found.tweet.id) + case _ => None + } + + val retweetedTweetUserId = found.retweetedTweet.flatMap(_.coreData).map(_.userId) + val quotedTweetUserId = quoteOpt.flatMap { + case quoteTweet: tp.TweetFieldsResultState.Found => + quoteTweet.found.tweet.coreData.map(_.userId) + case _ => None + } + + val isNsfw = isNsfwAdmin || isNsfwUser || sourceTweetIsNsfw || quotedTweetIsNsfw + + FeatureMapBuilder() + .add(AuthorIdFeature, tweetAuthorId) + .add(InReplyToTweetIdFeature, inReplyToTweetId) + .add(IsHydratedFeature, true) + .add(IsNsfw, Some(isNsfw)) + .add(IsNsfwFeature, isNsfw) + .add(IsRetweetFeature, retweetedTweetId.isDefined) + .add(QuotedTweetDroppedFeature, quotedTweetDropped) + .add(QuotedTweetIdFeature, quotedTweetId) + .add(QuotedUserIdFeature, quotedTweetUserId) + .add(SourceTweetIdFeature, retweetedTweetId) + .add(SourceUserIdFeature, retweetedTweetUserId) + .add(TweetTextFeature, tweetText) + .add(VisibilityReason, found.suppressReason) + .build() + + // If no tweet result found, return default and pre-existing features + case _ => + DefaultFeatureMap + + (AuthorIdFeature, existingFeatures.getOrElse(AuthorIdFeature, None)) + + (InReplyToTweetIdFeature, existingFeatures.getOrElse(InReplyToTweetIdFeature, None)) + + (IsRetweetFeature, existingFeatures.getOrElse(IsRetweetFeature, false)) + + (QuotedTweetIdFeature, existingFeatures.getOrElse(QuotedTweetIdFeature, None)) + + (QuotedUserIdFeature, existingFeatures.getOrElse(QuotedUserIdFeature, None)) + + (SourceTweetIdFeature, existingFeatures.getOrElse(SourceTweetIdFeature, None)) + + (SourceUserIdFeature, existingFeatures.getOrElse(SourceUserIdFeature, None)) + } + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/TweetypieStaticEntitiesFeatureHydrator.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/TweetypieStaticEntitiesFeatureHydrator.scala new file mode 100644 index 0000000000..773f2144e7 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/TweetypieStaticEntitiesFeatureHydrator.scala @@ -0,0 +1,161 @@ +package com.twitter.home_mixer.functional_component.feature_hydrator + +import com.google.inject.name.Named +import com.twitter.conversions.DurationOps.RichDuration +import com.twitter.home_mixer.model.HomeFeatures.AuthorIdFeature +import com.twitter.home_mixer.model.HomeFeatures.DirectedAtUserIdFeature +import com.twitter.home_mixer.model.HomeFeatures.HasImageFeature +import com.twitter.home_mixer.model.HomeFeatures.HasVideoFeature +import com.twitter.home_mixer.model.HomeFeatures.InReplyToTweetIdFeature +import com.twitter.home_mixer.model.HomeFeatures.InReplyToUserIdFeature +import com.twitter.home_mixer.model.HomeFeatures.IsRetweetFeature +import com.twitter.home_mixer.model.HomeFeatures.MentionScreenNameFeature +import com.twitter.home_mixer.model.HomeFeatures.MentionUserIdFeature +import com.twitter.home_mixer.model.HomeFeatures.QuotedTweetIdFeature +import com.twitter.home_mixer.model.HomeFeatures.QuotedUserIdFeature +import com.twitter.home_mixer.model.HomeFeatures.SemanticAnnotationFeature +import com.twitter.home_mixer.model.HomeFeatures.SourceTweetIdFeature +import com.twitter.home_mixer.model.HomeFeatures.SourceUserIdFeature +import com.twitter.home_mixer.param.HomeMixerInjectionNames.TweetypieStaticEntitiesCache +import com.twitter.home_mixer.util.tweetypie.RequestFields +import com.twitter.home_mixer.util.tweetypie.content.TweetMediaFeaturesExtractor +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.feature.Feature +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.feature.featuremap.FeatureMapBuilder +import com.twitter.product_mixer.core.functional_component.feature_hydrator.BulkCandidateFeatureHydrator +import com.twitter.product_mixer.core.model.common.CandidateWithFeatures +import com.twitter.product_mixer.core.model.common.identifier.FeatureHydratorIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.servo.cache.TtlCache +import com.twitter.spam.rtf.{thriftscala => sp} +import com.twitter.stitch.Stitch +import com.twitter.stitch.tweetypie.{TweetyPie => TweetypieStitchClient} +import com.twitter.tweetypie.{thriftscala => tp} +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class TweetypieStaticEntitiesFeatureHydrator @Inject() ( + tweetypieStitchClient: TweetypieStitchClient, + @Named(TweetypieStaticEntitiesCache) cacheClient: TtlCache[Long, tp.Tweet]) + extends BulkCandidateFeatureHydrator[PipelineQuery, TweetCandidate] { + + override val identifier: FeatureHydratorIdentifier = + FeatureHydratorIdentifier("TweetypieStaticEntities") + + override val features: Set[Feature[_, _]] = Set( + AuthorIdFeature, + DirectedAtUserIdFeature, + HasImageFeature, + HasVideoFeature, + InReplyToTweetIdFeature, + InReplyToUserIdFeature, + IsRetweetFeature, + MentionScreenNameFeature, + MentionUserIdFeature, + QuotedTweetIdFeature, + QuotedUserIdFeature, + SemanticAnnotationFeature, + SourceTweetIdFeature, + SourceUserIdFeature + ) + + private val CacheTTL = 24.hours + + private val DefaultFeatureMap = FeatureMapBuilder() + .add(AuthorIdFeature, None) + .add(DirectedAtUserIdFeature, None) + .add(HasImageFeature, false) + .add(HasVideoFeature, false) + .add(InReplyToTweetIdFeature, None) + .add(InReplyToUserIdFeature, None) + .add(IsRetweetFeature, false) + .add(MentionScreenNameFeature, Seq.empty) + .add(MentionUserIdFeature, Seq.empty) + .add(QuotedTweetIdFeature, None) + .add(QuotedUserIdFeature, None) + .add(SemanticAnnotationFeature, Seq.empty) + .add(SourceTweetIdFeature, None) + .add(SourceUserIdFeature, None) + .build() + + /** + * Steps: + * 1. query cache with all candidates + * 2. create a cached feature map + * 3. iterate candidates to hydrate features + * 3.a transform cached candidates + * 3.b hydrate non-cached candidates from Tweetypie and write to cache + */ + override def apply( + query: PipelineQuery, + candidates: Seq[CandidateWithFeatures[TweetCandidate]] + ): Stitch[Seq[FeatureMap]] = { + val tweetIds = candidates.map(_.candidate.id) + val cachedTweetsMapFu = cacheClient + .get(tweetIds) + .map(_.found) + + Stitch.callFuture(cachedTweetsMapFu).flatMap { cachedTweets => + Stitch.collect { + candidates.map { candidate => + if (cachedTweets.contains(candidate.candidate.id)) + Stitch.value(createFeatureMap(cachedTweets(candidate.candidate.id))) + else readFromTweetypie(query, candidate) + } + } + } + } + + private def createFeatureMap(tweet: tp.Tweet): FeatureMap = { + val coreData = tweet.coreData + val quotedTweet = tweet.quotedTweet + val mentions = tweet.mentions.getOrElse(Seq.empty) + val share = coreData.flatMap(_.share) + val reply = coreData.flatMap(_.reply) + val semanticAnnotations = + tweet.escherbirdEntityAnnotations.map(_.entityAnnotations).getOrElse(Seq.empty) + + FeatureMapBuilder() + .add(AuthorIdFeature, coreData.map(_.userId)) + .add(DirectedAtUserIdFeature, coreData.flatMap(_.directedAtUser.map(_.userId))) + .add(HasImageFeature, TweetMediaFeaturesExtractor.hasImage(tweet)) + .add(HasVideoFeature, TweetMediaFeaturesExtractor.hasVideo(tweet)) + .add(InReplyToTweetIdFeature, reply.flatMap(_.inReplyToStatusId)) + .add(InReplyToUserIdFeature, reply.map(_.inReplyToUserId)) + .add(IsRetweetFeature, share.isDefined) + .add(MentionScreenNameFeature, mentions.map(_.screenName)) + .add(MentionUserIdFeature, mentions.flatMap(_.userId)) + .add(QuotedTweetIdFeature, quotedTweet.map(_.tweetId)) + .add(QuotedUserIdFeature, quotedTweet.map(_.userId)) + .add(SemanticAnnotationFeature, semanticAnnotations) + .add(SourceTweetIdFeature, share.map(_.sourceStatusId)) + .add(SourceUserIdFeature, share.map(_.sourceUserId)) + .build() + } + + private def readFromTweetypie( + query: PipelineQuery, + candidate: CandidateWithFeatures[TweetCandidate] + ): Stitch[FeatureMap] = { + tweetypieStitchClient + .getTweetFields( + tweetId = candidate.candidate.id, + options = tp.GetTweetFieldsOptions( + tweetIncludes = RequestFields.TweetStaticEntitiesFields, + includeRetweetedTweet = false, + includeQuotedTweet = false, + forUserId = query.getOptionalUserId, // Needed to get protected Tweets for certain users + visibilityPolicy = tp.TweetVisibilityPolicy.UserVisible, + safetyLevel = Some(sp.SafetyLevel.FilterNone) // VF is handled in the For You product + ) + ).map { + case tp.GetTweetFieldsResult(_, tp.TweetFieldsResultState.Found(found), _, _) => + cacheClient.set(candidate.candidate.id, found.tweet, CacheTTL) + createFeatureMap(found.tweet) + case _ => + DefaultFeatureMap + (AuthorIdFeature, candidate.features.getOrElse(AuthorIdFeature, None)) + } + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/TwhinAuthorFollow20220101FeatureHydrator.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/TwhinAuthorFollow20220101FeatureHydrator.scala new file mode 100644 index 0000000000..5d928a90bb --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/TwhinAuthorFollow20220101FeatureHydrator.scala @@ -0,0 +1,96 @@ +package com.twitter.home_mixer.functional_component.feature_hydrator + +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.home_mixer.functional_component.feature_hydrator.adapters.twhin_embeddings.TwhinAuthorFollowEmbeddingsAdapter +import com.twitter.home_mixer.model.HomeFeatures.AuthorIdFeature +import com.twitter.home_mixer.param.HomeMixerInjectionNames.TwhinAuthorFollow20200101FeatureRepository +import com.twitter.home_mixer.util.ObservedKeyValueResultHandler +import com.twitter.ml.api.DataRecord +import com.twitter.ml.api.{thriftscala => ml} +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.feature.Feature +import com.twitter.product_mixer.core.feature.FeatureWithDefaultOnFailure +import com.twitter.product_mixer.core.feature.datarecord.DataRecordInAFeature +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.feature.featuremap.FeatureMapBuilder +import com.twitter.product_mixer.core.functional_component.feature_hydrator.BulkCandidateFeatureHydrator +import com.twitter.product_mixer.core.model.common.CandidateWithFeatures +import com.twitter.product_mixer.core.model.common.identifier.FeatureHydratorIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.servo.repository.KeyValueRepository +import com.twitter.servo.repository.KeyValueResult +import com.twitter.stitch.Stitch +import com.twitter.util.Future +import com.twitter.util.Try + +import javax.inject.Inject +import javax.inject.Named +import javax.inject.Singleton +import scala.collection.JavaConverters._ + +object TwhinAuthorFollow20220101Feature + extends DataRecordInAFeature[TweetCandidate] + with FeatureWithDefaultOnFailure[TweetCandidate, DataRecord] { + override def defaultValue: DataRecord = new DataRecord() +} + +@Singleton +class TwhinAuthorFollow20220101FeatureHydrator @Inject() ( + @Named(TwhinAuthorFollow20200101FeatureRepository) + client: KeyValueRepository[Seq[Long], Long, ml.Embedding], + override val statsReceiver: StatsReceiver) + extends BulkCandidateFeatureHydrator[PipelineQuery, TweetCandidate] + with ObservedKeyValueResultHandler { + + override val identifier: FeatureHydratorIdentifier = + FeatureHydratorIdentifier("TwhinAuthorFollow20220101") + + override val features: Set[Feature[_, _]] = Set(TwhinAuthorFollow20220101Feature) + + override val statScope: String = identifier.toString + + override def apply( + query: PipelineQuery, + candidates: Seq[CandidateWithFeatures[TweetCandidate]] + ): Stitch[Seq[FeatureMap]] = { + Stitch.callFuture { + val possiblyAuthorIds = extractKeys(candidates) + val authorIds = possiblyAuthorIds.flatten + + val response: Future[KeyValueResult[Long, ml.Embedding]] = + if (authorIds.isEmpty) { + Future.value(KeyValueResult.empty) + } else { + client(authorIds) + } + + response.map { result => + possiblyAuthorIds.map { possiblyAuthorId => + val value = observedGet(key = possiblyAuthorId, keyValueResult = result) + val transformedValue = postTransformer(value) + + FeatureMapBuilder() + .add(TwhinAuthorFollow20220101Feature, transformedValue) + .build() + } + } + } + } + + private def postTransformer(embedding: Try[Option[ml.Embedding]]): Try[DataRecord] = { + embedding.map { e => + TwhinAuthorFollowEmbeddingsAdapter.adaptToDataRecords(e).asScala.head + } + } + + private def extractKeys( + candidates: Seq[CandidateWithFeatures[TweetCandidate]] + ): Seq[Option[Long]] = { + candidates.map { candidate => + candidate.features + .getTry(AuthorIdFeature) + .toOption + .flatten + } + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/TwhinUserEngagementQueryFeatureHydrator.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/TwhinUserEngagementQueryFeatureHydrator.scala new file mode 100644 index 0000000000..bc602d90cb --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/TwhinUserEngagementQueryFeatureHydrator.scala @@ -0,0 +1,80 @@ +package com.twitter.home_mixer.functional_component.feature_hydrator + +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.home_mixer.functional_component.feature_hydrator.adapters.twhin_embeddings.TwhinUserEngagementEmbeddingsAdapter +import com.twitter.home_mixer.param.HomeMixerInjectionNames.TwhinUserEngagementFeatureRepository +import com.twitter.ml.api.DataRecord +import com.twitter.ml.api.RichDataRecord +import com.twitter.ml.api.util.ScalaToJavaDataRecordConversions +import com.twitter.ml.api.{thriftscala => ml} +import com.twitter.product_mixer.core.feature.datarecord.DataRecordInAFeature +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.feature.featuremap.FeatureMapBuilder +import com.twitter.product_mixer.core.feature.Feature +import com.twitter.product_mixer.core.feature.FeatureWithDefaultOnFailure +import com.twitter.product_mixer.core.functional_component.feature_hydrator.QueryFeatureHydrator +import com.twitter.product_mixer.core.model.common.identifier.FeatureHydratorIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.servo.repository.KeyValueRepository +import com.twitter.stitch.Stitch +import com.twitter.util.Return +import com.twitter.util.Throw +import javax.inject.Inject +import javax.inject.Named +import javax.inject.Singleton + +object TwhinUserEngagementFeature + extends DataRecordInAFeature[PipelineQuery] + with FeatureWithDefaultOnFailure[PipelineQuery, DataRecord] { + override def defaultValue: DataRecord = new DataRecord() +} + +@Singleton +class TwhinUserEngagementQueryFeatureHydrator @Inject() ( + @Named(TwhinUserEngagementFeatureRepository) + client: KeyValueRepository[Seq[Long], Long, ml.FloatTensor], + statsReceiver: StatsReceiver) + extends QueryFeatureHydrator[PipelineQuery] { + + override val identifier: FeatureHydratorIdentifier = + FeatureHydratorIdentifier("TwhinUserEngagement") + + override val features: Set[Feature[_, _]] = Set(TwhinUserEngagementFeature) + + private val scopedStatsReceiver = statsReceiver.scope(getClass.getSimpleName) + private val keyFoundCounter = scopedStatsReceiver.counter("key/found") + private val keyLossCounter = scopedStatsReceiver.counter("key/loss") + private val keyFailureCounter = scopedStatsReceiver.counter("key/failure") + + override def hydrate(query: PipelineQuery): Stitch[FeatureMap] = { + val userId = query.getRequiredUserId + Stitch.callFuture { + client(Seq(userId)).map { results => + val embedding: Option[ml.FloatTensor] = results(userId) match { + case Return(value) => + if (value.exists(_.floats.nonEmpty)) keyFoundCounter.incr() + else keyLossCounter.incr() + value + case Throw(_) => + keyFailureCounter.incr() + None + case _ => + None + } + val dataRecord = + new RichDataRecord(new DataRecord, TwhinUserEngagementEmbeddingsAdapter.getFeatureContext) + embedding.foreach { floatTensor => + dataRecord.setFeatureValue( + TwhinUserEngagementEmbeddingsAdapter.twhinEmbeddingsFeature, + ScalaToJavaDataRecordConversions.scalaTensor2Java( + ml.GeneralTensor.FloatTensor(floatTensor)) + ) + } + + FeatureMapBuilder() + .add(TwhinUserEngagementFeature, dataRecord.getRecord) + .build() + } + } + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/TwhinUserFollowQueryFeatureHydrator.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/TwhinUserFollowQueryFeatureHydrator.scala new file mode 100644 index 0000000000..c0efba1673 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/TwhinUserFollowQueryFeatureHydrator.scala @@ -0,0 +1,80 @@ +package com.twitter.home_mixer.functional_component.feature_hydrator + +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.home_mixer.functional_component.feature_hydrator.adapters.twhin_embeddings.TwhinUserFollowEmbeddingsAdapter +import com.twitter.home_mixer.param.HomeMixerInjectionNames.TwhinUserFollowFeatureRepository +import com.twitter.ml.api.DataRecord +import com.twitter.ml.api.RichDataRecord + +import com.twitter.ml.api.util.ScalaToJavaDataRecordConversions +import com.twitter.ml.api.{thriftscala => ml} +import com.twitter.product_mixer.core.feature.datarecord.DataRecordInAFeature +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.feature.featuremap.FeatureMapBuilder +import com.twitter.product_mixer.core.feature.Feature +import com.twitter.product_mixer.core.feature.FeatureWithDefaultOnFailure +import com.twitter.product_mixer.core.functional_component.feature_hydrator.QueryFeatureHydrator +import com.twitter.product_mixer.core.model.common.identifier.FeatureHydratorIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.servo.repository.KeyValueRepository +import com.twitter.stitch.Stitch +import com.twitter.util.Return +import com.twitter.util.Throw +import javax.inject.Inject +import javax.inject.Named +import javax.inject.Singleton + +object TwhinUserFollowFeature + extends DataRecordInAFeature[PipelineQuery] + with FeatureWithDefaultOnFailure[PipelineQuery, DataRecord] { + override def defaultValue: DataRecord = new DataRecord() +} + +@Singleton +class TwhinUserFollowQueryFeatureHydrator @Inject() ( + @Named(TwhinUserFollowFeatureRepository) + client: KeyValueRepository[Seq[Long], Long, ml.FloatTensor], + statsReceiver: StatsReceiver) + extends QueryFeatureHydrator[PipelineQuery] { + + override val identifier: FeatureHydratorIdentifier = + FeatureHydratorIdentifier("TwhinUserFollow") + + override val features: Set[Feature[_, _]] = Set(TwhinUserFollowFeature) + + private val scopedStatsReceiver = statsReceiver.scope(getClass.getSimpleName) + private val keyFoundCounter = scopedStatsReceiver.counter("key/found") + private val keyLossCounter = scopedStatsReceiver.counter("key/loss") + private val keyFailureCounter = scopedStatsReceiver.counter("key/failure") + + override def hydrate(query: PipelineQuery): Stitch[FeatureMap] = { + val userId = query.getRequiredUserId + Stitch.callFuture( + client(Seq(userId)).map { results => + val embedding: Option[ml.FloatTensor] = results(userId) match { + case Return(value) => + if (value.exists(_.floats.nonEmpty)) keyFoundCounter.incr() + else keyLossCounter.incr() + value + case Throw(_) => + keyFailureCounter.incr() + None + case _ => + None + } + val dataRecord = + new RichDataRecord(new DataRecord, TwhinUserFollowEmbeddingsAdapter.getFeatureContext) + embedding.foreach { floatTensor => + dataRecord.setFeatureValue( + TwhinUserFollowEmbeddingsAdapter.twhinEmbeddingsFeature, + ScalaToJavaDataRecordConversions.scalaTensor2Java( + ml.GeneralTensor + .FloatTensor(floatTensor))) + } + FeatureMapBuilder() + .add(TwhinUserFollowFeature, dataRecord.getRecord) + .build() + } + ) + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/UserFollowedTopicIdsFeatureHydrator.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/UserFollowedTopicIdsFeatureHydrator.scala new file mode 100644 index 0000000000..5aed770a1b --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/UserFollowedTopicIdsFeatureHydrator.scala @@ -0,0 +1,84 @@ +package com.twitter.home_mixer.functional_component.feature_hydrator + +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.home_mixer.model.HomeFeatures.AuthorIdFeature +import com.twitter.home_mixer.param.HomeMixerInjectionNames.UserFollowedTopicIdsRepository +import com.twitter.home_mixer.util.ObservedKeyValueResultHandler +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.feature.Feature +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.feature.featuremap.FeatureMapBuilder +import com.twitter.product_mixer.core.functional_component.feature_hydrator.BulkCandidateFeatureHydrator +import com.twitter.product_mixer.core.model.common.CandidateWithFeatures +import com.twitter.product_mixer.core.model.common.identifier.FeatureHydratorIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.servo.keyvalue.KeyValueResult +import com.twitter.servo.repository.KeyValueRepository +import com.twitter.stitch.Stitch +import com.twitter.util.Future +import com.twitter.util.Try + +import javax.inject.Inject +import javax.inject.Named +import javax.inject.Singleton + +object UserFollowedTopicIdsFeature extends Feature[TweetCandidate, Seq[Long]] + +@Singleton +class UserFollowedTopicIdsFeatureHydrator @Inject() ( + @Named(UserFollowedTopicIdsRepository) + client: KeyValueRepository[Seq[Long], Long, Seq[Long]], + override val statsReceiver: StatsReceiver) + extends BulkCandidateFeatureHydrator[PipelineQuery, TweetCandidate] + with ObservedKeyValueResultHandler { + + override val identifier: FeatureHydratorIdentifier = + FeatureHydratorIdentifier("UserFollowedTopicIds") + + override val features: Set[Feature[_, _]] = Set(UserFollowedTopicIdsFeature) + + override val statScope: String = identifier.toString + + override def apply( + query: PipelineQuery, + candidates: Seq[CandidateWithFeatures[TweetCandidate]] + ): Stitch[Seq[FeatureMap]] = { + Stitch.callFuture { + val possiblyAuthorIds = extractKeys(candidates) + val authorIds = possiblyAuthorIds.flatten + + val response: Future[KeyValueResult[Long, Seq[Long]]] = + if (authorIds.isEmpty) { + Future.value(KeyValueResult.empty) + } else { + client(authorIds) + } + + response.map { result => + possiblyAuthorIds.map { possiblyAuthorId => + val value = observedGet(key = possiblyAuthorId, keyValueResult = result) + val transformedValue = postTransformer(value) + + FeatureMapBuilder() + .add(UserFollowedTopicIdsFeature, transformedValue) + .build() + } + } + } + } + + private def postTransformer(input: Try[Option[Seq[Long]]]): Try[Seq[Long]] = { + input.map(_.getOrElse(Seq.empty[Long])) + } + + private def extractKeys( + candidates: Seq[CandidateWithFeatures[TweetCandidate]] + ): Seq[Option[Long]] = { + candidates.map { candidate => + candidate.features + .getTry(AuthorIdFeature) + .toOption + .flatten + } + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/UserLanguagesFeatureHydrator.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/UserLanguagesFeatureHydrator.scala new file mode 100644 index 0000000000..ad97f3a236 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/UserLanguagesFeatureHydrator.scala @@ -0,0 +1,35 @@ +package com.twitter.home_mixer.functional_component.feature_hydrator + +import com.twitter.home_mixer.param.HomeMixerInjectionNames.UserLanguagesStore +import com.twitter.product_mixer.core.feature.Feature +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.feature.featuremap.FeatureMapBuilder +import com.twitter.product_mixer.core.functional_component.feature_hydrator.QueryFeatureHydrator +import com.twitter.product_mixer.core.model.common.identifier.FeatureHydratorIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.search.common.constants.{thriftscala => scc} +import com.twitter.stitch.Stitch +import com.twitter.storehaus.ReadableStore +import javax.inject.Inject +import javax.inject.Named +import javax.inject.Singleton + +object UserLanguagesFeature extends Feature[PipelineQuery, Seq[scc.ThriftLanguage]] + +@Singleton +case class UserLanguagesFeatureHydrator @Inject() ( + @Named(UserLanguagesStore) store: ReadableStore[Long, Seq[scc.ThriftLanguage]]) + extends QueryFeatureHydrator[PipelineQuery] { + + override val identifier: FeatureHydratorIdentifier = FeatureHydratorIdentifier("UserLanguages") + + override val features: Set[Feature[_, _]] = Set(UserLanguagesFeature) + + override def hydrate(query: PipelineQuery): Stitch[FeatureMap] = { + Stitch.callFuture(store.get(query.getRequiredUserId)).map { languages => + FeatureMapBuilder() + .add(UserLanguagesFeature, languages.getOrElse(Seq.empty)) + .build() + } + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/UserStateQueryFeatureHydrator.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/UserStateQueryFeatureHydrator.scala new file mode 100644 index 0000000000..29532c8e17 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/UserStateQueryFeatureHydrator.scala @@ -0,0 +1,54 @@ +package com.twitter.home_mixer.functional_component.feature_hydrator + +import com.twitter.home_mixer.model.HomeFeatures.UserStateFeature +import com.twitter.home_mixer.service.HomeMixerAlertConfig +import com.twitter.product_mixer.core.feature.Feature +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.feature.featuremap.FeatureMapBuilder +import com.twitter.product_mixer.core.functional_component.feature_hydrator.QueryFeatureHydrator +import com.twitter.product_mixer.core.model.common.identifier.FeatureHydratorIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.stitch.Stitch +import com.twitter.timelines.user_health.{thriftscala => uh} +import com.twitter.timelines.user_health.v1.{thriftscala => uhv1} +import com.twitter.user_session_store.ReadOnlyUserSessionStore +import com.twitter.user_session_store.ReadRequest +import com.twitter.user_session_store.UserSessionDataset +import com.twitter.user_session_store.UserSessionDataset.UserSessionDataset + +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +case class UserStateQueryFeatureHydrator @Inject() ( + userSessionStore: ReadOnlyUserSessionStore) + extends QueryFeatureHydrator[PipelineQuery] { + + override val identifier: FeatureHydratorIdentifier = + FeatureHydratorIdentifier("UserState") + + override val features: Set[Feature[_, _]] = Set(UserStateFeature) + + private val datasets: Set[UserSessionDataset] = Set(UserSessionDataset.UserHealth) + + override def hydrate(query: PipelineQuery): Stitch[FeatureMap] = { + userSessionStore + .read(ReadRequest(query.getRequiredUserId, datasets)) + .map { userSession => + val userState = userSession.flatMap { + _.userHealth match { + case Some(uh.UserHealth.V1(uhv1.UserHealth(userState))) => userState + case _ => None + } + } + + FeatureMapBuilder() + .add(UserStateFeature, userState) + .build() + } + } + + override val alerts = Seq( + HomeMixerAlertConfig.BusinessHours.defaultSuccessRateAlert(99.9) + ) +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/UtegFeatureHydrator.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/UtegFeatureHydrator.scala new file mode 100644 index 0000000000..389a50d032 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/UtegFeatureHydrator.scala @@ -0,0 +1,88 @@ +package com.twitter.home_mixer.functional_component.feature_hydrator + +import com.twitter.home_mixer.model.HomeFeatures.FavoritedByUserIdsFeature +import com.twitter.home_mixer.model.HomeFeatures.InReplyToTweetIdFeature +import com.twitter.home_mixer.model.HomeFeatures.RealGraphInNetworkScoresFeature +import com.twitter.home_mixer.model.HomeFeatures.RepliedByEngagerIdsFeature +import com.twitter.home_mixer.model.HomeFeatures.RetweetedByEngagerIdsFeature +import com.twitter.home_mixer.model.HomeFeatures.SourceTweetIdFeature +import com.twitter.home_mixer.param.HomeMixerInjectionNames.UtegSocialProofRepository +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.feature.Feature +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.feature.featuremap.FeatureMapBuilder +import com.twitter.product_mixer.core.functional_component.feature_hydrator.BulkCandidateFeatureHydrator +import com.twitter.product_mixer.core.model.common.CandidateWithFeatures +import com.twitter.product_mixer.core.model.common.Conditionally +import com.twitter.product_mixer.core.model.common.identifier.FeatureHydratorIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.recos.recos_common.{thriftscala => rc} +import com.twitter.recos.user_tweet_entity_graph.{thriftscala => uteg} +import com.twitter.servo.keyvalue.KeyValueResult +import com.twitter.servo.repository.KeyValueRepository +import com.twitter.stitch.Stitch + +import javax.inject.Inject +import javax.inject.Named +import javax.inject.Singleton + +@Singleton +class UtegFeatureHydrator @Inject() ( + @Named(UtegSocialProofRepository) client: KeyValueRepository[ + (Seq[Long], (Long, Map[Long, Double])), + Long, + uteg.TweetRecommendation + ]) extends BulkCandidateFeatureHydrator[PipelineQuery, TweetCandidate] + with Conditionally[PipelineQuery] { + + override val identifier: FeatureHydratorIdentifier = FeatureHydratorIdentifier("Uteg") + + override val features: Set[Feature[_, _]] = Set( + FavoritedByUserIdsFeature, + RetweetedByEngagerIdsFeature, + RepliedByEngagerIdsFeature + ) + + override def onlyIf(query: PipelineQuery): Boolean = query.features + .exists(_.getOrElse(RealGraphInNetworkScoresFeature, Map.empty[Long, Double]).nonEmpty) + + override def apply( + query: PipelineQuery, + candidates: Seq[CandidateWithFeatures[TweetCandidate]] + ): Stitch[Seq[FeatureMap]] = { + val seedUserWeights = query.features.map(_.get(RealGraphInNetworkScoresFeature)).get + + val sourceTweetIds = candidates.flatMap(_.features.getOrElse(SourceTweetIdFeature, None)) + val inReplyToTweetIds = candidates.flatMap(_.features.getOrElse(InReplyToTweetIdFeature, None)) + val tweetIds = candidates.map(_.candidate.id) + val tweetIdsToSend = (tweetIds ++ sourceTweetIds ++ inReplyToTweetIds).distinct + + val utegQuery = (tweetIdsToSend, (query.getRequiredUserId, seedUserWeights)) + + Stitch + .callFuture(client(utegQuery)) + .map(handleResponse(candidates, _)) + } + + private def handleResponse( + candidates: Seq[CandidateWithFeatures[TweetCandidate]], + results: KeyValueResult[Long, uteg.TweetRecommendation], + ): Seq[FeatureMap] = { + candidates.map { candidate => + val candidateProof = results(candidate.candidate.id).toOption.flatten + val sourceProof = candidate.features + .getOrElse(SourceTweetIdFeature, None).flatMap(results(_).toOption.flatten) + val proofs = Seq(candidateProof, sourceProof).flatten.map(_.socialProofByType) + + val favoritedBy = proofs.flatMap(_.get(rc.SocialProofType.Favorite)).flatten + val retweetedBy = proofs.flatMap(_.get(rc.SocialProofType.Retweet)).flatten + val repliedBy = proofs.flatMap(_.get(rc.SocialProofType.Reply)).flatten + + FeatureMapBuilder() + .add(FavoritedByUserIdsFeature, favoritedBy) + .add(RetweetedByEngagerIdsFeature, retweetedBy) + .add(RepliedByEngagerIdsFeature, repliedBy) + .build() + } + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/adapters/author_features/AuthorFeaturesAdapter.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/adapters/author_features/AuthorFeaturesAdapter.scala new file mode 100644 index 0000000000..06fc42a538 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/adapters/author_features/AuthorFeaturesAdapter.scala @@ -0,0 +1,59 @@ +package com.twitter.home_mixer.functional_component.feature_hydrator.adapters.author_features + +import com.twitter.ml.api.DataRecordMerger +import com.twitter.ml.api.Feature +import com.twitter.ml.api.FeatureContext +import com.twitter.ml.api.RichDataRecord +import com.twitter.ml.api.util.CompactDataRecordConverter +import com.twitter.ml.api.util.FDsl._ +import com.twitter.timelines.author_features.v1.{thriftjava => af} +import com.twitter.timelines.prediction.common.adapters.TimelinesMutatingAdapterBase +import com.twitter.timelines.prediction.common.aggregates.TimelinesAggregationConfig +import com.twitter.timelines.prediction.features.user_health.UserHealthFeatures + +object AuthorFeaturesAdapter extends TimelinesMutatingAdapterBase[Option[af.AuthorFeatures]] { + + private val originalAuthorAggregatesFeatures = + TimelinesAggregationConfig.originalAuthorReciprocalEngagementAggregates + .buildTypedAggregateGroups().flatMap(_.allOutputFeatures) + private val authorFeatures = originalAuthorAggregatesFeatures ++ + Seq( + UserHealthFeatures.AuthorState, + UserHealthFeatures.NumAuthorFollowers, + UserHealthFeatures.NumAuthorConnectDays, + UserHealthFeatures.NumAuthorConnect) + private val featureContext = new FeatureContext(authorFeatures: _*) + + override def getFeatureContext: FeatureContext = featureContext + + override val commonFeatures: Set[Feature[_]] = Set.empty + + private val compactDataRecordConverter = new CompactDataRecordConverter() + private val drMerger = new DataRecordMerger() + + override def setFeatures( + authorFeaturesOpt: Option[af.AuthorFeatures], + richDataRecord: RichDataRecord + ): Unit = { + authorFeaturesOpt.foreach { authorFeatures => + val dataRecord = richDataRecord.getRecord + + dataRecord.setFeatureValue( + UserHealthFeatures.AuthorState, + authorFeatures.user_health.user_state.getValue.toLong) + dataRecord.setFeatureValue( + UserHealthFeatures.NumAuthorFollowers, + authorFeatures.user_health.num_followers.toDouble) + dataRecord.setFeatureValue( + UserHealthFeatures.NumAuthorConnectDays, + authorFeatures.user_health.num_connect_days.toDouble) + dataRecord.setFeatureValue( + UserHealthFeatures.NumAuthorConnect, + authorFeatures.user_health.num_connect.toDouble) + + val originalAuthorAggregatesDataRecord = + compactDataRecordConverter.compactDataRecordToDataRecord(authorFeatures.aggregates) + drMerger.merge(dataRecord, originalAuthorAggregatesDataRecord) + } + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/adapters/author_features/BUILD.bazel b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/adapters/author_features/BUILD.bazel new file mode 100644 index 0000000000..ad8e1cd827 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/adapters/author_features/BUILD.bazel @@ -0,0 +1,17 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + strict_deps = True, + tags = ["bazel-compatible"], + dependencies = [ + "src/java/com/twitter/ml/api:api-base", + "src/java/com/twitter/ml/api/util", + "src/scala/com/twitter/ml/api/util", + "src/scala/com/twitter/timelines/prediction/common/adapters:base", + "src/scala/com/twitter/timelines/prediction/common/aggregates", + "src/scala/com/twitter/timelines/prediction/features/user_health", + "src/thrift/com/twitter/ml/api:data-java", + "src/thrift/com/twitter/timelines/author_features:thrift-java", + "timelines/data_processing/ml_util/aggregation_framework", + ], +) diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/adapters/content/BUILD.bazel b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/adapters/content/BUILD.bazel new file mode 100644 index 0000000000..bf9d7e2b80 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/adapters/content/BUILD.bazel @@ -0,0 +1,17 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + strict_deps = True, + tags = ["bazel-compatible"], + dependencies = [ + "home-mixer/server/src/main/scala/com/twitter/home_mixer/model", + "src/java/com/twitter/ml/api:api-base", + "src/scala/com/twitter/ml/api:api-base", + "src/scala/com/twitter/ml/api/util", + "src/scala/com/twitter/timelines/prediction/common/adapters", + "src/scala/com/twitter/timelines/prediction/common/adapters:base", + "src/scala/com/twitter/timelines/prediction/features/common", + "src/thrift/com/twitter/ml/api:data-java", + "src/thrift/com/twitter/ml/api:data-scala", + ], +) diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/adapters/content/ContentFeatureAdapter.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/adapters/content/ContentFeatureAdapter.scala new file mode 100644 index 0000000000..93cb6036d8 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/adapters/content/ContentFeatureAdapter.scala @@ -0,0 +1,260 @@ +package com.twitter.home_mixer.functional_component.feature_hydrator.adapters.content + +import com.twitter.home_mixer.model.ContentFeatures +import com.twitter.ml.api.Feature +import com.twitter.ml.api.FeatureContext +import com.twitter.ml.api.RichDataRecord +import com.twitter.ml.api.util.DataRecordConverters._ +import com.twitter.timelines.prediction.common.adapters.TimelinesMutatingAdapterBase +import com.twitter.timelines.prediction.common.adapters.TweetLengthType +import com.twitter.timelines.prediction.features.common.TimelinesSharedFeatures +import scala.collection.JavaConverters._ + +object ContentFeatureAdapter extends TimelinesMutatingAdapterBase[Option[ContentFeatures]] { + + override val getFeatureContext: FeatureContext = new FeatureContext( + TimelinesSharedFeatures.ASPECT_RATIO_DEN, + TimelinesSharedFeatures.ASPECT_RATIO_NUM, + TimelinesSharedFeatures.BIT_RATE, + TimelinesSharedFeatures.CLASSIFICATION_LABELS, + TimelinesSharedFeatures.COLOR_1_BLUE, + TimelinesSharedFeatures.COLOR_1_GREEN, + TimelinesSharedFeatures.COLOR_1_PERCENTAGE, + TimelinesSharedFeatures.COLOR_1_RED, + TimelinesSharedFeatures.FACE_AREAS, + TimelinesSharedFeatures.HAS_APP_INSTALL_CALL_TO_ACTION, + TimelinesSharedFeatures.HAS_DESCRIPTION, + TimelinesSharedFeatures.HAS_QUESTION, + TimelinesSharedFeatures.HAS_SELECTED_PREVIEW_IMAGE, + TimelinesSharedFeatures.HAS_TITLE, + TimelinesSharedFeatures.HAS_VISIT_SITE_CALL_TO_ACTION, + TimelinesSharedFeatures.HAS_WATCH_NOW_CALL_TO_ACTION, + TimelinesSharedFeatures.HEIGHT_1, + TimelinesSharedFeatures.HEIGHT_2, + TimelinesSharedFeatures.HEIGHT_3, + TimelinesSharedFeatures.HEIGHT_4, + TimelinesSharedFeatures.IS_360, + TimelinesSharedFeatures.IS_EMBEDDABLE, + TimelinesSharedFeatures.IS_MANAGED, + TimelinesSharedFeatures.IS_MONETIZABLE, + TimelinesSharedFeatures.MEDIA_PROVIDERS, + TimelinesSharedFeatures.NUM_CAPS, + TimelinesSharedFeatures.NUM_COLOR_PALLETTE_ITEMS, + TimelinesSharedFeatures.NUM_FACES, + TimelinesSharedFeatures.NUM_MEDIA_TAGS, + TimelinesSharedFeatures.NUM_NEWLINES, + TimelinesSharedFeatures.NUM_STICKERS, + TimelinesSharedFeatures.NUM_WHITESPACES, + TimelinesSharedFeatures.RESIZE_METHOD_1, + TimelinesSharedFeatures.RESIZE_METHOD_2, + TimelinesSharedFeatures.RESIZE_METHOD_3, + TimelinesSharedFeatures.RESIZE_METHOD_4, + TimelinesSharedFeatures.TWEET_LENGTH, + TimelinesSharedFeatures.TWEET_LENGTH_TYPE, + TimelinesSharedFeatures.VIDEO_DURATION, + TimelinesSharedFeatures.VIEW_COUNT, + TimelinesSharedFeatures.WIDTH_1, + TimelinesSharedFeatures.WIDTH_2, + TimelinesSharedFeatures.WIDTH_3, + TimelinesSharedFeatures.WIDTH_4, + ) + + override val commonFeatures: Set[Feature[_]] = Set.empty + + private def getTweetLengthType(tweetLength: Int): Long = { + tweetLength match { + case x if 0 > x || 280 < x => TweetLengthType.INVALID + case x if 0 <= x && x <= 30 => TweetLengthType.VERY_SHORT + case x if 30 < x && x <= 60 => TweetLengthType.SHORT + case x if 60 < x && x <= 90 => TweetLengthType.MEDIUM + case x if 90 < x && x <= 140 => TweetLengthType.LENGTHY + case x if 140 < x && x <= 210 => TweetLengthType.VERY_LENGTHY + case x if x > 210 => TweetLengthType.MAXIMUM_LENGTH + } + } + + override def setFeatures( + contentFeatures: Option[ContentFeatures], + richDataRecord: RichDataRecord + ): Unit = { + if (contentFeatures.nonEmpty) { + val features = contentFeatures.get + // Media Features + richDataRecord.setFeatureValueFromOption( + TimelinesSharedFeatures.ASPECT_RATIO_DEN, + features.aspectRatioDen.map(_.toDouble) + ) + richDataRecord.setFeatureValueFromOption( + TimelinesSharedFeatures.ASPECT_RATIO_NUM, + features.aspectRatioNum.map(_.toDouble) + ) + richDataRecord.setFeatureValueFromOption( + TimelinesSharedFeatures.BIT_RATE, + features.bitRate.map(_.toDouble) + ) + richDataRecord.setFeatureValueFromOption( + TimelinesSharedFeatures.HEIGHT_1, + features.heights.flatMap(_.lift(0)).map(_.toDouble) + ) + richDataRecord.setFeatureValueFromOption( + TimelinesSharedFeatures.HEIGHT_2, + features.heights.flatMap(_.lift(1)).map(_.toDouble) + ) + richDataRecord.setFeatureValueFromOption( + TimelinesSharedFeatures.HEIGHT_3, + features.heights.flatMap(_.lift(2)).map(_.toDouble) + ) + richDataRecord.setFeatureValueFromOption( + TimelinesSharedFeatures.HEIGHT_4, + features.heights.flatMap(_.lift(3)).map(_.toDouble) + ) + richDataRecord.setFeatureValueFromOption( + TimelinesSharedFeatures.NUM_MEDIA_TAGS, + features.numMediaTags.map(_.toDouble) + ) + richDataRecord.setFeatureValueFromOption( + TimelinesSharedFeatures.RESIZE_METHOD_1, + features.resizeMethods.flatMap(_.lift(0)).map(_.toLong) + ) + richDataRecord.setFeatureValueFromOption( + TimelinesSharedFeatures.RESIZE_METHOD_2, + features.resizeMethods.flatMap(_.lift(1)).map(_.toLong) + ) + richDataRecord.setFeatureValueFromOption( + TimelinesSharedFeatures.RESIZE_METHOD_3, + features.resizeMethods.flatMap(_.lift(2)).map(_.toLong) + ) + richDataRecord.setFeatureValueFromOption( + TimelinesSharedFeatures.RESIZE_METHOD_4, + features.resizeMethods.flatMap(_.lift(3)).map(_.toLong) + ) + richDataRecord.setFeatureValueFromOption( + TimelinesSharedFeatures.VIDEO_DURATION, + features.videoDurationMs.map(_.toDouble) + ) + richDataRecord.setFeatureValueFromOption( + TimelinesSharedFeatures.WIDTH_1, + features.widths.flatMap(_.lift(0)).map(_.toDouble) + ) + richDataRecord.setFeatureValueFromOption( + TimelinesSharedFeatures.WIDTH_2, + features.widths.flatMap(_.lift(1)).map(_.toDouble) + ) + richDataRecord.setFeatureValueFromOption( + TimelinesSharedFeatures.WIDTH_3, + features.widths.flatMap(_.lift(2)).map(_.toDouble) + ) + richDataRecord.setFeatureValueFromOption( + TimelinesSharedFeatures.WIDTH_4, + features.widths.flatMap(_.lift(3)).map(_.toDouble) + ) + richDataRecord.setFeatureValueFromOption( + TimelinesSharedFeatures.NUM_COLOR_PALLETTE_ITEMS, + features.numColors.map(_.toDouble) + ) + richDataRecord.setFeatureValueFromOption( + TimelinesSharedFeatures.COLOR_1_RED, + features.dominantColorRed.map(_.toDouble) + ) + richDataRecord.setFeatureValueFromOption( + TimelinesSharedFeatures.COLOR_1_BLUE, + features.dominantColorBlue.map(_.toDouble) + ) + richDataRecord.setFeatureValueFromOption( + TimelinesSharedFeatures.COLOR_1_GREEN, + features.dominantColorGreen.map(_.toDouble) + ) + richDataRecord.setFeatureValueFromOption( + TimelinesSharedFeatures.COLOR_1_PERCENTAGE, + features.dominantColorPercentage + ) + richDataRecord.setFeatureValueFromOption( + TimelinesSharedFeatures.MEDIA_PROVIDERS, + features.mediaOriginProviders.map(_.toSet.asJava) + ) + richDataRecord.setFeatureValueFromOption( + TimelinesSharedFeatures.IS_360, + features.is360 + ) + richDataRecord.setFeatureValueFromOption( + TimelinesSharedFeatures.VIEW_COUNT, + features.viewCount.map(_.toDouble) + ) + richDataRecord.setFeatureValueFromOption( + TimelinesSharedFeatures.IS_MANAGED, + features.isManaged + ) + richDataRecord.setFeatureValueFromOption( + TimelinesSharedFeatures.IS_MONETIZABLE, + features.isMonetizable + ) + richDataRecord.setFeatureValueFromOption( + TimelinesSharedFeatures.IS_EMBEDDABLE, + features.isEmbeddable + ) + richDataRecord.setFeatureValueFromOption( + TimelinesSharedFeatures.NUM_STICKERS, + features.stickerIds.map(_.length.toDouble) + ) + richDataRecord.setFeatureValueFromOption( + TimelinesSharedFeatures.NUM_FACES, + features.faceAreas.map(_.length.toDouble) + ) + richDataRecord.setFeatureValueFromOption( + TimelinesSharedFeatures.FACE_AREAS, + // guard for exception from max on empty seq + features.faceAreas.map(faceAreas => + faceAreas.map(_.toDouble).reduceOption(_ max _).getOrElse(0.0)) + ) + richDataRecord.setFeatureValueFromOption( + TimelinesSharedFeatures.HAS_SELECTED_PREVIEW_IMAGE, + features.hasSelectedPreviewImage + ) + richDataRecord.setFeatureValueFromOption( + TimelinesSharedFeatures.HAS_TITLE, + features.hasTitle + ) + richDataRecord.setFeatureValueFromOption( + TimelinesSharedFeatures.HAS_DESCRIPTION, + features.hasDescription + ) + richDataRecord.setFeatureValueFromOption( + TimelinesSharedFeatures.HAS_VISIT_SITE_CALL_TO_ACTION, + features.hasVisitSiteCallToAction + ) + richDataRecord.setFeatureValueFromOption( + TimelinesSharedFeatures.HAS_APP_INSTALL_CALL_TO_ACTION, + features.hasAppInstallCallToAction + ) + richDataRecord.setFeatureValueFromOption( + TimelinesSharedFeatures.HAS_WATCH_NOW_CALL_TO_ACTION, + features.hasWatchNowCallToAction + ) + // text features + richDataRecord.setFeatureValueFromOption( + TimelinesSharedFeatures.NUM_CAPS, + Some(features.numCaps.toDouble) + ) + richDataRecord.setFeatureValueFromOption( + TimelinesSharedFeatures.TWEET_LENGTH, + Some(features.length.toDouble) + ) + richDataRecord.setFeatureValueFromOption( + TimelinesSharedFeatures.TWEET_LENGTH_TYPE, + Some(getTweetLengthType(features.length.toInt)) + ) + richDataRecord.setFeatureValueFromOption( + TimelinesSharedFeatures.NUM_WHITESPACES, + Some(features.numWhiteSpaces.toDouble) + ) + richDataRecord.setFeatureValueFromOption( + TimelinesSharedFeatures.HAS_QUESTION, + Some(features.hasQuestion) + ) + richDataRecord.setFeatureValueFromOption( + TimelinesSharedFeatures.NUM_NEWLINES, + features.numNewlines.map(_.toDouble) + ) + } + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/adapters/earlybird/BUILD.bazel b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/adapters/earlybird/BUILD.bazel new file mode 100644 index 0000000000..9428c0d394 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/adapters/earlybird/BUILD.bazel @@ -0,0 +1,19 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + strict_deps = True, + tags = ["bazel-compatible"], + dependencies = [ + "src/java/com/twitter/ml/api:api-base", + "src/scala/com/twitter/ml/api:api-base", + "src/scala/com/twitter/ml/api/util", + "src/scala/com/twitter/timelines/prediction/common/adapters:base", + "src/scala/com/twitter/timelines/prediction/features/common", + "src/scala/com/twitter/timelines/prediction/features/recap", + "src/scala/com/twitter/timelines/util", + "src/thrift/com/twitter/ml/api:data-java", + "src/thrift/com/twitter/ml/api:data-scala", + "src/thrift/com/twitter/search/common:features-scala", + "timelines/src/main/scala/com/twitter/timelines/util", + ], +) diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/adapters/earlybird/EarlybirdAdapter.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/adapters/earlybird/EarlybirdAdapter.scala new file mode 100644 index 0000000000..833733c167 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/adapters/earlybird/EarlybirdAdapter.scala @@ -0,0 +1,453 @@ +package com.twitter.home_mixer.functional_component.feature_hydrator.adapters.earlybird + +import com.twitter.ml.api.Feature +import com.twitter.ml.api.FeatureContext +import com.twitter.ml.api.RichDataRecord +import com.twitter.ml.api.util.DataRecordConverters._ +import com.twitter.timelines.prediction.common.adapters.TimelinesMutatingAdapterBase +import com.twitter.search.common.features.{thriftscala => sc} +import com.twitter.timelines.prediction.features.common.TimelinesSharedFeatures +import com.twitter.timelines.prediction.features.recap.RecapFeatures +import com.twitter.timelines.util.UrlExtractorUtil +import java.lang.{Boolean => JBoolean} +import java.lang.{Double => JDouble} +import java.util.{Map => JMap} +import scala.collection.JavaConverters._ + +object EarlybirdAdapter extends TimelinesMutatingAdapterBase[Option[sc.ThriftTweetFeatures]] { + + override val getFeatureContext: FeatureContext = new FeatureContext( + RecapFeatures.BIDIRECTIONAL_FAV_COUNT, + RecapFeatures.BIDIRECTIONAL_REPLY_COUNT, + RecapFeatures.BIDIRECTIONAL_RETWEET_COUNT, + RecapFeatures.BLENDER_SCORE, + RecapFeatures.CONTAINS_MEDIA, + RecapFeatures.CONVERSATIONAL_COUNT, + RecapFeatures.EMBEDS_IMPRESSION_COUNT, + RecapFeatures.EMBEDS_URL_COUNT, + RecapFeatures.FAV_COUNT, + RecapFeatures.FAV_COUNT_V2, + RecapFeatures.FROM_INACTIVE_USER, + RecapFeatures.FROM_MUTUAL_FOLLOW, + RecapFeatures.FROM_VERIFIED_ACCOUNT, + RecapFeatures.HAS_CARD, + RecapFeatures.HAS_CONSUMER_VIDEO, + RecapFeatures.HAS_HASHTAG, + RecapFeatures.HAS_IMAGE, + RecapFeatures.HAS_LINK, + RecapFeatures.HAS_MENTION, + RecapFeatures.HAS_MULTIPLE_HASHTAGS_OR_TRENDS, + RecapFeatures.HAS_MULTIPLE_MEDIA, + RecapFeatures.HAS_NATIVE_IMAGE, + RecapFeatures.HAS_NATIVE_VIDEO, + RecapFeatures.HAS_NEWS, + RecapFeatures.HAS_PERISCOPE, + RecapFeatures.HAS_PRO_VIDEO, + RecapFeatures.HAS_TREND, + RecapFeatures.HAS_VIDEO, + RecapFeatures.HAS_VINE, + RecapFeatures.HAS_VISIBLE_LINK, + RecapFeatures.IS_AUTHOR_BOT, + RecapFeatures.IS_AUTHOR_NEW, + RecapFeatures.IS_AUTHOR_NSFW, + RecapFeatures.IS_AUTHOR_PROFILE_EGG, + RecapFeatures.IS_AUTHOR_SPAM, + RecapFeatures.IS_BUSINESS_SCORE, + RecapFeatures.IS_OFFENSIVE, + RecapFeatures.IS_REPLY, + RecapFeatures.IS_RETWEET, + RecapFeatures.IS_RETWEETER_BOT, + RecapFeatures.IS_RETWEETER_NEW, + RecapFeatures.IS_RETWEETER_NSFW, + RecapFeatures.IS_RETWEETER_PROFILE_EGG, + RecapFeatures.IS_RETWEETER_SPAM, + RecapFeatures.IS_RETWEET_OF_REPLY, + RecapFeatures.IS_SENSITIVE, + RecapFeatures.LANGUAGE, + RecapFeatures.LINK_COUNT, + RecapFeatures.LINK_LANGUAGE, + RecapFeatures.MATCH_SEARCHER_LANGS, + RecapFeatures.MATCH_SEARCHER_MAIN_LANG, + RecapFeatures.MATCH_UI_LANG, + RecapFeatures.MENTIONED_SCREEN_NAMES, + RecapFeatures.MENTION_SEARCHER, + RecapFeatures.NUM_HASHTAGS, + RecapFeatures.NUM_MENTIONS, + RecapFeatures.PREV_USER_TWEET_ENGAGEMENT, + RecapFeatures.PROBABLY_FROM_FOLLOWED_AUTHOR, + RecapFeatures.REPLY_COUNT, + RecapFeatures.REPLY_COUNT_V2, + RecapFeatures.REPLY_OTHER, + RecapFeatures.REPLY_SEARCHER, + RecapFeatures.RETWEET_COUNT, + RecapFeatures.RETWEET_COUNT_V2, + RecapFeatures.RETWEET_DIRECTED_AT_USER_IN_FIRST_DEGREE, + RecapFeatures.RETWEET_OF_MUTUAL_FOLLOW, + RecapFeatures.RETWEET_OTHER, + RecapFeatures.RETWEET_SEARCHER, + RecapFeatures.SIGNATURE, + RecapFeatures.SOURCE_AUTHOR_REP, + RecapFeatures.TEXT_SCORE, + RecapFeatures.TWEET_COUNT_FROM_USER_IN_SNAPSHOT, + RecapFeatures.UNIDIRECTIONAL_FAV_COUNT, + RecapFeatures.UNIDIRECTIONAL_REPLY_COUNT, + RecapFeatures.UNIDIRECTIONAL_RETWEET_COUNT, + RecapFeatures.URL_DOMAINS, + RecapFeatures.USER_REP, + RecapFeatures.VIDEO_VIEW_COUNT, + // shared features + TimelinesSharedFeatures.WEIGHTED_FAV_COUNT, + TimelinesSharedFeatures.WEIGHTED_RETWEET_COUNT, + TimelinesSharedFeatures.WEIGHTED_REPLY_COUNT, + TimelinesSharedFeatures.WEIGHTED_QUOTE_COUNT, + TimelinesSharedFeatures.EMBEDS_IMPRESSION_COUNT_V2, + TimelinesSharedFeatures.EMBEDS_URL_COUNT_V2, + TimelinesSharedFeatures.DECAYED_FAVORITE_COUNT, + TimelinesSharedFeatures.DECAYED_RETWEET_COUNT, + TimelinesSharedFeatures.DECAYED_REPLY_COUNT, + TimelinesSharedFeatures.DECAYED_QUOTE_COUNT, + TimelinesSharedFeatures.FAKE_FAVORITE_COUNT, + TimelinesSharedFeatures.FAKE_RETWEET_COUNT, + TimelinesSharedFeatures.FAKE_REPLY_COUNT, + TimelinesSharedFeatures.FAKE_QUOTE_COUNT, + TimelinesSharedFeatures.QUOTE_COUNT, + TimelinesSharedFeatures.EARLYBIRD_SCORE, + // Safety features + TimelinesSharedFeatures.LABEL_ABUSIVE_FLAG, + TimelinesSharedFeatures.LABEL_ABUSIVE_HI_RCL_FLAG, + TimelinesSharedFeatures.LABEL_DUP_CONTENT_FLAG, + TimelinesSharedFeatures.LABEL_NSFW_HI_PRC_FLAG, + TimelinesSharedFeatures.LABEL_NSFW_HI_RCL_FLAG, + TimelinesSharedFeatures.LABEL_SPAM_FLAG, + TimelinesSharedFeatures.LABEL_SPAM_HI_RCL_FLAG, + // periscope features + TimelinesSharedFeatures.PERISCOPE_EXISTS, + TimelinesSharedFeatures.PERISCOPE_IS_LIVE, + TimelinesSharedFeatures.PERISCOPE_HAS_BEEN_FEATURED, + TimelinesSharedFeatures.PERISCOPE_IS_CURRENTLY_FEATURED, + TimelinesSharedFeatures.PERISCOPE_IS_FROM_QUALITY_SOURCE, + // VISIBLE_TOKEN_RATIO + TimelinesSharedFeatures.VISIBLE_TOKEN_RATIO, + TimelinesSharedFeatures.HAS_QUOTE, + TimelinesSharedFeatures.IS_COMPOSER_SOURCE_CAMERA, + // health features + TimelinesSharedFeatures.PREPORTED_TWEET_SCORE, + // media + TimelinesSharedFeatures.CLASSIFICATION_LABELS + ) + + override val commonFeatures: Set[Feature[_]] = Set.empty + + override def setFeatures( + ebFeatures: Option[sc.ThriftTweetFeatures], + richDataRecord: RichDataRecord + ): Unit = { + if (ebFeatures.nonEmpty) { + val features = ebFeatures.get + richDataRecord.setFeatureValue[JDouble]( + RecapFeatures.PREV_USER_TWEET_ENGAGEMENT, + features.prevUserTweetEngagement.toDouble + ) + richDataRecord + .setFeatureValue[JBoolean](RecapFeatures.IS_SENSITIVE, features.isSensitiveContent) + richDataRecord + .setFeatureValue[JBoolean](RecapFeatures.HAS_MULTIPLE_MEDIA, features.hasMultipleMedia) + richDataRecord + .setFeatureValue[JBoolean](RecapFeatures.IS_AUTHOR_PROFILE_EGG, features.isAuthorProfileEgg) + richDataRecord.setFeatureValue[JBoolean](RecapFeatures.IS_AUTHOR_NEW, features.isAuthorNew) + richDataRecord + .setFeatureValue[JDouble](RecapFeatures.NUM_MENTIONS, features.numMentions.toDouble) + richDataRecord.setFeatureValue[JBoolean](RecapFeatures.HAS_MENTION, features.numMentions > 0) + richDataRecord + .setFeatureValue[JDouble](RecapFeatures.NUM_HASHTAGS, features.numHashtags.toDouble) + richDataRecord.setFeatureValue[JBoolean](RecapFeatures.HAS_HASHTAG, features.numHashtags > 0) + richDataRecord + .setFeatureValue[JDouble](RecapFeatures.LINK_LANGUAGE, features.linkLanguage.toDouble) + richDataRecord.setFeatureValue[JBoolean](RecapFeatures.IS_AUTHOR_NSFW, features.isAuthorNSFW) + richDataRecord.setFeatureValue[JBoolean](RecapFeatures.IS_AUTHOR_SPAM, features.isAuthorSpam) + richDataRecord.setFeatureValue[JBoolean](RecapFeatures.IS_AUTHOR_BOT, features.isAuthorBot) + richDataRecord.setFeatureValueFromOption( + RecapFeatures.LANGUAGE, + features.language.map(_.getValue.toLong)) + richDataRecord.setFeatureValueFromOption( + RecapFeatures.SIGNATURE, + features.signature.map(_.toLong)) + richDataRecord + .setFeatureValue[JBoolean](RecapFeatures.FROM_INACTIVE_USER, features.fromInActiveUser) + richDataRecord + .setFeatureValue[JBoolean]( + RecapFeatures.PROBABLY_FROM_FOLLOWED_AUTHOR, + features.probablyFromFollowedAuthor) + richDataRecord + .setFeatureValue[JBoolean](RecapFeatures.FROM_MUTUAL_FOLLOW, features.fromMutualFollow) + richDataRecord.setFeatureValue[JBoolean]( + RecapFeatures.FROM_VERIFIED_ACCOUNT, + features.fromVerifiedAccount) + richDataRecord.setFeatureValue[JDouble](RecapFeatures.USER_REP, features.userRep) + richDataRecord + .setFeatureValue[JDouble](RecapFeatures.IS_BUSINESS_SCORE, features.isBusinessScore) + richDataRecord + .setFeatureValue[JBoolean](RecapFeatures.HAS_CONSUMER_VIDEO, features.hasConsumerVideo) + richDataRecord.setFeatureValue[JBoolean](RecapFeatures.HAS_PRO_VIDEO, features.hasProVideo) + richDataRecord.setFeatureValue[JBoolean](RecapFeatures.HAS_VINE, features.hasVine) + richDataRecord.setFeatureValue[JBoolean](RecapFeatures.HAS_PERISCOPE, features.hasPeriscope) + richDataRecord + .setFeatureValue[JBoolean](RecapFeatures.HAS_NATIVE_VIDEO, features.hasNativeVideo) + richDataRecord + .setFeatureValue[JBoolean](RecapFeatures.HAS_NATIVE_IMAGE, features.hasNativeImage) + richDataRecord.setFeatureValue[JBoolean](RecapFeatures.HAS_CARD, features.hasCard) + richDataRecord.setFeatureValue[JBoolean](RecapFeatures.HAS_IMAGE, features.hasImage) + richDataRecord.setFeatureValue[JBoolean](RecapFeatures.HAS_NEWS, features.hasNews) + richDataRecord.setFeatureValue[JBoolean](RecapFeatures.HAS_VIDEO, features.hasVideo) + richDataRecord.setFeatureValue[JBoolean](RecapFeatures.CONTAINS_MEDIA, features.containsMedia) + richDataRecord + .setFeatureValue[JBoolean](RecapFeatures.RETWEET_SEARCHER, features.retweetSearcher) + richDataRecord.setFeatureValue[JBoolean](RecapFeatures.REPLY_SEARCHER, features.replySearcher) + richDataRecord + .setFeatureValue[JBoolean](RecapFeatures.MENTION_SEARCHER, features.mentionSearcher) + richDataRecord.setFeatureValue[JBoolean](RecapFeatures.REPLY_OTHER, features.replyOther) + richDataRecord.setFeatureValue[JBoolean](RecapFeatures.RETWEET_OTHER, features.retweetOther) + richDataRecord.setFeatureValue[JBoolean](RecapFeatures.IS_REPLY, features.isReply) + richDataRecord.setFeatureValue[JBoolean](RecapFeatures.IS_RETWEET, features.isRetweet) + richDataRecord.setFeatureValue[JBoolean](RecapFeatures.IS_OFFENSIVE, features.isOffensive) + richDataRecord.setFeatureValue[JBoolean](RecapFeatures.MATCH_UI_LANG, features.matchesUILang) + richDataRecord + .setFeatureValue[JBoolean]( + RecapFeatures.MATCH_SEARCHER_MAIN_LANG, + features.matchesSearcherMainLang) + richDataRecord.setFeatureValue[JBoolean]( + RecapFeatures.MATCH_SEARCHER_LANGS, + features.matchesSearcherLangs) + richDataRecord + .setFeatureValue[JDouble]( + RecapFeatures.BIDIRECTIONAL_FAV_COUNT, + features.bidirectionalFavCount) + richDataRecord + .setFeatureValue[JDouble]( + RecapFeatures.UNIDIRECTIONAL_FAV_COUNT, + features.unidirectionalFavCount) + richDataRecord + .setFeatureValue[JDouble]( + RecapFeatures.BIDIRECTIONAL_REPLY_COUNT, + features.bidirectionalReplyCount) + richDataRecord + .setFeatureValue[JDouble]( + RecapFeatures.UNIDIRECTIONAL_REPLY_COUNT, + features.unidirectionalReplyCount) + richDataRecord + .setFeatureValue[JDouble]( + RecapFeatures.BIDIRECTIONAL_RETWEET_COUNT, + features.bidirectionalRetweetCount) + richDataRecord + .setFeatureValue[JDouble]( + RecapFeatures.UNIDIRECTIONAL_RETWEET_COUNT, + features.unidirectionalRetweetCount) + richDataRecord + .setFeatureValue[JDouble](RecapFeatures.CONVERSATIONAL_COUNT, features.conversationCount) + richDataRecord.setFeatureValue[JDouble]( + RecapFeatures.TWEET_COUNT_FROM_USER_IN_SNAPSHOT, + features.tweetCountFromUserInSnapshot + ) + richDataRecord + .setFeatureValue[JBoolean]( + RecapFeatures.IS_RETWEETER_PROFILE_EGG, + features.isRetweeterProfileEgg) + richDataRecord + .setFeatureValue[JBoolean](RecapFeatures.IS_RETWEETER_NEW, features.isRetweeterNew) + richDataRecord + .setFeatureValue[JBoolean](RecapFeatures.IS_RETWEETER_BOT, features.isRetweeterBot) + richDataRecord + .setFeatureValue[JBoolean](RecapFeatures.IS_RETWEETER_NSFW, features.isRetweeterNSFW) + richDataRecord + .setFeatureValue[JBoolean](RecapFeatures.IS_RETWEETER_SPAM, features.isRetweeterSpam) + richDataRecord + .setFeatureValue[JBoolean]( + RecapFeatures.RETWEET_OF_MUTUAL_FOLLOW, + features.retweetOfMutualFollow) + richDataRecord + .setFeatureValue[JDouble](RecapFeatures.SOURCE_AUTHOR_REP, features.sourceAuthorRep) + richDataRecord + .setFeatureValue[JBoolean](RecapFeatures.IS_RETWEET_OF_REPLY, features.isRetweetOfReply) + richDataRecord.setFeatureValueFromOption( + RecapFeatures.RETWEET_DIRECTED_AT_USER_IN_FIRST_DEGREE, + features.retweetDirectedAtUserInFirstDegree + ) + richDataRecord + .setFeatureValue[JDouble]( + RecapFeatures.EMBEDS_IMPRESSION_COUNT, + features.embedsImpressionCount.toDouble) + richDataRecord + .setFeatureValue[JDouble](RecapFeatures.EMBEDS_URL_COUNT, features.embedsUrlCount.toDouble) + richDataRecord + .setFeatureValue[JDouble](RecapFeatures.VIDEO_VIEW_COUNT, features.videoViewCount.toDouble) + richDataRecord + .setFeatureValue[JDouble](RecapFeatures.REPLY_COUNT, features.replyCount.toDouble) + richDataRecord + .setFeatureValue[JDouble](RecapFeatures.RETWEET_COUNT, features.retweetCount.toDouble) + richDataRecord.setFeatureValue[JDouble](RecapFeatures.FAV_COUNT, features.favCount.toDouble) + richDataRecord.setFeatureValue[JDouble](RecapFeatures.BLENDER_SCORE, features.blenderScore) + richDataRecord.setFeatureValue[JDouble](RecapFeatures.TEXT_SCORE, features.textScore) + richDataRecord + .setFeatureValue[JBoolean](RecapFeatures.HAS_VISIBLE_LINK, features.hasVisibleLink) + richDataRecord.setFeatureValue[JBoolean](RecapFeatures.HAS_LINK, features.hasLink) + richDataRecord.setFeatureValue[JBoolean](RecapFeatures.HAS_TREND, features.hasTrend) + richDataRecord.setFeatureValue[JBoolean]( + RecapFeatures.HAS_MULTIPLE_HASHTAGS_OR_TRENDS, + features.hasMultipleHashtagsOrTrends + ) + richDataRecord.setFeatureValueFromOption( + RecapFeatures.FAV_COUNT_V2, + features.favCountV2.map(_.toDouble)) + richDataRecord.setFeatureValueFromOption( + RecapFeatures.RETWEET_COUNT_V2, + features.retweetCountV2.map(_.toDouble) + ) + richDataRecord.setFeatureValueFromOption( + RecapFeatures.REPLY_COUNT_V2, + features.replyCountV2.map(_.toDouble)) + val urls = features.urlsList.getOrElse(Seq.empty) + richDataRecord.setFeatureValue( + RecapFeatures.URL_DOMAINS, + urls.toSet.flatMap(UrlExtractorUtil.extractDomain).asJava) + richDataRecord.setFeatureValue[JDouble](RecapFeatures.LINK_COUNT, urls.size.toDouble) + // shared features + richDataRecord.setFeatureValueFromOption( + TimelinesSharedFeatures.WEIGHTED_FAV_COUNT, + features.weightedFavoriteCount.map(_.toDouble) + ) + richDataRecord.setFeatureValueFromOption( + TimelinesSharedFeatures.WEIGHTED_RETWEET_COUNT, + features.weightedRetweetCount.map(_.toDouble) + ) + richDataRecord.setFeatureValueFromOption( + TimelinesSharedFeatures.WEIGHTED_REPLY_COUNT, + features.weightedReplyCount.map(_.toDouble) + ) + richDataRecord.setFeatureValueFromOption( + TimelinesSharedFeatures.WEIGHTED_QUOTE_COUNT, + features.weightedQuoteCount.map(_.toDouble) + ) + richDataRecord.setFeatureValueFromOption( + TimelinesSharedFeatures.EMBEDS_IMPRESSION_COUNT_V2, + features.embedsImpressionCountV2.map(_.toDouble) + ) + richDataRecord.setFeatureValueFromOption( + TimelinesSharedFeatures.EMBEDS_URL_COUNT_V2, + features.embedsUrlCountV2.map(_.toDouble) + ) + richDataRecord.setFeatureValueFromOption( + TimelinesSharedFeatures.DECAYED_FAVORITE_COUNT, + features.decayedFavoriteCount.map(_.toDouble) + ) + richDataRecord.setFeatureValueFromOption( + TimelinesSharedFeatures.DECAYED_RETWEET_COUNT, + features.decayedRetweetCount.map(_.toDouble) + ) + richDataRecord.setFeatureValueFromOption( + TimelinesSharedFeatures.DECAYED_REPLY_COUNT, + features.decayedReplyCount.map(_.toDouble) + ) + richDataRecord.setFeatureValueFromOption( + TimelinesSharedFeatures.DECAYED_QUOTE_COUNT, + features.decayedQuoteCount.map(_.toDouble) + ) + richDataRecord.setFeatureValueFromOption( + TimelinesSharedFeatures.FAKE_FAVORITE_COUNT, + features.fakeFavoriteCount.map(_.toDouble) + ) + richDataRecord.setFeatureValueFromOption( + TimelinesSharedFeatures.FAKE_RETWEET_COUNT, + features.fakeRetweetCount.map(_.toDouble) + ) + richDataRecord.setFeatureValueFromOption( + TimelinesSharedFeatures.FAKE_REPLY_COUNT, + features.fakeReplyCount.map(_.toDouble) + ) + richDataRecord.setFeatureValueFromOption( + TimelinesSharedFeatures.FAKE_QUOTE_COUNT, + features.fakeQuoteCount.map(_.toDouble) + ) + richDataRecord.setFeatureValueFromOption( + TimelinesSharedFeatures.QUOTE_COUNT, + features.quoteCount.map(_.toDouble) + ) + richDataRecord.setFeatureValue[JDouble]( + TimelinesSharedFeatures.EARLYBIRD_SCORE, + features.earlybirdScore + ) + // safety features + richDataRecord.setFeatureValueFromOption( + TimelinesSharedFeatures.LABEL_ABUSIVE_FLAG, + features.labelAbusiveFlag + ) + richDataRecord.setFeatureValueFromOption( + TimelinesSharedFeatures.LABEL_ABUSIVE_HI_RCL_FLAG, + features.labelAbusiveHiRclFlag + ) + richDataRecord.setFeatureValueFromOption( + TimelinesSharedFeatures.LABEL_DUP_CONTENT_FLAG, + features.labelDupContentFlag + ) + richDataRecord.setFeatureValueFromOption( + TimelinesSharedFeatures.LABEL_NSFW_HI_PRC_FLAG, + features.labelNsfwHiPrcFlag + ) + richDataRecord.setFeatureValueFromOption( + TimelinesSharedFeatures.LABEL_NSFW_HI_RCL_FLAG, + features.labelNsfwHiRclFlag + ) + richDataRecord.setFeatureValueFromOption( + TimelinesSharedFeatures.LABEL_SPAM_FLAG, + features.labelSpamFlag + ) + richDataRecord.setFeatureValueFromOption( + TimelinesSharedFeatures.LABEL_SPAM_HI_RCL_FLAG, + features.labelSpamHiRclFlag + ) + // periscope features + richDataRecord.setFeatureValueFromOption( + TimelinesSharedFeatures.PERISCOPE_EXISTS, + features.periscopeExists + ) + richDataRecord.setFeatureValueFromOption( + TimelinesSharedFeatures.PERISCOPE_IS_LIVE, + features.periscopeIsLive + ) + richDataRecord.setFeatureValueFromOption( + TimelinesSharedFeatures.PERISCOPE_HAS_BEEN_FEATURED, + features.periscopeHasBeenFeatured + ) + richDataRecord.setFeatureValueFromOption( + TimelinesSharedFeatures.PERISCOPE_IS_CURRENTLY_FEATURED, + features.periscopeIsCurrentlyFeatured + ) + richDataRecord.setFeatureValueFromOption( + TimelinesSharedFeatures.PERISCOPE_IS_FROM_QUALITY_SOURCE, + features.periscopeIsFromQualitySource + ) + // misc features + richDataRecord.setFeatureValueFromOption( + TimelinesSharedFeatures.VISIBLE_TOKEN_RATIO, + features.visibleTokenRatio.map(_.toDouble) + ) + richDataRecord.setFeatureValueFromOption( + TimelinesSharedFeatures.HAS_QUOTE, + features.hasQuote + ) + richDataRecord.setFeatureValueFromOption( + TimelinesSharedFeatures.IS_COMPOSER_SOURCE_CAMERA, + features.isComposerSourceCamera + ) + // health scores + richDataRecord.setFeatureValueFromOption( + TimelinesSharedFeatures.PREPORTED_TWEET_SCORE, + features.pReportedTweetScore + ) + // media + richDataRecord.setFeatureValueFromOption( + TimelinesSharedFeatures.CLASSIFICATION_LABELS, + features.mediaClassificationInfo.map(_.toMap.asJava.asInstanceOf[JMap[String, JDouble]]) + ) + } + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/adapters/inferred_topic/BUILD.bazel b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/adapters/inferred_topic/BUILD.bazel new file mode 100644 index 0000000000..acf19eabb1 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/adapters/inferred_topic/BUILD.bazel @@ -0,0 +1,13 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + strict_deps = True, + tags = ["bazel-compatible"], + dependencies = [ + "src/java/com/twitter/ml/api:api-base", + "src/scala/com/twitter/ml/api/util", + "src/scala/com/twitter/timelines/prediction/common/adapters:base", + "src/scala/com/twitter/timelines/prediction/features/common", + "src/thrift/com/twitter/ml/api:data-java", + ], +) diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/adapters/inferred_topic/InferredTopicAdapter.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/adapters/inferred_topic/InferredTopicAdapter.scala new file mode 100644 index 0000000000..62125ea5ad --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/adapters/inferred_topic/InferredTopicAdapter.scala @@ -0,0 +1,25 @@ +package com.twitter.home_mixer.functional_component.feature_hydrator.adapters.inferred_topic + +import com.twitter.ml.api.Feature +import com.twitter.ml.api.FeatureContext +import com.twitter.ml.api.RichDataRecord +import com.twitter.timelines.prediction.common.adapters.TimelinesMutatingAdapterBase +import com.twitter.timelines.prediction.features.common.TimelinesSharedFeatures +import scala.collection.JavaConverters._ + +object InferredTopicAdapter extends TimelinesMutatingAdapterBase[Map[Long, Double]] { + + override val getFeatureContext: FeatureContext = new FeatureContext( + TimelinesSharedFeatures.INFERRED_TOPIC_IDS) + + override val commonFeatures: Set[Feature[_]] = Set.empty + + override def setFeatures( + inferredTopicFeatures: Map[Long, Double], + richDataRecord: RichDataRecord + ): Unit = { + richDataRecord.setFeatureValue( + TimelinesSharedFeatures.INFERRED_TOPIC_IDS, + inferredTopicFeatures.keys.map(_.toString).toSet.asJava) + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/adapters/non_ml_features/BUILD.bazel b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/adapters/non_ml_features/BUILD.bazel new file mode 100644 index 0000000000..d1a7281c5b --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/adapters/non_ml_features/BUILD.bazel @@ -0,0 +1,15 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + strict_deps = True, + tags = ["bazel-compatible"], + dependencies = [ + "src/java/com/twitter/ml/api:api-base", + "src/java/com/twitter/ml/api/constant", + "src/scala/com/twitter/ml/api/util", + "src/scala/com/twitter/timelines/prediction/common/adapters:base", + "src/scala/com/twitter/timelines/prediction/features/common", + "src/thrift/com/twitter/ml/api:data-java", + "src/thrift/com/twitter/timelines/author_features:thrift-java", + ], +) diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/adapters/non_ml_features/NonMLCandidateFeaturesAdapter.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/adapters/non_ml_features/NonMLCandidateFeaturesAdapter.scala new file mode 100644 index 0000000000..24e0edb9b5 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/adapters/non_ml_features/NonMLCandidateFeaturesAdapter.scala @@ -0,0 +1,44 @@ +package com.twitter.home_mixer.functional_component.feature_hydrator.adapters.non_ml_features + +import com.twitter.ml.api.constant.SharedFeatures +import com.twitter.ml.api.Feature +import com.twitter.ml.api.FeatureContext +import com.twitter.ml.api.RichDataRecord +import com.twitter.timelines.prediction.common.adapters.TimelinesMutatingAdapterBase +import com.twitter.timelines.prediction.features.common.TimelinesSharedFeatures +import java.lang.{Long => JLong} + +case class NonMLCandidateFeatures( + tweetId: Long, + sourceTweetId: Option[Long], + originalAuthorId: Option[Long], +) + +/** + * define non ml features adapter to create a data record which includes many non ml features + * e.g. predictionRequestId, userId, tweetId to be used as joined key in batch pipeline. + */ +object NonMLCandidateFeaturesAdapter extends TimelinesMutatingAdapterBase[NonMLCandidateFeatures] { + + private val featureContext = new FeatureContext( + SharedFeatures.TWEET_ID, + // For Secondary Engagement data generation + TimelinesSharedFeatures.SOURCE_TWEET_ID, + TimelinesSharedFeatures.ORIGINAL_AUTHOR_ID, + ) + + override def getFeatureContext: FeatureContext = featureContext + + override val commonFeatures: Set[Feature[_]] = Set.empty + + override def setFeatures( + nonMLCandidateFeatures: NonMLCandidateFeatures, + richDataRecord: RichDataRecord + ): Unit = { + richDataRecord.setFeatureValue[JLong](SharedFeatures.TWEET_ID, nonMLCandidateFeatures.tweetId) + nonMLCandidateFeatures.sourceTweetId.foreach( + richDataRecord.setFeatureValue[JLong](TimelinesSharedFeatures.SOURCE_TWEET_ID, _)) + nonMLCandidateFeatures.originalAuthorId.foreach( + richDataRecord.setFeatureValue[JLong](TimelinesSharedFeatures.ORIGINAL_AUTHOR_ID, _)) + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/adapters/non_ml_features/NonMLCommonFeaturesAdapter.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/adapters/non_ml_features/NonMLCommonFeaturesAdapter.scala new file mode 100644 index 0000000000..612c4900cc --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/adapters/non_ml_features/NonMLCommonFeaturesAdapter.scala @@ -0,0 +1,48 @@ +package com.twitter.home_mixer.functional_component.feature_hydrator.adapters.non_ml_features + +import com.twitter.ml.api.constant.SharedFeatures +import com.twitter.ml.api.Feature +import com.twitter.ml.api.FeatureContext +import com.twitter.ml.api.RichDataRecord +import com.twitter.timelines.prediction.common.adapters.TimelinesMutatingAdapterBase +import com.twitter.timelines.prediction.features.common.TimelinesSharedFeatures +import java.lang.{Long => JLong} + +case class NonMLCommonFeatures( + userId: Long, + predictionRequestId: Option[Long], + servedTimestamp: Long, +) + +/** + * define non ml features adapter to create a data record which includes many non ml features + * e.g. predictionRequestId, userId, tweetId to be used as joined key in batch pipeline. + */ +object NonMLCommonFeaturesAdapter extends TimelinesMutatingAdapterBase[NonMLCommonFeatures] { + + private val featureContext = new FeatureContext( + SharedFeatures.USER_ID, + TimelinesSharedFeatures.PREDICTION_REQUEST_ID, + TimelinesSharedFeatures.SERVED_TIMESTAMP, + ) + + override def getFeatureContext: FeatureContext = featureContext + + override val commonFeatures: Set[Feature[_]] = Set( + SharedFeatures.USER_ID, + TimelinesSharedFeatures.PREDICTION_REQUEST_ID, + TimelinesSharedFeatures.SERVED_TIMESTAMP, + ) + + override def setFeatures( + nonMLCommonFeatures: NonMLCommonFeatures, + richDataRecord: RichDataRecord + ): Unit = { + richDataRecord.setFeatureValue[JLong](SharedFeatures.USER_ID, nonMLCommonFeatures.userId) + nonMLCommonFeatures.predictionRequestId.foreach( + richDataRecord.setFeatureValue[JLong](TimelinesSharedFeatures.PREDICTION_REQUEST_ID, _)) + richDataRecord.setFeatureValue[JLong]( + TimelinesSharedFeatures.SERVED_TIMESTAMP, + nonMLCommonFeatures.servedTimestamp) + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/adapters/offline_aggregates/BUILD.bazel b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/adapters/offline_aggregates/BUILD.bazel new file mode 100644 index 0000000000..bfff86d3ff --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/adapters/offline_aggregates/BUILD.bazel @@ -0,0 +1,13 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + strict_deps = True, + tags = ["bazel-compatible"], + dependencies = [ + "src/java/com/twitter/ml/api:api-base", + "src/scala/com/twitter/timelines/prediction/common/adapters:base", + "src/thrift/com/twitter/ml/api:data-java", + "timelines/data_processing/ml_util/aggregation_framework/conversion:for-timelines", + "timelineservice/common/src/main/scala/com/twitter/timelineservice/model", + ], +) diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/adapters/offline_aggregates/PassThroughAdapter.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/adapters/offline_aggregates/PassThroughAdapter.scala new file mode 100644 index 0000000000..cd5ab020f6 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/adapters/offline_aggregates/PassThroughAdapter.scala @@ -0,0 +1,12 @@ +package com.twitter.home_mixer.functional_component.feature_hydrator.adapters.offline_aggregates + +import com.twitter.ml.api.DataRecord +import com.twitter.ml.api.IRecordOneToOneAdapter + +object PassThroughAdapter extends IRecordOneToOneAdapter[Seq[DataRecord]] { + override def adaptToDataRecord(record: Seq[DataRecord]): DataRecord = + record.headOption.getOrElse(new DataRecord) + + // This is not necessary and should not be used. + override def getFeatureContext = ??? +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/adapters/offline_aggregates/SparseAggregatesToDenseAdapter.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/adapters/offline_aggregates/SparseAggregatesToDenseAdapter.scala new file mode 100644 index 0000000000..816b350690 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/adapters/offline_aggregates/SparseAggregatesToDenseAdapter.scala @@ -0,0 +1,17 @@ +package com.twitter.home_mixer.functional_component.feature_hydrator.adapters.offline_aggregates + +import com.twitter.ml.api.DataRecord +import com.twitter.ml.api.FeatureContext +import com.twitter.ml.api.RichDataRecord +import com.twitter.timelines.data_processing.ml_util.aggregation_framework.conversion.CombineCountsPolicy +import com.twitter.timelines.prediction.common.adapters.TimelinesIRecordAdapter + +class SparseAggregatesToDenseAdapter(policy: CombineCountsPolicy) + extends TimelinesIRecordAdapter[Seq[DataRecord]] { + + override def setFeatures(input: Seq[DataRecord], mutableDataRecord: RichDataRecord): Unit = + policy.defaultMergeRecord(mutableDataRecord.getRecord, input.toList) + + override val getFeatureContext: FeatureContext = + new FeatureContext(policy.outputFeaturesPostMerge.toSeq: _*) +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/adapters/twhin_embeddings/BUILD.bazel b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/adapters/twhin_embeddings/BUILD.bazel new file mode 100644 index 0000000000..c32c29ce5c --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/adapters/twhin_embeddings/BUILD.bazel @@ -0,0 +1,16 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + strict_deps = True, + tags = ["bazel-compatible"], + dependencies = [ + "src/java/com/twitter/ml/api:api-base", + "src/scala/com/twitter/ml/api:api-base", + "src/scala/com/twitter/ml/api/util", + "src/scala/com/twitter/timelines/prediction/common/adapters:base", + "src/thrift/com/twitter/ml/api:data-java", + "src/thrift/com/twitter/ml/api:data-scala", + "src/thrift/com/twitter/ml/api:embedding-java", + "src/thrift/com/twitter/ml/api:embedding-scala", + ], +) diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/adapters/twhin_embeddings/TwhinEmbeddingsAdapter.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/adapters/twhin_embeddings/TwhinEmbeddingsAdapter.scala new file mode 100644 index 0000000000..99c4ba8dd9 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/adapters/twhin_embeddings/TwhinEmbeddingsAdapter.scala @@ -0,0 +1,81 @@ +package com.twitter.home_mixer.functional_component.feature_hydrator.adapters.twhin_embeddings + +import com.twitter.ml.api.util.BufferToIterators.RichFloatBuffer +import com.twitter.ml.api.util.ScalaToJavaDataRecordConversions +import com.twitter.ml.api.DataType +import com.twitter.ml.api.Feature +import com.twitter.ml.api.FeatureContext +import com.twitter.ml.api.RichDataRecord +import com.twitter.ml.api.{thriftscala => ml} +import com.twitter.timelines.prediction.common.adapters.TimelinesMutatingAdapterBase + +import java.nio.ByteOrder + +sealed trait TwhinEmbeddingsAdapter extends TimelinesMutatingAdapterBase[Option[ml.Embedding]] { + def twhinEmbeddingsFeature: Feature.Tensor + + override def getFeatureContext: FeatureContext = new FeatureContext( + twhinEmbeddingsFeature + ) + + override def setFeatures( + embedding: Option[ml.Embedding], + richDataRecord: RichDataRecord + ): Unit = { + embedding.foreach { embedding => + val floatTensor = embedding.tensor map { tensor => + ml.FloatTensor( + tensor.content + .order(ByteOrder.LITTLE_ENDIAN) + .asFloatBuffer + .iterator.toList + .map(_.toDouble)) + } + + floatTensor.foreach { v => + richDataRecord.setFeatureValue( + twhinEmbeddingsFeature, + ScalaToJavaDataRecordConversions.scalaTensor2Java(ml.GeneralTensor.FloatTensor(v)) + ) + } + } + } +} + +object TwhinEmbeddingsFeatures { + val TwhinAuthorFollowEmbeddingsFeature: Feature.Tensor = new Feature.Tensor( + "original_author.timelines.twhin_author_follow_embeddings.twhin_author_follow_embeddings", + DataType.FLOAT + ) + + val TwhinUserEngagementEmbeddingsFeature: Feature.Tensor = new Feature.Tensor( + "user.timelines.twhin_user_engagement_embeddings.twhin_user_engagement_embeddings", + DataType.FLOAT + ) + + val TwhinUserFollowEmbeddingsFeature: Feature.Tensor = new Feature.Tensor( + "user.timelines.twhin_user_follow_embeddings.twhin_user_follow_embeddings", + DataType.FLOAT + ) +} + +object TwhinAuthorFollowEmbeddingsAdapter extends TwhinEmbeddingsAdapter { + override val twhinEmbeddingsFeature: Feature.Tensor = + TwhinEmbeddingsFeatures.TwhinAuthorFollowEmbeddingsFeature + + override val commonFeatures: Set[Feature[_]] = Set.empty +} + +object TwhinUserEngagementEmbeddingsAdapter extends TwhinEmbeddingsAdapter { + override val twhinEmbeddingsFeature: Feature.Tensor = + TwhinEmbeddingsFeatures.TwhinUserEngagementEmbeddingsFeature + + override val commonFeatures: Set[Feature[_]] = Set(twhinEmbeddingsFeature) +} + +object TwhinUserFollowEmbeddingsAdapter extends TwhinEmbeddingsAdapter { + override val twhinEmbeddingsFeature: Feature.Tensor = + TwhinEmbeddingsFeatures.TwhinUserFollowEmbeddingsFeature + + override val commonFeatures: Set[Feature[_]] = Set(twhinEmbeddingsFeature) +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/offline_aggregates/AggregateFeatureInfo.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/offline_aggregates/AggregateFeatureInfo.scala new file mode 100644 index 0000000000..a1186e2f40 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/offline_aggregates/AggregateFeatureInfo.scala @@ -0,0 +1,37 @@ +package com.twitter.home_mixer.functional_component.feature_hydrator.offline_aggregates + +import com.twitter.ml.api.FeatureContext +import com.twitter.timelines.data_processing.ml_util.aggregation_framework.AggregateGroup +import com.twitter.timelines.data_processing.ml_util.aggregation_framework.AggregateType.AggregateType +import com.twitter.timelines.data_processing.ml_util.aggregation_framework.TypedAggregateGroup +import scala.jdk.CollectionConverters.asJavaIterableConverter + +// A helper class deriving aggregate feature info from the given configuration parameters. +class AggregateFeatureInfo( + val aggregateGroups: Set[AggregateGroup], + val aggregateType: AggregateType) { + + private val typedAggregateGroups = aggregateGroups.flatMap(_.buildTypedAggregateGroups()).toList + + val featureContext: FeatureContext = + new FeatureContext( + (typedAggregateGroups.flatMap(_.allOutputFeatures) ++ + typedAggregateGroups.flatMap(_.allOutputKeys) ++ + Seq(TypedAggregateGroup.timestampFeature)).asJava) + + val feature: BaseAggregateRootFeature = + AggregateFeatureInfo.pickFeature(aggregateType) +} + +object AggregateFeatureInfo { + val features: Set[BaseAggregateRootFeature] = + Set(PartAAggregateRootFeature, PartBAggregateRootFeature) + + def pickFeature(aggregateType: AggregateType): BaseAggregateRootFeature = { + val filtered = features.filter(_.aggregateTypes.contains(aggregateType)) + require( + filtered.size == 1, + "requested AggregateType must be backed by exactly one physical store.") + filtered.head + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/offline_aggregates/AggregateFeaturesToDecodeWithMetadata.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/offline_aggregates/AggregateFeaturesToDecodeWithMetadata.scala new file mode 100644 index 0000000000..cbda354379 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/offline_aggregates/AggregateFeaturesToDecodeWithMetadata.scala @@ -0,0 +1,68 @@ +package com.twitter.home_mixer.functional_component.feature_hydrator.offline_aggregates + +import com.twitter.finagle.stats.NullStatsReceiver +import com.twitter.timelinemixer.injection.repository.uss.VersionedAggregateFeaturesDecoder +import com.twitter.ml.api.DataRecord +import com.twitter.timelines.aggregate_interactions.thriftjava.UserAggregateInteractions +import com.twitter.timelines.aggregate_interactions.v17.thriftjava.{ + UserAggregateInteractions => V17UserAggregateInteractions +} +import com.twitter.timelines.aggregate_interactions.v1.thriftjava.{ + UserAggregateInteractions => V1UserAggregateInteractions +} +import com.twitter.timelines.suggests.common.dense_data_record.thriftjava.DenseCompactDataRecord +import com.twitter.timelines.suggests.common.dense_data_record.thriftscala.DenseFeatureMetadata +import java.lang.{Long => JLong} +import java.util.Collections +import java.util.{Map => JMap} + +private[offline_aggregates] case class AggregateFeaturesToDecodeWithMetadata( + metadataOpt: Option[DenseFeatureMetadata], + aggregates: UserAggregateInteractions) { + def toDataRecord(dr: DenseCompactDataRecord): DataRecord = + VersionedAggregateFeaturesDecoder.fromJDenseCompact( + metadataOpt, + dr.versionId, + NullStatsReceiver, + s"V${dr.versionId}" + )(dr) + + def userAggregatesOpt: Option[DenseCompactDataRecord] = { + aggregates.getSetField match { + case UserAggregateInteractions._Fields.V17 => + Option(aggregates.getV17.user_aggregates) + case _ => + None + } + } + + def userAuthorAggregates = extract(_.user_author_aggregates) + def userEngagerAggregates = extract(_.user_engager_aggregates) + def userMentionAggregates = extract(_.user_mention_aggregates) + def userOriginalAuthorAggregates = extract(_.user_original_author_aggregates) + def userRequestDowAggregates = extract(_.user_request_dow_aggregates) + def userRequestHourAggregates = extract(_.user_request_hour_aggregates) + def rectweetUserSimclustersTweetAggregates = extract(_.rectweet_user_simclusters_tweet_aggregates) + def userTwitterListAggregates = extract(_.user_list_aggregates) + def userTopicAggregates = extract(_.user_topic_aggregates) + def userInferredTopicAggregates = extract(_.user_inferred_topic_aggregates) + def userMediaUnderstandingAnnotationAggregates = extract( + _.user_media_understanding_annotation_aggregates) + + private def extract[T]( + v17Fn: V17UserAggregateInteractions => JMap[JLong, DenseCompactDataRecord] + ): JMap[JLong, DenseCompactDataRecord] = { + aggregates.getSetField match { + case UserAggregateInteractions._Fields.V17 => + v17Fn(aggregates.getV17) + case _ => + Collections.emptyMap() + } + } +} + +object AggregateFeaturesToDecodeWithMetadata { + val empty = new AggregateFeaturesToDecodeWithMetadata( + None, + UserAggregateInteractions.v1(new V1UserAggregateInteractions())) +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/offline_aggregates/BUILD.bazel b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/offline_aggregates/BUILD.bazel new file mode 100644 index 0000000000..d9c353df18 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/offline_aggregates/BUILD.bazel @@ -0,0 +1,38 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + strict_deps = True, + tags = ["bazel-compatible"], + dependencies = [ + "finatra/inject/inject-core/src/main/scala", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/adapters/offline_aggregates", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/model", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/param", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/util", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/candidate", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/feature/datarecord", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/feature/featuremap/datarecord", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/feature_hydrator", + "servo/repo/src/main/scala", + "src/java/com/twitter/ml/api:api-base", + "src/scala/com/twitter/timelines/prediction/adapters/request_context", + "src/scala/com/twitter/timelines/prediction/common/adapters:base", + "src/scala/com/twitter/timelines/prediction/common/aggregates", + "src/scala/com/twitter/timelines/util", + "src/thrift/com/twitter/ml/api:data-java", + "src/thrift/com/twitter/timelines/suggests/common:data_record_metadata-scala", + "src/thrift/com/twitter/timelines/suggests/common:dense_data_record-scala", + "src/thrift/com/twitter/tweetypie:service-scala", + "src/thrift/com/twitter/tweetypie:tweet-scala", + "src/thrift/com/twitter/user_session_store:thrift-java", + "stitch/stitch-core", + "timelinemixer/server/src/main/scala/com/twitter/timelinemixer/injection/repository/uss:versioned-aggregate-features-decoder", + "timelines/data_processing/jobs/timeline_ranking_user_features:mini", + "timelines/data_processing/ml_util/aggregation_framework:common_types", + "timelines/data_processing/ml_util/aggregation_framework/conversion:for-timelines", + "timelineservice/common/src/main/scala/com/twitter/timelineservice/model", + "user_session_store/src/main/scala/com/twitter/user_session_store", + "util/util-core", + ], +) diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/offline_aggregates/BaseAggregateQueryFeatureHydrator.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/offline_aggregates/BaseAggregateQueryFeatureHydrator.scala new file mode 100644 index 0000000000..b58f23d2ee --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/offline_aggregates/BaseAggregateQueryFeatureHydrator.scala @@ -0,0 +1,76 @@ +package com.twitter.home_mixer.functional_component.feature_hydrator.offline_aggregates + +import com.twitter.product_mixer.core.feature.Feature +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.feature.featuremap.FeatureMapBuilder +import com.twitter.product_mixer.core.functional_component.feature_hydrator.QueryFeatureHydrator +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.servo.repository.Repository +import com.twitter.stitch.Stitch +import com.twitter.timelines.aggregate_interactions.thriftjava.UserAggregateInteractions +import com.twitter.timelines.data_processing.ml_util.aggregation_framework.AggregateType.AggregateType +import com.twitter.timelines.data_processing.ml_util.aggregation_framework.StoreConfig +import com.twitter.timelines.suggests.common.dense_data_record.thriftscala.DenseFeatureMetadata +import com.twitter.user_session_store.thriftjava.UserSession +import com.twitter.util.Future + +abstract class BaseAggregateQueryFeatureHydrator( + featureRepository: Repository[Long, Option[UserSession]], + metadataRepository: Repository[Int, Option[DenseFeatureMetadata]], + feature: Feature[PipelineQuery, Option[AggregateFeaturesToDecodeWithMetadata]]) + extends QueryFeatureHydrator[PipelineQuery] { + + override def hydrate(query: PipelineQuery): Stitch[FeatureMap] = { + val viewerId = query.getRequiredUserId + + Stitch.callFuture( + featureRepository(viewerId) + .flatMap { userSession: Option[UserSession] => + val featuresWithMetadata: Option[Future[AggregateFeaturesToDecodeWithMetadata]] = + userSession + .flatMap(decodeUserSession(_)) + + featuresWithMetadata + .map { fu: Future[AggregateFeaturesToDecodeWithMetadata] => fu.map(Some(_)) } + .getOrElse(Future.None) + .map { value => + FeatureMapBuilder() + .add(feature, value) + .build() + } + } + ) + } + + private def decodeUserSession( + session: UserSession + ): Option[Future[AggregateFeaturesToDecodeWithMetadata]] = { + Option(session.user_aggregate_interactions).flatMap { aggregates => + aggregates.getSetField match { + case UserAggregateInteractions._Fields.V17 => + Some( + getAggregateFeaturesWithMetadata( + aggregates.getV17.user_aggregates.versionId, + UserAggregateInteractions.v17(aggregates.getV17)) + ) + case _ => + None + } + } + } + + private def getAggregateFeaturesWithMetadata( + versionId: Int, + userAggregateInteractions: UserAggregateInteractions, + ): Future[AggregateFeaturesToDecodeWithMetadata] = { + metadataRepository(versionId) + .map(AggregateFeaturesToDecodeWithMetadata(_, userAggregateInteractions)) + } +} + +trait BaseAggregateRootFeature + extends Feature[PipelineQuery, Option[AggregateFeaturesToDecodeWithMetadata]] { + def aggregateStores: Set[StoreConfig[_]] + + lazy val aggregateTypes: Set[AggregateType] = aggregateStores.map(_.aggregateType) +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/offline_aggregates/BaseEdgeAggregateFeatureHydrator.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/offline_aggregates/BaseEdgeAggregateFeatureHydrator.scala new file mode 100644 index 0000000000..fde5ab909e --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/offline_aggregates/BaseEdgeAggregateFeatureHydrator.scala @@ -0,0 +1,93 @@ +package com.twitter.home_mixer.functional_component.feature_hydrator.offline_aggregates + +import com.twitter.ml.api.DataRecord +import com.twitter.ml.api.FeatureContext +import com.twitter.ml.api.IRecordOneToOneAdapter +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.feature.Feature +import com.twitter.product_mixer.core.feature.FeatureWithDefaultOnFailure +import com.twitter.product_mixer.core.feature.datarecord.DataRecordInAFeature +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.feature.featuremap.FeatureMapBuilder +import com.twitter.product_mixer.core.functional_component.feature_hydrator.BulkCandidateFeatureHydrator +import com.twitter.product_mixer.core.model.common.CandidateWithFeatures +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.stitch.Stitch +import com.twitter.timelines.data_processing.ml_util.aggregation_framework.AggregateGroup +import com.twitter.timelines.data_processing.ml_util.aggregation_framework.AggregateType.AggregateType +import com.twitter.timelines.suggests.common.dense_data_record.thriftjava.DenseCompactDataRecord +import java.lang.{Long => JLong} +import java.util.{Map => JMap} + +abstract case class BaseEdgeAggregateFeature( + aggregateGroups: Set[AggregateGroup], + aggregateType: AggregateType, + extractMapFn: AggregateFeaturesToDecodeWithMetadata => JMap[JLong, DenseCompactDataRecord], + adapter: IRecordOneToOneAdapter[Seq[DataRecord]], + getSecondaryKeysFn: CandidateWithFeatures[TweetCandidate] => Seq[Long]) + extends DataRecordInAFeature[PipelineQuery] + with FeatureWithDefaultOnFailure[PipelineQuery, DataRecord] { + override def defaultValue: DataRecord = new DataRecord + + private val rootFeatureInfo = new AggregateFeatureInfo(aggregateGroups, aggregateType) + val featureContext: FeatureContext = rootFeatureInfo.featureContext + val rootFeature: BaseAggregateRootFeature = rootFeatureInfo.feature +} + +trait BaseEdgeAggregateFeatureHydrator + extends BulkCandidateFeatureHydrator[PipelineQuery, TweetCandidate] { + + def aggregateFeatures: Set[BaseEdgeAggregateFeature] + + override def features = aggregateFeatures.asInstanceOf[Set[Feature[_, _]]] + + override def apply( + query: PipelineQuery, + candidates: Seq[CandidateWithFeatures[TweetCandidate]] + ): Stitch[Seq[FeatureMap]] = { + + val featureMapBuilders: Seq[FeatureMapBuilder] = + for (_ <- candidates) yield FeatureMapBuilder() + + aggregateFeatures.foreach { feature => + val featureValues = hydrateAggregateFeature(query, candidates, feature) + (featureMapBuilders zip featureValues).foreach { + case (featureMapBuilder, featureValue) => featureMapBuilder.add(feature, featureValue) + } + } + + Stitch.value(featureMapBuilders.map(_.build())) + } + + private def hydrateAggregateFeature( + query: PipelineQuery, + candidates: Seq[CandidateWithFeatures[TweetCandidate]], + feature: BaseEdgeAggregateFeature + ): Seq[DataRecord] = { + val rootFeature = feature.rootFeature + val extractMapFn = feature.extractMapFn + val featureContext = feature.featureContext + val secondaryIds: Seq[Seq[Long]] = candidates.map(feature.getSecondaryKeysFn) + + val featuresToDecodeWithMetadata = query.features + .flatMap(_.getOrElse(rootFeature, None)) + .getOrElse(AggregateFeaturesToDecodeWithMetadata.empty) + + // Decode the DenseCompactDataRecords into DataRecords for each required secondary id. + val decoded: Map[Long, DataRecord] = + Utils.selectAndTransform( + secondaryIds.flatten.distinct, + featuresToDecodeWithMetadata.toDataRecord, + extractMapFn(featuresToDecodeWithMetadata)) + + // Remove unnecessary features in-place. This is safe because the underlying DataRecords + // are unique and have just been generated in the previous step. + decoded.values.foreach(Utils.filterDataRecord(_, featureContext)) + + // Put features into the FeatureMapBuilder's + secondaryIds.map { ids => + val dataRecords = ids.flatMap(decoded.get) + feature.adapter.adaptToDataRecord(dataRecords) + } + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/offline_aggregates/EdgeAggregateFeatures.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/offline_aggregates/EdgeAggregateFeatures.scala new file mode 100644 index 0000000000..d637b3b3dc --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/offline_aggregates/EdgeAggregateFeatures.scala @@ -0,0 +1,107 @@ +package com.twitter.home_mixer.functional_component.feature_hydrator.offline_aggregates + +import com.twitter.home_mixer.functional_component.feature_hydrator.TSPInferredTopicFeature +import com.twitter.home_mixer.functional_component.feature_hydrator.adapters.offline_aggregates.PassThroughAdapter +import com.twitter.home_mixer.functional_component.feature_hydrator.adapters.offline_aggregates.SparseAggregatesToDenseAdapter +import com.twitter.home_mixer.model.HomeFeatures.AuthorIdFeature +import com.twitter.home_mixer.model.HomeFeatures.MentionUserIdFeature +import com.twitter.home_mixer.model.HomeFeatures.TopicIdSocialContextFeature +import com.twitter.home_mixer.util.CandidatesUtil +import com.twitter.timelines.data_processing.ml_util.aggregation_framework.AggregateType +import com.twitter.timelines.prediction.common.aggregates.TimelinesAggregationConfig +import com.twitter.timelines.prediction.common.aggregates.TimelinesAggregationConfig.CombineCountPolicies + +object EdgeAggregateFeatures { + + object UserAuthorAggregateFeature + extends BaseEdgeAggregateFeature( + aggregateGroups = TimelinesAggregationConfig.userAuthorAggregatesV2 ++ Set( + TimelinesAggregationConfig.userAuthorAggregatesV5, + TimelinesAggregationConfig.tweetSourceUserAuthorAggregatesV1, + TimelinesAggregationConfig.twitterWideUserAuthorAggregates + ), + aggregateType = AggregateType.UserAuthor, + extractMapFn = _.userAuthorAggregates, + adapter = PassThroughAdapter, + getSecondaryKeysFn = _.features.getOrElse(AuthorIdFeature, None).toSeq + ) + + object UserOriginalAuthorAggregateFeature + extends BaseEdgeAggregateFeature( + aggregateGroups = Set( + TimelinesAggregationConfig.userOriginalAuthorReciprocalEngagementAggregates), + aggregateType = AggregateType.UserOriginalAuthor, + extractMapFn = _.userOriginalAuthorAggregates, + adapter = PassThroughAdapter, + getSecondaryKeysFn = candidate => + CandidatesUtil.getOriginalAuthorId(candidate.features).toSeq + ) + + object UserTopicAggregateFeature + extends BaseEdgeAggregateFeature( + aggregateGroups = Set( + TimelinesAggregationConfig.userTopicAggregates, + TimelinesAggregationConfig.userTopicAggregatesV2, + ), + aggregateType = AggregateType.UserTopic, + extractMapFn = _.userTopicAggregates, + adapter = PassThroughAdapter, + getSecondaryKeysFn = candidate => + candidate.features.getOrElse(TopicIdSocialContextFeature, None).toSeq + ) + + object UserMentionAggregateFeature + extends BaseEdgeAggregateFeature( + aggregateGroups = Set(TimelinesAggregationConfig.userMentionAggregates), + aggregateType = AggregateType.UserMention, + extractMapFn = _.userMentionAggregates, + adapter = new SparseAggregatesToDenseAdapter(CombineCountPolicies.MentionCountsPolicy), + getSecondaryKeysFn = candidate => + candidate.features.getOrElse(MentionUserIdFeature, Seq.empty) + ) + + object UserInferredTopicAggregateFeature + extends BaseEdgeAggregateFeature( + aggregateGroups = Set( + TimelinesAggregationConfig.userInferredTopicAggregates, + TimelinesAggregationConfig.userInferredTopicAggregatesV2 + ), + aggregateType = AggregateType.UserInferredTopic, + extractMapFn = _.userInferredTopicAggregates, + adapter = new SparseAggregatesToDenseAdapter( + CombineCountPolicies.UserInferredTopicV2CountsPolicy), + getSecondaryKeysFn = candidate => + candidate.features.getOrElse(TSPInferredTopicFeature, Map.empty[Long, Double]).keys.toSeq + ) + + object UserMediaUnderstandingAnnotationAggregateFeature + extends BaseEdgeAggregateFeature( + aggregateGroups = Set( + TimelinesAggregationConfig.userMediaUnderstandingAnnotationAggregates), + aggregateType = AggregateType.UserMediaUnderstandingAnnotation, + extractMapFn = _.userMediaUnderstandingAnnotationAggregates, + adapter = new SparseAggregatesToDenseAdapter( + CombineCountPolicies.UserMediaUnderstandingAnnotationCountsPolicy), + getSecondaryKeysFn = candidate => + CandidatesUtil.getMediaUnderstandingAnnotationIds(candidate.features) + ) + + object UserEngagerAggregateFeature + extends BaseEdgeAggregateFeature( + aggregateGroups = Set(TimelinesAggregationConfig.userEngagerAggregates), + aggregateType = AggregateType.UserEngager, + extractMapFn = _.userEngagerAggregates, + adapter = new SparseAggregatesToDenseAdapter(CombineCountPolicies.EngagerCountsPolicy), + getSecondaryKeysFn = candidate => CandidatesUtil.getEngagerUserIds(candidate.features) + ) + + object UserEngagerGoodClickAggregateFeature + extends BaseEdgeAggregateFeature( + aggregateGroups = Set(TimelinesAggregationConfig.userEngagerGoodClickAggregates), + aggregateType = AggregateType.UserEngager, + extractMapFn = _.userEngagerAggregates, + adapter = new SparseAggregatesToDenseAdapter( + CombineCountPolicies.EngagerGoodClickCountsPolicy), + getSecondaryKeysFn = candidate => CandidatesUtil.getEngagerUserIds(candidate.features) + ) +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/offline_aggregates/PartAAggregateQueryFeatureHydrator.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/offline_aggregates/PartAAggregateQueryFeatureHydrator.scala new file mode 100644 index 0000000000..aa75e2c05e --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/offline_aggregates/PartAAggregateQueryFeatureHydrator.scala @@ -0,0 +1,35 @@ +package com.twitter.home_mixer.functional_component.feature_hydrator.offline_aggregates + +import com.twitter.home_mixer.param.HomeMixerInjectionNames.TimelineAggregateMetadataRepository +import com.twitter.home_mixer.param.HomeMixerInjectionNames.TimelineAggregatePartARepository +import com.twitter.product_mixer.core.model.common.identifier.FeatureHydratorIdentifier +import com.twitter.servo.repository.Repository +import com.twitter.timelines.data_processing.jobs.timeline_ranking_user_features.TimelinesPartAStoreRegister +import com.twitter.timelines.data_processing.ml_util.aggregation_framework.StoreConfig +import com.twitter.timelines.suggests.common.dense_data_record.thriftscala.DenseFeatureMetadata +import com.twitter.user_session_store.thriftjava.UserSession +import javax.inject.Inject +import javax.inject.Named +import javax.inject.Singleton + +object PartAAggregateRootFeature extends BaseAggregateRootFeature { + override val aggregateStores: Set[StoreConfig[_]] = TimelinesPartAStoreRegister.allStores +} + +@Singleton +class PartAAggregateQueryFeatureHydrator @Inject() ( + @Named(TimelineAggregatePartARepository) + repository: Repository[Long, Option[UserSession]], + @Named(TimelineAggregateMetadataRepository) + metadataRepository: Repository[Int, Option[DenseFeatureMetadata]]) + extends BaseAggregateQueryFeatureHydrator( + repository, + metadataRepository, + PartAAggregateRootFeature + ) { + + override val identifier: FeatureHydratorIdentifier = + FeatureHydratorIdentifier("PartAAggregateQuery") + + override val features = Set(PartAAggregateRootFeature) +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/offline_aggregates/PartBAggregateQueryFeatureHydrator.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/offline_aggregates/PartBAggregateQueryFeatureHydrator.scala new file mode 100644 index 0000000000..4a00c7ca49 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/offline_aggregates/PartBAggregateQueryFeatureHydrator.scala @@ -0,0 +1,144 @@ +package com.twitter.home_mixer.functional_component.feature_hydrator.offline_aggregates + +import com.twitter.home_mixer.param.HomeMixerInjectionNames.TimelineAggregateMetadataRepository +import com.twitter.home_mixer.param.HomeMixerInjectionNames.TimelineAggregatePartBRepository +import com.twitter.ml.api.DataRecord +import com.twitter.ml.api.DataRecordMerger +import com.twitter.ml.api.FeatureContext +import com.twitter.ml.api.RichDataRecord +import com.twitter.product_mixer.core.feature.Feature +import com.twitter.product_mixer.core.feature.FeatureWithDefaultOnFailure +import com.twitter.product_mixer.core.feature.datarecord.DataRecordInAFeature +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.model.common.identifier.FeatureHydratorIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.servo.repository.Repository +import com.twitter.stitch.Stitch +import com.twitter.timelines.data_processing.jobs.timeline_ranking_user_features.TimelinesPartBStoreRegister +import com.twitter.timelines.data_processing.ml_util.aggregation_framework.AggregateType +import com.twitter.timelines.data_processing.ml_util.aggregation_framework.StoreConfig +import com.twitter.timelines.prediction.adapters.request_context.RequestContextAdapter +import com.twitter.timelines.prediction.common.aggregates.TimelinesAggregationConfig +import com.twitter.timelines.suggests.common.dense_data_record.thriftscala.DenseFeatureMetadata +import com.twitter.user_session_store.thriftjava.UserSession +import com.twitter.util.Time +import javax.inject.Inject +import javax.inject.Named +import javax.inject.Singleton + +object PartBAggregateRootFeature extends BaseAggregateRootFeature { + override val aggregateStores: Set[StoreConfig[_]] = TimelinesPartBStoreRegister.allStores +} + +object UserAggregateFeature + extends DataRecordInAFeature[PipelineQuery] + with FeatureWithDefaultOnFailure[PipelineQuery, DataRecord] { + override def defaultValue: DataRecord = new DataRecord() +} + +@Singleton +class PartBAggregateQueryFeatureHydrator @Inject() ( + @Named(TimelineAggregatePartBRepository) + repository: Repository[Long, Option[UserSession]], + @Named(TimelineAggregateMetadataRepository) + metadataRepository: Repository[Int, Option[DenseFeatureMetadata]]) + extends BaseAggregateQueryFeatureHydrator( + repository, + metadataRepository, + PartBAggregateRootFeature + ) { + + override val identifier: FeatureHydratorIdentifier = + FeatureHydratorIdentifier("PartBAggregateQuery") + + override val features: Set[Feature[_, _]] = + Set(PartBAggregateRootFeature, UserAggregateFeature) + + private val userAggregateFeatureInfo = new AggregateFeatureInfo( + aggregateGroups = Set( + TimelinesAggregationConfig.userAggregatesV2, + TimelinesAggregationConfig.userAggregatesV5Continuous, + TimelinesAggregationConfig.userReciprocalEngagementAggregates, + TimelinesAggregationConfig.twitterWideUserAggregates, + ), + aggregateType = AggregateType.User + ) + + private val userHourAggregateFeatureInfo = new AggregateFeatureInfo( + aggregateGroups = Set( + TimelinesAggregationConfig.userRequestHourAggregates, + ), + aggregateType = AggregateType.UserRequestHour + ) + + private val userDowAggregateFeatureInfo = new AggregateFeatureInfo( + aggregateGroups = Set( + TimelinesAggregationConfig.userRequestDowAggregates + ), + aggregateType = AggregateType.UserRequestDow + ) + + require( + userAggregateFeatureInfo.feature == PartBAggregateRootFeature, + "UserAggregates feature must be provided by the PartB data source.") + require( + userHourAggregateFeatureInfo.feature == PartBAggregateRootFeature, + "UserRequstHourAggregates feature must be provided by the PartB data source.") + require( + userDowAggregateFeatureInfo.feature == PartBAggregateRootFeature, + "UserRequestDowAggregates feature must be provided by the PartB data source.") + + override def hydrate(query: PipelineQuery): Stitch[FeatureMap] = { + // Hydrate TimelineAggregatePartBFeature and UserAggregateFeature sequentially. + super.hydrate(query).map { featureMap => + val time: Time = Time.now + val hourOfDay = RequestContextAdapter.hourFromTimestamp(time.inMilliseconds) + val dayOfWeek = RequestContextAdapter.dowFromTimestamp(time.inMilliseconds) + + val dr = featureMap + .get(PartBAggregateRootFeature).map { featuresWithMetadata => + val userAggregatesDr = + featuresWithMetadata.userAggregatesOpt + .map(featuresWithMetadata.toDataRecord) + val userRequestHourAggregatesDr = + Option(featuresWithMetadata.userRequestHourAggregates.get(hourOfDay)) + .map(featuresWithMetadata.toDataRecord) + val userRequestDowAggregatesDr = + Option(featuresWithMetadata.userRequestHourAggregates.get(dayOfWeek)) + .map(featuresWithMetadata.toDataRecord) + + dropUnknownFeatures(userAggregatesDr, userAggregateFeatureInfo.featureContext) + + dropUnknownFeatures( + userRequestHourAggregatesDr, + userHourAggregateFeatureInfo.featureContext) + + dropUnknownFeatures( + userRequestDowAggregatesDr, + userDowAggregateFeatureInfo.featureContext) + + mergeDataRecordOpts( + userAggregatesDr, + userRequestHourAggregatesDr, + userRequestDowAggregatesDr) + + }.getOrElse(new DataRecord()) + + featureMap + (UserAggregateFeature, dr) + } + } + + private val drMerger = new DataRecordMerger + private def mergeDataRecordOpts(dataRecordOpts: Option[DataRecord]*): DataRecord = + dataRecordOpts.flatten.foldLeft(new DataRecord) { (l, r) => + drMerger.merge(l, r) + l + } + + private def dropUnknownFeatures( + dataRecordOpt: Option[DataRecord], + featureContext: FeatureContext + ): Unit = + dataRecordOpt.foreach(new RichDataRecord(_, featureContext).dropUnknownFeatures()) + +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/offline_aggregates/Phase1EdgeAggregateFeatureHydrator.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/offline_aggregates/Phase1EdgeAggregateFeatureHydrator.scala new file mode 100644 index 0000000000..c8c912b633 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/offline_aggregates/Phase1EdgeAggregateFeatureHydrator.scala @@ -0,0 +1,20 @@ +package com.twitter.home_mixer.functional_component.feature_hydrator.offline_aggregates + +import com.twitter.home_mixer.functional_component.feature_hydrator.offline_aggregates.EdgeAggregateFeatures._ +import com.twitter.product_mixer.core.model.common.identifier.FeatureHydratorIdentifier +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class Phase1EdgeAggregateFeatureHydrator @Inject() extends BaseEdgeAggregateFeatureHydrator { + + override val identifier: FeatureHydratorIdentifier = + FeatureHydratorIdentifier("Phase1EdgeAggregate") + + override val aggregateFeatures: Set[BaseEdgeAggregateFeature] = + Set( + UserAuthorAggregateFeature, + UserOriginalAuthorAggregateFeature, + UserMentionAggregateFeature + ) +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/offline_aggregates/Phase2EdgeAggregateFeatureHydrator.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/offline_aggregates/Phase2EdgeAggregateFeatureHydrator.scala new file mode 100644 index 0000000000..fb5c9e2fb2 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/offline_aggregates/Phase2EdgeAggregateFeatureHydrator.scala @@ -0,0 +1,22 @@ +package com.twitter.home_mixer.functional_component.feature_hydrator.offline_aggregates + +import com.twitter.home_mixer.functional_component.feature_hydrator.offline_aggregates.EdgeAggregateFeatures._ +import com.twitter.product_mixer.core.model.common.identifier.FeatureHydratorIdentifier +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class Phase2EdgeAggregateFeatureHydrator @Inject() extends BaseEdgeAggregateFeatureHydrator { + + override val identifier: FeatureHydratorIdentifier = + FeatureHydratorIdentifier("Phase2EdgeAggregate") + + override val aggregateFeatures: Set[BaseEdgeAggregateFeature] = + Set( + UserEngagerAggregateFeature, + UserEngagerGoodClickAggregateFeature, + UserInferredTopicAggregateFeature, + UserTopicAggregateFeature, + UserMediaUnderstandingAnnotationAggregateFeature + ) +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/offline_aggregates/Utils.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/offline_aggregates/Utils.scala new file mode 100644 index 0000000000..45d205888a --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/offline_aggregates/Utils.scala @@ -0,0 +1,36 @@ +package com.twitter.home_mixer.functional_component.feature_hydrator.offline_aggregates + +import com.twitter.ml.api.DataRecord +import com.twitter.ml.api.FeatureContext +import com.twitter.ml.api.RichDataRecord +import com.twitter.timelines.suggests.common.dense_data_record.thriftjava.DenseCompactDataRecord + +private[offline_aggregates] object Utils { + + /** + * Selects only those values in map that correspond to the keys in ids and apply the provided + * transform to the selected values. This is a convenience method for use by Timelines Aggregation + * Framework based features. + * + * @param idsToSelect The set of ids to extract values for. + * @param transform A transform to apply to the selected values. + * @param map Map[Long, DenseCompactDataRecord] + */ + def selectAndTransform( + idsToSelect: Seq[Long], + transform: DenseCompactDataRecord => DataRecord, + map: java.util.Map[java.lang.Long, DenseCompactDataRecord], + ): Map[Long, DataRecord] = { + val filtered: Seq[(Long, DataRecord)] = + for { + id <- idsToSelect if map.containsKey(id) + } yield { + id -> transform(map.get(id)) + } + filtered.toMap + } + + def filterDataRecord(dr: DataRecord, featureContext: FeatureContext): Unit = { + new RichDataRecord(dr, featureContext).dropUnknownFeatures() + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/real_time_aggregates/BUILD.bazel b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/real_time_aggregates/BUILD.bazel new file mode 100644 index 0000000000..93f042fca2 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/real_time_aggregates/BUILD.bazel @@ -0,0 +1,25 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + strict_deps = True, + tags = ["bazel-compatible"], + dependencies = [ + "finatra/inject/inject-core/src/main/scala", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/model", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/model/request", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/param", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/util", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/feature/datarecord", + "servo/repo/src/main/scala", + "src/java/com/twitter/ml/api:api-base", + "src/java/com/twitter/ml/api/constant", + "src/scala/com/twitter/ml/api/util", + "src/scala/com/twitter/timelines/prediction/common/aggregates/real_time:base-config", + "src/thrift/com/twitter/ml/api:data-java", + "src/thrift/com/twitter/wtf/real_time_interaction_graph:wtf-real_time_interaction_graph-thrift-java", + "stitch/stitch-core", + "timelines/data_processing/ml_util/aggregation_framework:common_types", + "timelines/data_processing/ml_util/aggregation_framework/heron", + "util/util-core", + ], +) diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/real_time_aggregates/BaseRealTimeAggregateBulkCandidateFeatureHydrator.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/real_time_aggregates/BaseRealTimeAggregateBulkCandidateFeatureHydrator.scala new file mode 100644 index 0000000000..d892a072a6 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/real_time_aggregates/BaseRealTimeAggregateBulkCandidateFeatureHydrator.scala @@ -0,0 +1,41 @@ +package com.twitter.home_mixer.functional_component.feature_hydrator.real_time_aggregates + +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.feature.Feature +import com.twitter.product_mixer.core.feature.datarecord.DataRecordInAFeature +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.feature.featuremap.FeatureMapBuilder +import com.twitter.product_mixer.core.functional_component.feature_hydrator.BulkCandidateFeatureHydrator +import com.twitter.product_mixer.core.model.common.CandidateWithFeatures +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.stitch.Stitch + +trait BaseRealTimeAggregateBulkCandidateFeatureHydrator[K] + extends BulkCandidateFeatureHydrator[PipelineQuery, TweetCandidate] + with BaseRealtimeAggregateHydrator[K] { + + val outputFeature: DataRecordInAFeature[TweetCandidate] + + override def features: Set[Feature[_, _]] = Set(outputFeature) + + override lazy val statScope: String = identifier.toString + + def keysFromQueryAndCandidates( + query: PipelineQuery, + candidates: Seq[CandidateWithFeatures[TweetCandidate]] + ): Seq[Option[K]] + + override def apply( + query: PipelineQuery, + candidates: Seq[CandidateWithFeatures[TweetCandidate]] + ): Stitch[Seq[FeatureMap]] = { + val possiblyKeys = keysFromQueryAndCandidates(query, candidates) + fetchAndConstructDataRecords(possiblyKeys).map { dataRecords => + dataRecords.map { dataRecord => + FeatureMapBuilder() + .add(outputFeature, dataRecord) + .build() + } + } + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/real_time_aggregates/BaseRealTimeAggregateQueryFeatureHydrator.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/real_time_aggregates/BaseRealTimeAggregateQueryFeatureHydrator.scala new file mode 100644 index 0000000000..772e0bb927 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/real_time_aggregates/BaseRealTimeAggregateQueryFeatureHydrator.scala @@ -0,0 +1,35 @@ +package com.twitter.home_mixer.functional_component.feature_hydrator.real_time_aggregates + +import com.twitter.product_mixer.core.feature.Feature +import com.twitter.product_mixer.core.feature.datarecord.DataRecordInAFeature +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.feature.featuremap.FeatureMapBuilder +import com.twitter.product_mixer.core.functional_component.feature_hydrator.QueryFeatureHydrator +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.stitch.Stitch + +trait BaseRealTimeAggregateQueryFeatureHydrator[K] + extends QueryFeatureHydrator[PipelineQuery] + with BaseRealtimeAggregateHydrator[K] { + + val outputFeature: DataRecordInAFeature[PipelineQuery] + + override def features: Set[Feature[_, _]] = Set(outputFeature) + + override lazy val statScope: String = identifier.toString + + def keysFromQueryAndCandidates( + query: PipelineQuery + ): Option[K] + + override def hydrate( + query: PipelineQuery + ): Stitch[FeatureMap] = { + val possiblyKeys = keysFromQueryAndCandidates(query) + fetchAndConstructDataRecords(Seq(possiblyKeys)).map { dataRecords => + FeatureMapBuilder() + .add(outputFeature, dataRecords.head) + .build() + } + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/real_time_aggregates/BaseRealtimeAggregateHydrator.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/real_time_aggregates/BaseRealtimeAggregateHydrator.scala new file mode 100644 index 0000000000..004312f2fc --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/real_time_aggregates/BaseRealtimeAggregateHydrator.scala @@ -0,0 +1,154 @@ +package com.twitter.home_mixer.functional_component.feature_hydrator.real_time_aggregates + +import com.twitter.home_mixer.functional_component.feature_hydrator.real_time_aggregates.BaseRealtimeAggregateHydrator._ +import com.twitter.home_mixer.util.ObservedKeyValueResultHandler +import com.twitter.ml.api.DataRecord +import com.twitter.ml.api.DataRecordMerger +import com.twitter.ml.api.FeatureContext +import com.twitter.ml.api.util.SRichDataRecord +import com.twitter.ml.api.{Feature => MLApiFeature} +import com.twitter.servo.cache.ReadCache +import com.twitter.servo.keyvalue.KeyValueResult +import com.twitter.stitch.Stitch +import com.twitter.timelines.data_processing.ml_util.aggregation_framework.AggregateGroup +import com.twitter.util.Future +import com.twitter.util.Time +import com.twitter.util.Try +import scala.collection.JavaConverters._ +import java.lang.{Double => JDouble} + +trait BaseRealtimeAggregateHydrator[K] extends ObservedKeyValueResultHandler { + + val client: ReadCache[K, DataRecord] + + val aggregateGroups: Seq[AggregateGroup] + + val aggregateGroupToPrefix: Map[AggregateGroup, String] = Map.empty + + private lazy val typedAggregateGroupsList = aggregateGroups.map(_.buildTypedAggregateGroups()) + + private lazy val featureContexts: Seq[FeatureContext] = typedAggregateGroupsList.map { + typedAggregateGroups => + new FeatureContext(typedAggregateGroups.flatMap(_.allOutputFeatures).asJava) + } + + private lazy val aggregateFeaturesRenameMap: Map[MLApiFeature[_], MLApiFeature[_]] = { + val prefixes: Seq[Option[String]] = aggregateGroups.map(aggregateGroupToPrefix.get) + + typedAggregateGroupsList + .zip(prefixes).map { + case (typedAggregateGroups, prefix) => + if (prefix.nonEmpty) + typedAggregateGroups + .map { + _.outputFeaturesToRenamedOutputFeatures(prefix.get) + }.reduce(_ ++ _) + else + Map.empty[MLApiFeature[_], MLApiFeature[_]] + }.reduce(_ ++ _) + } + + private lazy val renamedFeatureContexts: Seq[FeatureContext] = + typedAggregateGroupsList.map { typedAggregateGroups => + val renamedAllOutputFeatures = typedAggregateGroups.flatMap(_.allOutputFeatures).map { + feature => aggregateFeaturesRenameMap.getOrElse(feature, feature) + } + + new FeatureContext(renamedAllOutputFeatures.asJava) + } + + private lazy val decays: Seq[TimeDecay] = typedAggregateGroupsList.map { typedAggregateGroups => + RealTimeAggregateTimeDecay( + typedAggregateGroups.flatMap(_.continuousFeatureIdsToHalfLives).toMap) + .apply(_, _) + } + + private val drMerger = new DataRecordMerger + + private def postTransformer(dataRecord: Try[Option[DataRecord]]): Try[DataRecord] = { + dataRecord.map { + case Some(dr) => + val newDr = new DataRecord() + featureContexts.zip(renamedFeatureContexts).zip(decays).foreach { + case ((featureContext, renamedFeatureContext), decay) => + val decayedDr = applyDecay(dr, featureContext, decay) + val renamedDr = applyRename( + dataRecord = decayedDr, + featureContext, + renamedFeatureContext, + aggregateFeaturesRenameMap) + drMerger.merge(newDr, renamedDr) + } + newDr + case _ => + new DataRecord + } + } + + def fetchAndConstructDataRecords(possiblyKeys: Seq[Option[K]]): Stitch[Seq[Try[DataRecord]]] = { + Stitch.callFuture { + val keys = possiblyKeys.flatten + + val response: Future[KeyValueResult[K, DataRecord]] = + if (keys.isEmpty) { + Future.value(KeyValueResult.empty) + } else { + client.get(keys) + } + + response.map { result => + possiblyKeys.map { possiblyKey => + val value = observedGet(key = possiblyKey, keyValueResult = result) + postTransformer(value) + } + } + } + } +} + +object BaseRealtimeAggregateHydrator { + type TimeDecay = scala.Function2[com.twitter.ml.api.DataRecord, scala.Long, scala.Unit] + + private def applyDecay( + dataRecord: DataRecord, + featureContext: FeatureContext, + decay: TimeDecay + ): DataRecord = { + def time: Long = Time.now.inMillis + + val richFullDr = new SRichDataRecord(dataRecord, featureContext) + val richNewDr = new SRichDataRecord(new DataRecord, featureContext) + val featureIterator = featureContext.iterator() + featureIterator.forEachRemaining { feature => + if (richFullDr.hasFeature(feature)) { + val typedFeature = feature.asInstanceOf[MLApiFeature[JDouble]] + richNewDr.setFeatureValue(typedFeature, richFullDr.getFeatureValue(typedFeature)) + } + } + val resultDr = richNewDr.getRecord + decay(resultDr, time) + resultDr + } + + private def applyRename( + dataRecord: DataRecord, + featureContext: FeatureContext, + renamedFeatureContext: FeatureContext, + featureRenamingMap: Map[MLApiFeature[_], MLApiFeature[_]] + ): DataRecord = { + val richFullDr = new SRichDataRecord(dataRecord, featureContext) + val richNewDr = new SRichDataRecord(new DataRecord, renamedFeatureContext) + val featureIterator = featureContext.iterator() + featureIterator.forEachRemaining { feature => + if (richFullDr.hasFeature(feature)) { + val renamedFeature = featureRenamingMap.getOrElse(feature, feature) + + val typedFeature = feature.asInstanceOf[MLApiFeature[JDouble]] + val typedRenamedFeature = renamedFeature.asInstanceOf[MLApiFeature[JDouble]] + + richNewDr.setFeatureValue(typedRenamedFeature, richFullDr.getFeatureValue(typedFeature)) + } + } + richNewDr.getRecord + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/real_time_aggregates/EngagementsReceivedByAuthorRealTimeAggregateFeatureHydrator.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/real_time_aggregates/EngagementsReceivedByAuthorRealTimeAggregateFeatureHydrator.scala new file mode 100644 index 0000000000..57463dd79e --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/real_time_aggregates/EngagementsReceivedByAuthorRealTimeAggregateFeatureHydrator.scala @@ -0,0 +1,52 @@ +package com.twitter.home_mixer.functional_component.feature_hydrator.real_time_aggregates + +import com.google.inject.name.Named +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.home_mixer.param.HomeMixerInjectionNames.EngagementsReceivedByAuthorCache +import com.twitter.home_mixer.util.CandidatesUtil +import com.twitter.ml.api.DataRecord +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.feature.FeatureWithDefaultOnFailure +import com.twitter.product_mixer.core.feature.datarecord.DataRecordInAFeature +import com.twitter.product_mixer.core.model.common.CandidateWithFeatures +import com.twitter.product_mixer.core.model.common.identifier.FeatureHydratorIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.servo.cache.ReadCache +import com.twitter.timelines.data_processing.ml_util.aggregation_framework.AggregateGroup +import com.twitter.timelines.prediction.common.aggregates.real_time.TimelinesOnlineAggregationFeaturesOnlyConfig._ +import javax.inject.Inject +import javax.inject.Singleton + +object EngagementsReceivedByAuthorRealTimeAggregateFeature + extends DataRecordInAFeature[TweetCandidate] + with FeatureWithDefaultOnFailure[TweetCandidate, DataRecord] { + override def defaultValue: DataRecord = new DataRecord() +} + +@Singleton +class EngagementsReceivedByAuthorRealTimeAggregateFeatureHydrator @Inject() ( + @Named(EngagementsReceivedByAuthorCache) override val client: ReadCache[Long, DataRecord], + override val statsReceiver: StatsReceiver) + extends BaseRealTimeAggregateBulkCandidateFeatureHydrator[Long] { + + override val identifier: FeatureHydratorIdentifier = + FeatureHydratorIdentifier("EngagementsReceivedByAuthorRealTimeAggregate") + + override val outputFeature: DataRecordInAFeature[TweetCandidate] = + EngagementsReceivedByAuthorRealTimeAggregateFeature + + override val aggregateGroups: Seq[AggregateGroup] = Seq( + authorEngagementRealTimeAggregatesProd, + authorShareEngagementsRealTimeAggregates + ) + + override val aggregateGroupToPrefix: Map[AggregateGroup, String] = Map( + authorShareEngagementsRealTimeAggregates -> "original_author.timelines.author_share_engagements_real_time_aggregates." + ) + + override def keysFromQueryAndCandidates( + query: PipelineQuery, + candidates: Seq[CandidateWithFeatures[TweetCandidate]] + ): Seq[Option[Long]] = + candidates.map(candidate => CandidatesUtil.getOriginalAuthorId(candidate.features)) +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/real_time_aggregates/RealTimeAggregateTimeDecay.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/real_time_aggregates/RealTimeAggregateTimeDecay.scala new file mode 100644 index 0000000000..110a826353 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/real_time_aggregates/RealTimeAggregateTimeDecay.scala @@ -0,0 +1,50 @@ +package com.twitter.home_mixer.functional_component.feature_hydrator.real_time_aggregates + +import com.twitter.ml.api.DataRecord +import com.twitter.ml.api.constant.SharedFeatures.TIMESTAMP +import com.twitter.util.Duration + +/** + * The default TimeDecay implementation for real time aggregates. + * + * @param featureIdToHalfLife A precomputed map from aggregate feature ids to their half lives. + * @param timestampFeatureId A discrete timestamp feature id. + */ +case class RealTimeAggregateTimeDecay( + featureIdToHalfLife: Map[Long, Duration], + timestampFeatureId: Long = TIMESTAMP.getFeatureId) { + + /** + * Mutates the data record which is just a reference to the input. + * + * @param record Data record to apply decay to (is mutated). + * @param timeNow The current read time (in milliseconds) to decay counts forward to. + */ + def apply(record: DataRecord, timeNow: Long): Unit = { + if (record.isSetDiscreteFeatures) { + val discreteFeatures = record.getDiscreteFeatures + if (discreteFeatures.containsKey(timestampFeatureId)) { + if (record.isSetContinuousFeatures) { + val ctsFeatures = record.getContinuousFeatures + + val storedTimestamp: Long = discreteFeatures.get(timestampFeatureId) + val scaledDt = if (timeNow > storedTimestamp) { + (timeNow - storedTimestamp).toDouble * math.log(2) + } else 0.0 + featureIdToHalfLife.foreach { + case (featureId, halfLife) => + if (ctsFeatures.containsKey(featureId)) { + val storedValue = ctsFeatures.get(featureId) + val alpha = + if (halfLife.inMilliseconds != 0) math.exp(-scaledDt / halfLife.inMilliseconds) + else 0 + val decayedValue: Double = alpha * storedValue + record.putToContinuousFeatures(featureId, decayedValue) + } + } + } + discreteFeatures.remove(timestampFeatureId) + } + } + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/real_time_aggregates/TopicCountryEngagementRealTimeAggregateFeatureHydrator.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/real_time_aggregates/TopicCountryEngagementRealTimeAggregateFeatureHydrator.scala new file mode 100644 index 0000000000..d8ffad3112 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/real_time_aggregates/TopicCountryEngagementRealTimeAggregateFeatureHydrator.scala @@ -0,0 +1,64 @@ +package com.twitter.home_mixer.functional_component.feature_hydrator.real_time_aggregates + +import com.google.inject.name.Named +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.home_mixer.model.HomeFeatures.TopicIdSocialContextFeature +import com.twitter.home_mixer.param.HomeMixerInjectionNames.TopicCountryEngagementCache +import com.twitter.ml.api.DataRecord +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.feature.FeatureWithDefaultOnFailure +import com.twitter.product_mixer.core.feature.datarecord.DataRecordInAFeature +import com.twitter.product_mixer.core.model.common.CandidateWithFeatures +import com.twitter.product_mixer.core.model.common.identifier.FeatureHydratorIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.servo.cache.ReadCache +import com.twitter.timelines.data_processing.ml_util.aggregation_framework.AggregateGroup +import com.twitter.timelines.prediction.common.aggregates.real_time.TimelinesOnlineAggregationFeaturesOnlyConfig._ +import javax.inject.Inject +import javax.inject.Singleton + +object TopicCountryEngagementRealTimeAggregateFeature + extends DataRecordInAFeature[TweetCandidate] + with FeatureWithDefaultOnFailure[TweetCandidate, DataRecord] { + override def defaultValue: DataRecord = new DataRecord() +} + +@Singleton +class TopicCountryEngagementRealTimeAggregateFeatureHydrator @Inject() ( + @Named(TopicCountryEngagementCache) override val client: ReadCache[(Long, String), DataRecord], + override val statsReceiver: StatsReceiver) + extends BaseRealTimeAggregateBulkCandidateFeatureHydrator[(Long, String)] { + + override val identifier: FeatureHydratorIdentifier = + FeatureHydratorIdentifier("TopicCountryEngagementRealTimeAggregate") + + override val outputFeature: DataRecordInAFeature[TweetCandidate] = + TopicCountryEngagementRealTimeAggregateFeature + + override val aggregateGroups: Seq[AggregateGroup] = Seq( + topicCountryRealTimeAggregates + ) + + override val aggregateGroupToPrefix: Map[AggregateGroup, String] = Map( + topicCountryRealTimeAggregates -> "topic-country_code.timelines.topic_country_engagement_real_time_aggregates." + ) + + override def keysFromQueryAndCandidates( + query: PipelineQuery, + candidates: Seq[CandidateWithFeatures[TweetCandidate]] + ): Seq[Option[(Long, String)]] = { + candidates.map { candidate => + val maybeTopicId = candidate.features + .getTry(TopicIdSocialContextFeature) + .toOption + .flatten + + val maybeCountryCode = query.clientContext.countryCode + + for { + topicId <- maybeTopicId + countryCode <- maybeCountryCode + } yield (topicId, countryCode) + } + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/real_time_aggregates/TopicEngagementRealTimeAggregateFeatureHydrator.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/real_time_aggregates/TopicEngagementRealTimeAggregateFeatureHydrator.scala new file mode 100644 index 0000000000..6d71f45b31 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/real_time_aggregates/TopicEngagementRealTimeAggregateFeatureHydrator.scala @@ -0,0 +1,53 @@ +package com.twitter.home_mixer.functional_component.feature_hydrator.real_time_aggregates + +import com.google.inject.name.Named +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.home_mixer.model.HomeFeatures.TopicIdSocialContextFeature +import com.twitter.home_mixer.param.HomeMixerInjectionNames.TopicEngagementCache +import com.twitter.ml.api.DataRecord +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.feature.FeatureWithDefaultOnFailure +import com.twitter.product_mixer.core.feature.datarecord.DataRecordInAFeature +import com.twitter.product_mixer.core.model.common.CandidateWithFeatures +import com.twitter.product_mixer.core.model.common.identifier.FeatureHydratorIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.servo.cache.ReadCache +import com.twitter.timelines.data_processing.ml_util.aggregation_framework.AggregateGroup +import com.twitter.timelines.prediction.common.aggregates.real_time.TimelinesOnlineAggregationFeaturesOnlyConfig._ +import javax.inject.Inject +import javax.inject.Singleton + +object TopicEngagementRealTimeAggregateFeature + extends DataRecordInAFeature[TweetCandidate] + with FeatureWithDefaultOnFailure[TweetCandidate, DataRecord] { + override def defaultValue: DataRecord = new DataRecord() +} + +@Singleton +class TopicEngagementRealTimeAggregateFeatureHydrator @Inject() ( + @Named(TopicEngagementCache) override val client: ReadCache[Long, DataRecord], + override val statsReceiver: StatsReceiver) + extends BaseRealTimeAggregateBulkCandidateFeatureHydrator[Long] { + + override val identifier: FeatureHydratorIdentifier = + FeatureHydratorIdentifier("TopicEngagementRealTimeAggregate") + + override val outputFeature: DataRecordInAFeature[TweetCandidate] = + TopicEngagementRealTimeAggregateFeature + + override val aggregateGroups: Seq[AggregateGroup] = Seq( + topicEngagementRealTimeAggregatesProd + ) + + override def keysFromQueryAndCandidates( + query: PipelineQuery, + candidates: Seq[CandidateWithFeatures[TweetCandidate]] + ): Seq[Option[Long]] = { + candidates.map { candidate => + candidate.features + .getTry(TopicIdSocialContextFeature) + .toOption + .flatten + } + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/real_time_aggregates/TweetCountryEngagementRealTimeAggregateFeatureHydrator.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/real_time_aggregates/TweetCountryEngagementRealTimeAggregateFeatureHydrator.scala new file mode 100644 index 0000000000..4656b1945f --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/real_time_aggregates/TweetCountryEngagementRealTimeAggregateFeatureHydrator.scala @@ -0,0 +1,55 @@ +package com.twitter.home_mixer.functional_component.feature_hydrator.real_time_aggregates + +import com.google.inject.name.Named +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.home_mixer.param.HomeMixerInjectionNames.TweetCountryEngagementCache +import com.twitter.ml.api.DataRecord +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.feature.FeatureWithDefaultOnFailure +import com.twitter.product_mixer.core.feature.datarecord.DataRecordInAFeature +import com.twitter.product_mixer.core.model.common.CandidateWithFeatures +import com.twitter.product_mixer.core.model.common.identifier.FeatureHydratorIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.servo.cache.ReadCache +import com.twitter.timelines.data_processing.ml_util.aggregation_framework.AggregateGroup +import com.twitter.timelines.prediction.common.aggregates.real_time.TimelinesOnlineAggregationFeaturesOnlyConfig._ +import javax.inject.Inject +import javax.inject.Singleton + +object TweetCountryEngagementRealTimeAggregateFeature + extends DataRecordInAFeature[TweetCandidate] + with FeatureWithDefaultOnFailure[TweetCandidate, DataRecord] { + override def defaultValue: DataRecord = new DataRecord() +} + +@Singleton +class TweetCountryEngagementRealTimeAggregateFeatureHydrator @Inject() ( + @Named(TweetCountryEngagementCache) override val client: ReadCache[(Long, String), DataRecord], + override val statsReceiver: StatsReceiver) + extends BaseRealTimeAggregateBulkCandidateFeatureHydrator[(Long, String)] { + + override val identifier: FeatureHydratorIdentifier = + FeatureHydratorIdentifier("TweetCountryEngagementRealTimeAggregate") + + override val outputFeature: DataRecordInAFeature[TweetCandidate] = + TweetCountryEngagementRealTimeAggregateFeature + + override val aggregateGroups: Seq[AggregateGroup] = Seq( + tweetCountryRealTimeAggregates + ) + + override val aggregateGroupToPrefix: Map[AggregateGroup, String] = Map( + tweetCountryRealTimeAggregates -> "tweet-country_code.timelines.tweet_country_engagement_real_time_aggregates." + ) + + override def keysFromQueryAndCandidates( + query: PipelineQuery, + candidates: Seq[CandidateWithFeatures[TweetCandidate]] + ): Seq[Option[(Long, String)]] = { + val countryCode = query.clientContext.countryCode + candidates.map { candidate => + val tweetId = candidate.candidate.id + countryCode.map((tweetId, _)) + } + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/real_time_aggregates/TweetEngagementRealTimeAggregateFeatureHydrator.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/real_time_aggregates/TweetEngagementRealTimeAggregateFeatureHydrator.scala new file mode 100644 index 0000000000..5a607dec26 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/real_time_aggregates/TweetEngagementRealTimeAggregateFeatureHydrator.scala @@ -0,0 +1,49 @@ +package com.twitter.home_mixer.functional_component.feature_hydrator.real_time_aggregates + +import com.google.inject.name.Named +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.home_mixer.param.HomeMixerInjectionNames.TweetEngagementCache +import com.twitter.ml.api.DataRecord +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.feature.FeatureWithDefaultOnFailure +import com.twitter.product_mixer.core.feature.datarecord.DataRecordInAFeature +import com.twitter.product_mixer.core.model.common.CandidateWithFeatures +import com.twitter.product_mixer.core.model.common.identifier.FeatureHydratorIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.servo.cache.ReadCache +import com.twitter.timelines.data_processing.ml_util.aggregation_framework.AggregateGroup +import com.twitter.timelines.prediction.common.aggregates.real_time.TimelinesOnlineAggregationFeaturesOnlyConfig._ +import javax.inject.Inject +import javax.inject.Singleton + +object TweetEngagementRealTimeAggregateFeature + extends DataRecordInAFeature[TweetCandidate] + with FeatureWithDefaultOnFailure[TweetCandidate, DataRecord] { + override def defaultValue: DataRecord = new DataRecord() +} + +@Singleton +class TweetEngagementRealTimeAggregateFeatureHydrator @Inject() ( + @Named(TweetEngagementCache) override val client: ReadCache[Long, DataRecord], + override val statsReceiver: StatsReceiver) + extends BaseRealTimeAggregateBulkCandidateFeatureHydrator[Long] { + + override val identifier: FeatureHydratorIdentifier = + FeatureHydratorIdentifier("TweetEngagementRealTimeAggregate") + + override val outputFeature: DataRecordInAFeature[TweetCandidate] = + TweetEngagementRealTimeAggregateFeature + + override val aggregateGroups: Seq[AggregateGroup] = Seq( + tweetEngagement30MinuteCountsProd, + tweetEngagementTotalCountsProd + ) + + override def keysFromQueryAndCandidates( + query: PipelineQuery, + candidates: Seq[CandidateWithFeatures[TweetCandidate]] + ): Seq[Option[Long]] = { + candidates + .map(candidate => Some(candidate.candidate.id)) + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/real_time_aggregates/TwitterListEngagementRealTimeAggregateFeatureHydrator.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/real_time_aggregates/TwitterListEngagementRealTimeAggregateFeatureHydrator.scala new file mode 100644 index 0000000000..e6d7b86f1a --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/real_time_aggregates/TwitterListEngagementRealTimeAggregateFeatureHydrator.scala @@ -0,0 +1,57 @@ +package com.twitter.home_mixer.functional_component.feature_hydrator.real_time_aggregates + +import com.google.inject.name.Named +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.home_mixer.model.HomeFeatures.TwitterListIdFeature +import com.twitter.home_mixer.param.HomeMixerInjectionNames.TwitterListEngagementCache +import com.twitter.ml.api.DataRecord +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.feature.FeatureWithDefaultOnFailure +import com.twitter.product_mixer.core.feature.datarecord.DataRecordInAFeature +import com.twitter.product_mixer.core.model.common.CandidateWithFeatures +import com.twitter.product_mixer.core.model.common.identifier.FeatureHydratorIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.servo.cache.ReadCache +import com.twitter.timelines.data_processing.ml_util.aggregation_framework.AggregateGroup +import com.twitter.timelines.prediction.common.aggregates.real_time.TimelinesOnlineAggregationFeaturesOnlyConfig._ +import javax.inject.Inject +import javax.inject.Singleton + +object TwitterListEngagementRealTimeAggregateFeature + extends DataRecordInAFeature[TweetCandidate] + with FeatureWithDefaultOnFailure[TweetCandidate, DataRecord] { + override def defaultValue: DataRecord = new DataRecord() +} + +@Singleton +class TwitterListEngagementRealTimeAggregateFeatureHydrator @Inject() ( + @Named(TwitterListEngagementCache) override val client: ReadCache[Long, DataRecord], + override val statsReceiver: StatsReceiver) + extends BaseRealTimeAggregateBulkCandidateFeatureHydrator[Long] { + + override val identifier: FeatureHydratorIdentifier = + FeatureHydratorIdentifier("TwitterListEngagementRealTimeAggregate") + + override val outputFeature: DataRecordInAFeature[TweetCandidate] = + TwitterListEngagementRealTimeAggregateFeature + + override val aggregateGroups: Seq[AggregateGroup] = Seq( + listEngagementRealTimeAggregatesProd + ) + + override val aggregateGroupToPrefix: Map[AggregateGroup, String] = Map( + listEngagementRealTimeAggregatesProd -> "twitter_list.timelines.twitter_list_engagement_real_time_aggregates." + ) + + override def keysFromQueryAndCandidates( + query: PipelineQuery, + candidates: Seq[CandidateWithFeatures[TweetCandidate]] + ): Seq[Option[Long]] = { + candidates.map { candidate => + candidate.features + .getTry(TwitterListIdFeature) + .toOption + .flatten + } + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/real_time_aggregates/UserAuthorEngagementRealTimeAggregateFeatureHydrator.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/real_time_aggregates/UserAuthorEngagementRealTimeAggregateFeatureHydrator.scala new file mode 100644 index 0000000000..9671c7bf80 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/real_time_aggregates/UserAuthorEngagementRealTimeAggregateFeatureHydrator.scala @@ -0,0 +1,61 @@ +package com.twitter.home_mixer.functional_component.feature_hydrator.real_time_aggregates + +import com.google.inject.name.Named +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.home_mixer.model.HomeFeatures.AuthorIdFeature +import com.twitter.home_mixer.param.HomeMixerInjectionNames.UserAuthorEngagementCache +import com.twitter.ml.api.DataRecord +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.feature.FeatureWithDefaultOnFailure +import com.twitter.product_mixer.core.feature.datarecord.DataRecordInAFeature +import com.twitter.product_mixer.core.model.common.CandidateWithFeatures +import com.twitter.product_mixer.core.model.common.identifier.FeatureHydratorIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.servo.cache.ReadCache +import com.twitter.timelines.data_processing.ml_util.aggregation_framework.AggregateGroup +import com.twitter.timelines.prediction.common.aggregates.real_time.TimelinesOnlineAggregationFeaturesOnlyConfig._ +import javax.inject.Inject +import javax.inject.Singleton + +object UserAuthorEngagementRealTimeAggregateFeature + extends DataRecordInAFeature[TweetCandidate] + with FeatureWithDefaultOnFailure[TweetCandidate, DataRecord] { + override def defaultValue: DataRecord = new DataRecord() +} + +@Singleton +class UserAuthorEngagementRealTimeAggregateFeatureHydrator @Inject() ( + @Named(UserAuthorEngagementCache) override val client: ReadCache[(Long, Long), DataRecord], + override val statsReceiver: StatsReceiver) + extends BaseRealTimeAggregateBulkCandidateFeatureHydrator[(Long, Long)] { + + override val identifier: FeatureHydratorIdentifier = + FeatureHydratorIdentifier("UserAuthorEngagementRealTimeAggregate") + + override val outputFeature: DataRecordInAFeature[TweetCandidate] = + UserAuthorEngagementRealTimeAggregateFeature + + override val aggregateGroups: Seq[AggregateGroup] = Seq( + userAuthorEngagementRealTimeAggregatesProd, + userAuthorShareEngagementsRealTimeAggregates + ) + + override val aggregateGroupToPrefix: Map[AggregateGroup, String] = Map( + userAuthorEngagementRealTimeAggregatesProd -> "user-author.timelines.user_author_engagement_real_time_aggregates.", + userAuthorShareEngagementsRealTimeAggregates -> "user-author.timelines.user_author_share_engagements_real_time_aggregates." + ) + + override def keysFromQueryAndCandidates( + query: PipelineQuery, + candidates: Seq[CandidateWithFeatures[TweetCandidate]] + ): Seq[Option[(Long, Long)]] = { + val userId = query.getRequiredUserId + candidates.map { candidate => + candidate.features + .getTry(AuthorIdFeature) + .toOption + .flatten + .map((userId, _)) + } + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/real_time_aggregates/UserEngagementRealTimeAggregatesFeatureHydrator.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/real_time_aggregates/UserEngagementRealTimeAggregatesFeatureHydrator.scala new file mode 100644 index 0000000000..7526f75dfd --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/real_time_aggregates/UserEngagementRealTimeAggregatesFeatureHydrator.scala @@ -0,0 +1,56 @@ +package com.twitter.home_mixer.functional_component.feature_hydrator.real_time_aggregates + +import com.google.inject.name.Named +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.home_mixer.param.HomeMixerInjectionNames.UserEngagementCache +import com.twitter.ml.api.DataRecord +import com.twitter.product_mixer.core.feature.FeatureWithDefaultOnFailure +import com.twitter.product_mixer.core.feature.datarecord.DataRecordInAFeature +import com.twitter.product_mixer.core.model.common.identifier.FeatureHydratorIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.servo.cache.ReadCache +import com.twitter.timelines.data_processing.ml_util.aggregation_framework.AggregateGroup +import com.twitter.timelines.prediction.common.aggregates.real_time.TimelinesOnlineAggregationFeaturesOnlyConfig._ +import javax.inject.Inject +import javax.inject.Singleton + +object UserEngagementRealTimeAggregateFeature + extends DataRecordInAFeature[PipelineQuery] + with FeatureWithDefaultOnFailure[PipelineQuery, DataRecord] { + override def defaultValue: DataRecord = new DataRecord() +} + +@Singleton +class UserEngagementRealTimeAggregatesFeatureHydrator @Inject() ( + @Named(UserEngagementCache) override val client: ReadCache[Long, DataRecord], + override val statsReceiver: StatsReceiver) + extends BaseRealTimeAggregateQueryFeatureHydrator[Long] { + + override val identifier: FeatureHydratorIdentifier = + FeatureHydratorIdentifier("UserEngagementRealTimeAggregates") + + override val outputFeature: DataRecordInAFeature[PipelineQuery] = + UserEngagementRealTimeAggregateFeature + + val aggregateGroups: Seq[AggregateGroup] = Seq( + userEngagementRealTimeAggregatesProd, + userShareEngagementsRealTimeAggregates, + userBCEDwellEngagementsRealTimeAggregates, + userEngagement48HourRealTimeAggregatesProd, + userNegativeEngagementAuthorUserState72HourRealTimeAggregates, + userNegativeEngagementAuthorUserStateRealTimeAggregates, + userProfileEngagementRealTimeAggregates, + ) + + override val aggregateGroupToPrefix: Map[AggregateGroup, String] = Map( + userEngagementRealTimeAggregatesProd -> "user.timelines.user_share_engagements_real_time_aggregates.", + userBCEDwellEngagementsRealTimeAggregates -> "user.timelines.user_bce_dwell_engagements_real_time_aggregates.", + userEngagement48HourRealTimeAggregatesProd -> "user.timelines.user_engagement_48_hour_real_time_aggregates.", + userNegativeEngagementAuthorUserState72HourRealTimeAggregates -> "user.timelines.user_negative_engagement_author_user_state_72_hour_real_time_aggregates.", + userProfileEngagementRealTimeAggregates -> "user.timelines.user_profile_engagement_real_time_aggregates." + ) + + override def keysFromQueryAndCandidates(query: PipelineQuery): Option[Long] = { + Some(query.getRequiredUserId) + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/filter/BUILD.bazel b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/filter/BUILD.bazel new file mode 100644 index 0000000000..00885f70aa --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/filter/BUILD.bazel @@ -0,0 +1,26 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + strict_deps = True, + tags = ["bazel-compatible"], + dependencies = [ + "home-mixer/server/src/main/scala/com/twitter/home_mixer/model", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/model/request", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/param", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/util", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/feature_hydrator/query/impressed_tweets", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/candidate", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/filter", + "src/thrift/com/twitter/spam/rtf:safety-result-scala", + "src/thrift/com/twitter/timelines/impression:thrift-scala", + "src/thrift/com/twitter/tweetypie:service-scala", + "src/thrift/com/twitter/tweetypie:tweet-scala", + "stitch/stitch-core", + "stitch/stitch-tweetypie", + "timelinemixer/server/src/main/scala/com/twitter/timelinemixer/injection/store/persistence", + "timelineservice/common/src/main/scala/com/twitter/timelineservice/model", + ], + exports = [ + "timelinemixer/server/src/main/scala/com/twitter/timelinemixer/injection/store/persistence", + ], +) diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/filter/DropMaxCandidatesFilter.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/filter/DropMaxCandidatesFilter.scala new file mode 100644 index 0000000000..94943fc4a5 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/filter/DropMaxCandidatesFilter.scala @@ -0,0 +1,27 @@ +package com.twitter.home_mixer.functional_component.filter + +import com.twitter.product_mixer.core.functional_component.filter.Filter +import com.twitter.product_mixer.core.functional_component.filter.FilterResult +import com.twitter.product_mixer.core.model.common.CandidateWithFeatures +import com.twitter.product_mixer.core.model.common.UniversalNoun +import com.twitter.product_mixer.core.model.common.identifier.FilterIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.stitch.Stitch +import com.twitter.timelines.configapi.FSBoundedParam + +case class DropMaxCandidatesFilter[Candidate <: UniversalNoun[Any]]( + maxCandidatesParam: FSBoundedParam[Int]) + extends Filter[PipelineQuery, Candidate] { + + override val identifier: FilterIdentifier = FilterIdentifier("DropMaxCandidates") + + override def apply( + query: PipelineQuery, + candidates: Seq[CandidateWithFeatures[Candidate]] + ): Stitch[FilterResult[Candidate]] = { + val maxCandidates = query.params(maxCandidatesParam) + val (kept, removed) = candidates.map(_.candidate).splitAt(maxCandidates) + + Stitch.value(FilterResult(kept, removed)) + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/filter/FeedbackFatigueFilter.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/filter/FeedbackFatigueFilter.scala new file mode 100644 index 0000000000..5b6541f512 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/filter/FeedbackFatigueFilter.scala @@ -0,0 +1,89 @@ +package com.twitter.home_mixer.functional_component.filter + +import com.twitter.conversions.DurationOps._ +import com.twitter.home_mixer.model.HomeFeatures.AuthorIdFeature +import com.twitter.home_mixer.model.HomeFeatures.FeedbackHistoryFeature +import com.twitter.home_mixer.model.HomeFeatures.IsRetweetFeature +import com.twitter.home_mixer.model.HomeFeatures.SGSValidFollowedByUserIdsFeature +import com.twitter.home_mixer.model.HomeFeatures.SGSValidLikedByUserIdsFeature +import com.twitter.home_mixer.util.CandidatesUtil +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.functional_component.filter.Filter +import com.twitter.product_mixer.core.functional_component.filter.FilterResult +import com.twitter.product_mixer.core.model.common.CandidateWithFeatures +import com.twitter.product_mixer.core.model.common.identifier.FilterIdentifier +import com.twitter.product_mixer.core.pipeline +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.stitch.Stitch +import com.twitter.timelines.common.thriftscala.FeedbackEntity +import com.twitter.timelineservice.model.FeedbackEntry +import com.twitter.timelineservice.{thriftscala => tls} + +object FeedbackFatigueFilter + extends Filter[PipelineQuery, TweetCandidate] + with Filter.Conditionally[PipelineQuery, TweetCandidate] { + + override val identifier: FilterIdentifier = FilterIdentifier("FeedbackFatigue") + + override def onlyIf( + query: PipelineQuery, + candidates: Seq[CandidateWithFeatures[TweetCandidate]] + ): Boolean = + query.features.exists(_.getOrElse(FeedbackHistoryFeature, Seq.empty).nonEmpty) + + private val DurationForFiltering = 14.days + + override def apply( + query: pipeline.PipelineQuery, + candidates: Seq[CandidateWithFeatures[TweetCandidate]] + ): Stitch[FilterResult[TweetCandidate]] = { + val feedbackEntriesByEngagementType = + query.features + .getOrElse(FeatureMap.empty).getOrElse(FeedbackHistoryFeature, Seq.empty) + .filter { entry => + val timeSinceFeedback = query.queryTime.minus(entry.timestamp) + timeSinceFeedback < DurationForFiltering && + entry.feedbackType == tls.FeedbackType.SeeFewer + }.groupBy(_.engagementType) + + val authorsToFilter = + getUserIds( + feedbackEntriesByEngagementType.getOrElse(tls.FeedbackEngagementType.Tweet, Seq.empty)) + val likersToFilter = + getUserIds( + feedbackEntriesByEngagementType.getOrElse(tls.FeedbackEngagementType.Like, Seq.empty)) + val followersToFilter = + getUserIds( + feedbackEntriesByEngagementType.getOrElse(tls.FeedbackEngagementType.Follow, Seq.empty)) + val retweetersToFilter = + getUserIds( + feedbackEntriesByEngagementType.getOrElse(tls.FeedbackEngagementType.Retweet, Seq.empty)) + + val (removed, kept) = candidates.partition { candidate => + val originalAuthorId = CandidatesUtil.getOriginalAuthorId(candidate.features) + val authorId = candidate.features.getOrElse(AuthorIdFeature, None) + + val likers = candidate.features.getOrElse(SGSValidLikedByUserIdsFeature, Seq.empty) + val eligibleLikers = likers.filterNot(likersToFilter.contains) + + val followers = candidate.features.getOrElse(SGSValidFollowedByUserIdsFeature, Seq.empty) + val eligibleFollowers = followers.filterNot(followersToFilter.contains) + + originalAuthorId.exists(authorsToFilter.contains) || + (likers.nonEmpty && eligibleLikers.isEmpty) || + (followers.nonEmpty && eligibleFollowers.isEmpty) || + (candidate.features.getOrElse(IsRetweetFeature, false) && + authorId.exists(retweetersToFilter.contains)) + } + + Stitch.value(FilterResult(kept = kept.map(_.candidate), removed = removed.map(_.candidate))) + } + + private def getUserIds( + feedbackEntries: Seq[FeedbackEntry], + ): Set[Long] = + feedbackEntries.collect { + case FeedbackEntry(_, _, FeedbackEntity.UserId(userId), _, _) => userId + }.toSet +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/filter/InvalidConversationModuleFilter.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/filter/InvalidConversationModuleFilter.scala new file mode 100644 index 0000000000..0d3f1ca1f5 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/filter/InvalidConversationModuleFilter.scala @@ -0,0 +1,50 @@ +package com.twitter.home_mixer.functional_component.filter + +import com.twitter.home_mixer.model.HomeFeatures.ConversationModuleFocalTweetIdFeature +import com.twitter.home_mixer.model.HomeFeatures.InReplyToTweetIdFeature +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.functional_component.filter.Filter +import com.twitter.product_mixer.core.functional_component.filter.FilterResult +import com.twitter.product_mixer.core.model.common.CandidateWithFeatures +import com.twitter.product_mixer.core.model.common.identifier.FilterIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.stitch.Stitch + +/** + * Exclude conversation modules where Tweets have been dropped by other filters + * + * Largest conversation modules have 3 Tweets, so if all 3 are present, module is valid. + * For 2 Tweet modules, check if the head is the root (not a reply) and the last item + * is actually replying to the root directly with no missing intermediate tweets + */ +object InvalidConversationModuleFilter extends Filter[PipelineQuery, TweetCandidate] { + + override val identifier: FilterIdentifier = FilterIdentifier("InvalidConversationModule") + + val ValidThreeTweetModuleSize = 3 + val ValidTwoTweetModuleSize = 2 + + override def apply( + query: PipelineQuery, + candidates: Seq[CandidateWithFeatures[TweetCandidate]] + ): Stitch[FilterResult[TweetCandidate]] = { + val allowedTweetIds = candidates + .groupBy(_.features.getOrElse(ConversationModuleFocalTweetIdFeature, None)) + .map { case (id, candidates) => (id, candidates.sortBy(_.candidate.id)) } + .filter { + case (Some(_), conversation) if conversation.size == ValidThreeTweetModuleSize => true + case (Some(focalId), conversation) if conversation.size == ValidTwoTweetModuleSize => + conversation.head.features.getOrElse(InReplyToTweetIdFeature, None).isEmpty && + conversation.last.candidate.id == focalId && + conversation.last.features + .getOrElse(InReplyToTweetIdFeature, None) + .contains(conversation.head.candidate.id) + case (None, _) => true + case _ => false + }.values.flatten.toSeq.map(_.candidate.id).toSet + + val (kept, removed) = + candidates.map(_.candidate).partition(candidate => allowedTweetIds.contains(candidate.id)) + Stitch.value(FilterResult(kept = kept, removed = removed)) + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/filter/KeepBestOutOfNetworkTweetPerAuthorFilter.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/filter/KeepBestOutOfNetworkTweetPerAuthorFilter.scala new file mode 100644 index 0000000000..87c324b897 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/filter/KeepBestOutOfNetworkTweetPerAuthorFilter.scala @@ -0,0 +1,36 @@ +package com.twitter.home_mixer.functional_component.filter + +import com.twitter.home_mixer.model.HomeFeatures.AuthorIdFeature +import com.twitter.home_mixer.model.HomeFeatures.InNetworkFeature +import com.twitter.home_mixer.model.HomeFeatures.ScoreFeature +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.functional_component.filter.Filter +import com.twitter.product_mixer.core.functional_component.filter.FilterResult +import com.twitter.product_mixer.core.model.common.CandidateWithFeatures +import com.twitter.product_mixer.core.model.common.identifier.FilterIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.stitch.Stitch + +object KeepBestOutOfNetworkTweetPerAuthorFilter extends Filter[PipelineQuery, TweetCandidate] { + + override val identifier: FilterIdentifier = FilterIdentifier("KeepBestOutOfNetworkTweetPerAuthor") + + override def apply( + query: PipelineQuery, + candidates: Seq[CandidateWithFeatures[TweetCandidate]] + ): Stitch[FilterResult[TweetCandidate]] = { + // Set containing best OON tweet for each authorId + val bestCandidatesForAuthorId = candidates + .filter(!_.features.getOrElse(InNetworkFeature, true)) + .groupBy(_.features.getOrElse(AuthorIdFeature, None)) + .values.map(_.maxBy(_.features.getOrElse(ScoreFeature, None))) + .toSet + + val (removed, kept) = candidates.partition { candidate => + !candidate.features.getOrElse(InNetworkFeature, true) && + !bestCandidatesForAuthorId.contains(candidate) + } + + Stitch.value(FilterResult(kept = kept.map(_.candidate), removed = removed.map(_.candidate))) + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/filter/OutOfNetworkCompetitorFilter.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/filter/OutOfNetworkCompetitorFilter.scala new file mode 100644 index 0000000000..5bbf35185e --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/filter/OutOfNetworkCompetitorFilter.scala @@ -0,0 +1,38 @@ +package com.twitter.home_mixer.functional_component.filter + +import com.twitter.home_mixer.model.HomeFeatures.AuthorIdFeature +import com.twitter.home_mixer.model.HomeFeatures.InNetworkFeature +import com.twitter.home_mixer.model.HomeFeatures.IsRetweetFeature +import com.twitter.home_mixer.product.scored_tweets.param.ScoredTweetsParam.CompetitorSetParam +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.functional_component.filter.Filter +import com.twitter.product_mixer.core.functional_component.filter.FilterResult +import com.twitter.product_mixer.core.model.common.CandidateWithFeatures +import com.twitter.product_mixer.core.model.common.identifier.FilterIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.stitch.Stitch + +object OutOfNetworkCompetitorFilter extends Filter[PipelineQuery, TweetCandidate] { + + override val identifier: FilterIdentifier = FilterIdentifier("OutOfNetworkCompetitor") + + override def apply( + query: PipelineQuery, + candidates: Seq[CandidateWithFeatures[TweetCandidate]] + ): Stitch[FilterResult[TweetCandidate]] = { + val competitorAuthors = query.params(CompetitorSetParam) + val (removed, kept) = + candidates.partition(isOutOfNetworkTweetFromCompetitor(_, competitorAuthors)) + + Stitch.value(FilterResult(kept = kept.map(_.candidate), removed = removed.map(_.candidate))) + } + + def isOutOfNetworkTweetFromCompetitor( + candidate: CandidateWithFeatures[TweetCandidate], + competitorAuthors: Set[Long] + ): Boolean = { + !candidate.features.getOrElse(InNetworkFeature, true) && + !candidate.features.getOrElse(IsRetweetFeature, false) && + candidate.features.getOrElse(AuthorIdFeature, None).exists(competitorAuthors.contains) + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/filter/OutOfNetworkCompetitorURLFilter.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/filter/OutOfNetworkCompetitorURLFilter.scala new file mode 100644 index 0000000000..f7163bee32 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/filter/OutOfNetworkCompetitorURLFilter.scala @@ -0,0 +1,37 @@ +package com.twitter.home_mixer.functional_component.filter + +import com.twitter.home_mixer.model.HomeFeatures.InNetworkFeature +import com.twitter.home_mixer.model.HomeFeatures.IsRetweetFeature +import com.twitter.home_mixer.model.HomeFeatures.TweetUrlsFeature +import com.twitter.home_mixer.product.scored_tweets.param.ScoredTweetsParam.CompetitorURLSeqParam +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.functional_component.filter.Filter +import com.twitter.product_mixer.core.functional_component.filter.FilterResult +import com.twitter.product_mixer.core.model.common.CandidateWithFeatures +import com.twitter.product_mixer.core.model.common.identifier.FilterIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.stitch.Stitch + +object OutOfNetworkCompetitorURLFilter extends Filter[PipelineQuery, TweetCandidate] { + + override val identifier: FilterIdentifier = FilterIdentifier("OutOfNetworkCompetitorURL") + + override def apply( + query: PipelineQuery, + candidates: Seq[CandidateWithFeatures[TweetCandidate]] + ): Stitch[FilterResult[TweetCandidate]] = { + val competitorUrls = query.params(CompetitorURLSeqParam).toSet + val (removed, kept) = candidates.partition(hasOutOfNetworkUrlFromCompetitor(_, competitorUrls)) + + Stitch.value(FilterResult(kept = kept.map(_.candidate), removed = removed.map(_.candidate))) + } + + def hasOutOfNetworkUrlFromCompetitor( + candidate: CandidateWithFeatures[TweetCandidate], + competitorUrls: Set[String] + ): Boolean = { + !candidate.features.getOrElse(InNetworkFeature, true) && + !candidate.features.getOrElse(IsRetweetFeature, false) && + candidate.features.get(TweetUrlsFeature).toSet.intersect(competitorUrls).nonEmpty + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/filter/PredicateFeatureFilter.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/filter/PredicateFeatureFilter.scala new file mode 100644 index 0000000000..ae1fe9f49a --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/filter/PredicateFeatureFilter.scala @@ -0,0 +1,59 @@ +package com.twitter.home_mixer.functional_component.filter + +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.functional_component.filter.Filter +import com.twitter.product_mixer.core.functional_component.filter.FilterResult +import com.twitter.product_mixer.core.model.common.CandidateWithFeatures +import com.twitter.product_mixer.core.model.common.UniversalNoun +import com.twitter.product_mixer.core.model.common.identifier.FilterIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.stitch.Stitch + +/** + * Predicate which will be applied to each candidate. True indicates that the candidate will be + * @tparam Candidate - the type of the candidate + */ +trait ShouldKeepCandidate { + def apply(features: FeatureMap): Boolean +} + +object PredicateFeatureFilter { + + /** + * Builds a simple Filter out of a predicate function from the candidate to a boolean. For clarity, + * we recommend including the name of the shouldKeepCandidate parameter. + * + * @param identifier A FilterIdentifier for the new filter + * @param shouldKeepCandidate A predicate function. Candidates will be kept when + * this function returns True. + */ + def fromPredicate[Candidate <: UniversalNoun[Any]]( + identifier: FilterIdentifier, + shouldKeepCandidate: ShouldKeepCandidate + ): Filter[PipelineQuery, Candidate] = { + val i = identifier + + new Filter[PipelineQuery, Candidate] { + override val identifier: FilterIdentifier = i + + /** + * Filter the list of candidates + * + * @return a FilterResult including both the list of kept candidate and the list of removed candidates + */ + override def apply( + query: PipelineQuery, + candidates: Seq[CandidateWithFeatures[Candidate]] + ): Stitch[FilterResult[Candidate]] = { + val allowedIds = candidates + .filter(candidate => shouldKeepCandidate(candidate.features)).map(_.candidate.id).toSet + + val (keptCandidates, removedCandidates) = candidates.map(_.candidate).partition { + candidate => allowedIds.contains(candidate.id) + } + + Stitch.value(FilterResult(kept = keptCandidates, removed = removedCandidates)) + } + } + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/filter/PredicateGatedFilter.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/filter/PredicateGatedFilter.scala new file mode 100644 index 0000000000..b263a87fc4 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/filter/PredicateGatedFilter.scala @@ -0,0 +1,47 @@ +package com.twitter.home_mixer.functional_component.filter + +import com.twitter.product_mixer.core.functional_component.common.alert.Alert +import com.twitter.product_mixer.core.functional_component.filter.Filter +import com.twitter.product_mixer.core.functional_component.filter.FilterResult +import com.twitter.product_mixer.core.model.common.CandidateWithFeatures +import com.twitter.product_mixer.core.model.common.Conditionally +import com.twitter.product_mixer.core.model.common.UniversalNoun +import com.twitter.product_mixer.core.model.common.identifier.FilterIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.stitch.Stitch + +trait FilterPredicate[-Query <: PipelineQuery] { + def apply(query: Query): Boolean +} + +/** + * A [[Filter]] with [[Conditionally]] based on a [[FilterPredicate]] + * + * @param predicate the predicate to turn this filter on and off + * @param filter the underlying filter to run when `predicate` is true + * @tparam Query The domain model for the query or request + * @tparam Candidate The type of the candidates + */ +case class PredicateGatedFilter[-Query <: PipelineQuery, Candidate <: UniversalNoun[Any]]( + predicate: FilterPredicate[Query], + filter: Filter[Query, Candidate]) + extends Filter[Query, Candidate] + with Filter.Conditionally[Query, Candidate] { + + override val identifier: FilterIdentifier = FilterIdentifier( + PredicateGatedFilter.IdentifierPrefix + filter.identifier.name) + + override val alerts: Seq[Alert] = filter.alerts + + override def onlyIf(query: Query, candidates: Seq[CandidateWithFeatures[Candidate]]): Boolean = + Conditionally.and(Filter.Input(query, candidates), filter, predicate(query)) + + override def apply( + query: Query, + candidates: Seq[CandidateWithFeatures[Candidate]] + ): Stitch[FilterResult[Candidate]] = filter.apply(query, candidates) +} + +object PredicateGatedFilter { + val IdentifierPrefix = "PredicateGated" +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/filter/PreviouslySeenTweetsFilter.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/filter/PreviouslySeenTweetsFilter.scala new file mode 100644 index 0000000000..047233a417 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/filter/PreviouslySeenTweetsFilter.scala @@ -0,0 +1,37 @@ +package com.twitter.home_mixer.functional_component.filter + +import com.twitter.home_mixer.util.CandidatesUtil +import com.twitter.home_mixer.util.TweetImpressionsHelper +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.functional_component.filter.Filter +import com.twitter.product_mixer.core.functional_component.filter.FilterResult +import com.twitter.product_mixer.core.model.common.CandidateWithFeatures +import com.twitter.product_mixer.core.model.common.identifier.FilterIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.stitch.Stitch + +/** + * Filter out users' previously seen tweets from 2 sources: + * 1. Heron Topology Impression Store in Memcache; + * 2. Manhattan Impression Store; + */ +object PreviouslySeenTweetsFilter extends Filter[PipelineQuery, TweetCandidate] { + + override val identifier: FilterIdentifier = FilterIdentifier("PreviouslySeenTweets") + + override def apply( + query: PipelineQuery, + candidates: Seq[CandidateWithFeatures[TweetCandidate]] + ): Stitch[FilterResult[TweetCandidate]] = { + + val seenTweetIds = + query.features.map(TweetImpressionsHelper.tweetImpressions).getOrElse(Set.empty) + + val (removed, kept) = candidates.partition { candidate => + val tweetIdAndSourceId = CandidatesUtil.getTweetIdAndSourceId(candidate) + tweetIdAndSourceId.exists(seenTweetIds.contains) + } + + Stitch.value(FilterResult(kept = kept.map(_.candidate), removed = removed.map(_.candidate))) + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/filter/PreviouslyServedAncestorsFilter.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/filter/PreviouslyServedAncestorsFilter.scala new file mode 100644 index 0000000000..42b0ad51a0 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/filter/PreviouslyServedAncestorsFilter.scala @@ -0,0 +1,44 @@ +package com.twitter.home_mixer.functional_component.filter + +import com.twitter.common_internal.analytics.twitter_client_user_agent_parser.UserAgent +import com.twitter.home_mixer.model.HomeFeatures.IsAncestorCandidateFeature +import com.twitter.home_mixer.model.HomeFeatures.PersistenceEntriesFeature +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.functional_component.filter.Filter +import com.twitter.product_mixer.core.functional_component.filter.FilterResult +import com.twitter.product_mixer.core.model.common.CandidateWithFeatures +import com.twitter.product_mixer.core.model.common.identifier.FilterIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.stitch.Stitch +import com.twitter.timelinemixer.injection.store.persistence.TimelinePersistenceUtils +import com.twitter.timelines.util.client_info.ClientPlatform + +object PreviouslyServedAncestorsFilter + extends Filter[PipelineQuery, TweetCandidate] + with TimelinePersistenceUtils { + + override val identifier: FilterIdentifier = FilterIdentifier("PreviouslyServedAncestors") + + override def apply( + query: PipelineQuery, + candidates: Seq[CandidateWithFeatures[TweetCandidate]] + ): Stitch[FilterResult[TweetCandidate]] = { + val clientPlatform = ClientPlatform.fromQueryOptions( + clientAppId = query.clientContext.appId, + userAgent = query.clientContext.userAgent.flatMap(UserAgent.fromString)) + val entries = + query.features.map(_.getOrElse(PersistenceEntriesFeature, Seq.empty)).toSeq.flatten + val tweetIds = applicableResponses(clientPlatform, entries) + .flatMap(_.entries.flatMap(_.tweetIds(includeSourceTweets = true))).toSet + val ancestorIds = + candidates + .filter(_.features.getOrElse(IsAncestorCandidateFeature, false)).map(_.candidate.id).toSet + + val (removed, kept) = + candidates + .map(_.candidate).partition(candidate => + tweetIds.contains(candidate.id) && ancestorIds.contains(candidate.id)) + + Stitch.value(FilterResult(kept = kept, removed = removed)) + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/filter/PreviouslyServedTweetsFilter.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/filter/PreviouslyServedTweetsFilter.scala new file mode 100644 index 0000000000..56b6a1c0b1 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/filter/PreviouslyServedTweetsFilter.scala @@ -0,0 +1,42 @@ +package com.twitter.home_mixer.functional_component.filter + +import com.twitter.home_mixer.model.HomeFeatures.GetOlderFeature +import com.twitter.home_mixer.model.HomeFeatures.ServedTweetIdsFeature +import com.twitter.home_mixer.util.CandidatesUtil +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.functional_component.filter.Filter +import com.twitter.product_mixer.core.functional_component.filter.FilterResult +import com.twitter.product_mixer.core.model.common.CandidateWithFeatures +import com.twitter.product_mixer.core.model.common.identifier.FilterIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.stitch.Stitch + +object PreviouslyServedTweetsFilter + extends Filter[PipelineQuery, TweetCandidate] + with Filter.Conditionally[PipelineQuery, TweetCandidate] { + + override val identifier: FilterIdentifier = FilterIdentifier("PreviouslyServedTweets") + + override def onlyIf( + query: PipelineQuery, + candidates: Seq[CandidateWithFeatures[TweetCandidate]] + ): Boolean = { + query.features.exists(_.getOrElse(GetOlderFeature, false)) + } + + override def apply( + query: PipelineQuery, + candidates: Seq[CandidateWithFeatures[TweetCandidate]] + ): Stitch[FilterResult[TweetCandidate]] = { + + val servedTweetIds = + query.features.map(_.getOrElse(ServedTweetIdsFeature, Seq.empty)).toSeq.flatten.toSet + + val (removed, kept) = candidates.partition { candidate => + val tweetIdAndSourceId = CandidatesUtil.getTweetIdAndSourceId(candidate) + tweetIdAndSourceId.exists(servedTweetIds.contains) + } + + Stitch.value(FilterResult(kept = kept.map(_.candidate), removed = removed.map(_.candidate))) + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/filter/RejectTweetFromViewerFilter.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/filter/RejectTweetFromViewerFilter.scala new file mode 100644 index 0000000000..c25d9ec2b9 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/filter/RejectTweetFromViewerFilter.scala @@ -0,0 +1,24 @@ +package com.twitter.home_mixer.functional_component.filter + +import com.twitter.home_mixer.util.CandidatesUtil +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.functional_component.filter.Filter +import com.twitter.product_mixer.core.functional_component.filter.FilterResult +import com.twitter.product_mixer.core.model.common.CandidateWithFeatures +import com.twitter.product_mixer.core.model.common.identifier.FilterIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.stitch.Stitch + +object RejectTweetFromViewerFilter extends Filter[PipelineQuery, TweetCandidate] { + + override val identifier: FilterIdentifier = FilterIdentifier("RejectTweetFromViewer") + + override def apply( + query: PipelineQuery, + candidates: Seq[CandidateWithFeatures[TweetCandidate]] + ): Stitch[FilterResult[TweetCandidate]] = { + val (removed, kept) = candidates.partition(candidate => + CandidatesUtil.isAuthoredByViewer(query, candidate.features)) + Stitch.value(FilterResult(kept = kept.map(_.candidate), removed = removed.map(_.candidate))) + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/filter/RetweetDeduplicationFilter.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/filter/RetweetDeduplicationFilter.scala new file mode 100644 index 0000000000..1e1f7f03ad --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/filter/RetweetDeduplicationFilter.scala @@ -0,0 +1,45 @@ +package com.twitter.home_mixer.functional_component.filter + +import com.twitter.home_mixer.model.HomeFeatures.IsRetweetFeature +import com.twitter.home_mixer.util.CandidatesUtil +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.functional_component.filter.Filter +import com.twitter.product_mixer.core.functional_component.filter.FilterResult +import com.twitter.product_mixer.core.model.common.CandidateWithFeatures +import com.twitter.product_mixer.core.model.common.identifier.FilterIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.stitch.Stitch +import scala.collection.mutable + +object RetweetDeduplicationFilter extends Filter[PipelineQuery, TweetCandidate] { + + override val identifier: FilterIdentifier = FilterIdentifier("RetweetDeduplication") + + override def apply( + query: PipelineQuery, + candidates: Seq[CandidateWithFeatures[TweetCandidate]] + ): Stitch[FilterResult[TweetCandidate]] = { + // If there are 2 retweets of the same native tweet, we will choose the first one + // The tweets are returned in descending score order, so we will choose the higher scored tweet + val dedupedTweetIdsSet = + candidates.partition(_.features.getOrElse(IsRetweetFeature, false)) match { + case (retweets, nativeTweets) => + val nativeTweetIds = nativeTweets.map(_.candidate.id) + val seenTweetIds = mutable.Set[Long]() ++ nativeTweetIds + val dedupedRetweets = retweets.filter { retweet => + val tweetIdAndSourceId = CandidatesUtil.getTweetIdAndSourceId(retweet) + val retweetIsUnique = tweetIdAndSourceId.forall(!seenTweetIds.contains(_)) + if (retweetIsUnique) { + seenTweetIds ++= tweetIdAndSourceId + } + retweetIsUnique + } + (nativeTweets ++ dedupedRetweets).map(_.candidate.id).toSet + } + + val (kept, removed) = + candidates + .map(_.candidate).partition(candidate => dedupedTweetIdsSet.contains(candidate.id)) + Stitch.value(FilterResult(kept = kept, removed = removed)) + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/filter/RetweetSourceTweetRemovingFilter.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/filter/RetweetSourceTweetRemovingFilter.scala new file mode 100644 index 0000000000..6dc88f02c3 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/filter/RetweetSourceTweetRemovingFilter.scala @@ -0,0 +1,40 @@ +package com.twitter.home_mixer.functional_component.filter + +import com.twitter.home_mixer.model.HomeFeatures.EarlybirdFeature +import com.twitter.home_mixer.model.HomeFeatures.InReplyToTweetIdFeature +import com.twitter.home_mixer.util.ReplyRetweetUtil +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.functional_component.filter.Filter +import com.twitter.product_mixer.core.functional_component.filter.FilterResult +import com.twitter.product_mixer.core.model.common.CandidateWithFeatures +import com.twitter.product_mixer.core.model.common.identifier.FilterIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.stitch.Stitch + +/** + * This filter removes source tweets of retweets, added via second EB call in TLR + */ +object RetweetSourceTweetRemovingFilter extends Filter[PipelineQuery, TweetCandidate] { + + override val identifier: FilterIdentifier = FilterIdentifier("RetweetSourceTweetRemoving") + + override def apply( + query: PipelineQuery, + candidates: Seq[CandidateWithFeatures[TweetCandidate]] + ): Stitch[FilterResult[TweetCandidate]] = { + val (kept, removed) = + candidates.partition( + _.features.getOrElse(EarlybirdFeature, None).exists(_.isSourceTweet)) match { + case (sourceTweets, nonSourceTweets) => + val inReplyToTweetIds: Set[Long] = + nonSourceTweets + .filter(ReplyRetweetUtil.isEligibleReply(_)).flatMap( + _.features.getOrElse(InReplyToTweetIdFeature, None)).toSet + val (keptSourceTweets, removedSourceTweets) = sourceTweets + .map(_.candidate) + .partition(candidate => inReplyToTweetIds.contains(candidate.id)) + (nonSourceTweets.map(_.candidate) ++ keptSourceTweets, removedSourceTweets) + } + Stitch.value(FilterResult(kept = kept, removed = removed)) + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/filter/SocialContextFilter.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/filter/SocialContextFilter.scala new file mode 100644 index 0000000000..5002137c43 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/filter/SocialContextFilter.scala @@ -0,0 +1,57 @@ +package com.twitter.home_mixer.functional_component.filter + +import com.twitter.home_mixer.model.HomeFeatures.ConversationModuleFocalTweetIdFeature +import com.twitter.home_mixer.model.HomeFeatures.InNetworkFeature +import com.twitter.home_mixer.model.HomeFeatures.PerspectiveFilteredLikedByUserIdsFeature +import com.twitter.home_mixer.model.HomeFeatures.SGSValidFollowedByUserIdsFeature +import com.twitter.home_mixer.model.HomeFeatures.SGSValidLikedByUserIdsFeature +import com.twitter.home_mixer.model.HomeFeatures.TopicContextFunctionalityTypeFeature +import com.twitter.home_mixer.model.HomeFeatures.TopicIdSocialContextFeature +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.functional_component.filter.Filter +import com.twitter.product_mixer.core.functional_component.filter.FilterResult +import com.twitter.product_mixer.core.model.common.CandidateWithFeatures +import com.twitter.product_mixer.core.model.common.identifier.FilterIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.stitch.Stitch + +object SocialContextFilter extends Filter[PipelineQuery, TweetCandidate] { + + override val identifier: FilterIdentifier = FilterIdentifier("SocialContext") + + override def apply( + query: PipelineQuery, + candidates: Seq[CandidateWithFeatures[TweetCandidate]] + ): Stitch[FilterResult[TweetCandidate]] = { + val validTweetIds = candidates + .filter { candidate => + candidate.features.getOrElse(InNetworkFeature, true) || + hasLikedBySocialContext(candidate.features) || + hasFollowedBySocialContext(candidate.features) || + hasTopicSocialContext(candidate.features) || + candidate.features.getOrElse(ConversationModuleFocalTweetIdFeature, None).isDefined + }.map(_.candidate.id).toSet + + val (kept, removed) = + candidates.map(_.candidate).partition(candidate => validTweetIds.contains(candidate.id)) + + Stitch.value(FilterResult(kept = kept, removed = removed)) + } + + private def hasLikedBySocialContext(candidateFeatures: FeatureMap): Boolean = + candidateFeatures + .getOrElse(SGSValidLikedByUserIdsFeature, Seq.empty) + .exists( + candidateFeatures + .getOrElse(PerspectiveFilteredLikedByUserIdsFeature, Seq.empty) + .toSet.contains + ) + + private def hasFollowedBySocialContext(candidateFeatures: FeatureMap): Boolean = + candidateFeatures.getOrElse(SGSValidFollowedByUserIdsFeature, Seq.empty).nonEmpty + + private def hasTopicSocialContext(candidateFeatures: FeatureMap): Boolean = + candidateFeatures.getOrElse(TopicIdSocialContextFeature, None).isDefined && + candidateFeatures.getOrElse(TopicContextFunctionalityTypeFeature, None).isDefined +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/gate/BUILD.bazel b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/gate/BUILD.bazel new file mode 100644 index 0000000000..d3b77dda87 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/gate/BUILD.bazel @@ -0,0 +1,21 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + strict_deps = True, + tags = ["bazel-compatible"], + dependencies = [ + "configapi/configapi-core/src/main/scala/com/twitter/timelines/configapi", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/model", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/model/request", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/util", + "home-mixer/thrift/src/main/thrift:thrift-scala", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/common", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/gate", + "src/thrift/com/twitter/gizmoduck:thrift-scala", + "stitch/stitch-socialgraph", + "timelinemixer/common/src/main/scala/com/twitter/timelinemixer/clients/manhattan", + "timelinemixer/server/src/main/scala/com/twitter/timelinemixer/injection/store/persistence", + "timelineservice/common/src/main/scala/com/twitter/timelineservice/model", + ], +) diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/gate/DismissFatigueGate.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/gate/DismissFatigueGate.scala new file mode 100644 index 0000000000..c97a3931eb --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/gate/DismissFatigueGate.scala @@ -0,0 +1,48 @@ +package com.twitter.home_mixer.functional_component.gate + +import com.twitter.product_mixer.core.functional_component.gate.Gate +import com.twitter.product_mixer.core.model.common.identifier.GateIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.timelinemixer.clients.manhattan.DismissInfo +import com.twitter.stitch.Stitch +import com.twitter.util.Duration +import com.twitter.conversions.DurationOps._ +import com.twitter.product_mixer.core.feature.Feature +import com.twitter.timelineservice.suggests.thriftscala.SuggestType + +object DismissFatigueGate { + // how long a dismiss action from user needs to be respected + val DefaultBaseDismissDuration = 7.days + val MaximumDismissalCountMultiplier = 4 +} + +case class DismissFatigueGate( + suggestType: SuggestType, + dismissInfoFeature: Feature[PipelineQuery, Map[SuggestType, Option[DismissInfo]]], + baseDismissDuration: Duration = DismissFatigueGate.DefaultBaseDismissDuration, +) extends Gate[PipelineQuery] { + + override val identifier: GateIdentifier = GateIdentifier("DismissFatigue") + + override def shouldContinue(query: PipelineQuery): Stitch[Boolean] = { + val dismissInfoMap = query.features.map( + _.getOrElse(dismissInfoFeature, Map.empty[SuggestType, Option[DismissInfo]])) + + val isVisible = dismissInfoMap + .flatMap(_.get(suggestType)) + .flatMap(_.map { info => + val currentDismissalDuration = query.queryTime.since(info.lastDismissed) + val targetDismissalDuration = dismissDurationForCount(info.count, baseDismissDuration) + + currentDismissalDuration > targetDismissalDuration + }).getOrElse(true) + Stitch.value(isVisible) + } + + private def dismissDurationForCount( + dismissCount: Int, + dismissDuration: Duration + ): Duration = + // limit to maximum dismissal duration + dismissDuration * Math.min(dismissCount, DismissFatigueGate.MaximumDismissalCountMultiplier) +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/gate/ExcludeSoftUserGate.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/gate/ExcludeSoftUserGate.scala new file mode 100644 index 0000000000..a861590362 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/gate/ExcludeSoftUserGate.scala @@ -0,0 +1,23 @@ +package com.twitter.home_mixer.functional_component.gate + +import com.twitter.gizmoduck.{thriftscala => t} +import com.twitter.home_mixer.model.HomeFeatures.UserTypeFeature +import com.twitter.product_mixer.core.functional_component.gate.Gate +import com.twitter.product_mixer.core.model.common.identifier.GateIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.stitch.Stitch + +/** + * A Soft User is a user who is in the gradual onboarding state. This gate can be + * used to turn off certain functionality like ads for these users. + */ +object ExcludeSoftUserGate extends Gate[PipelineQuery] { + + override val identifier: GateIdentifier = GateIdentifier("ExcludeSoftUser") + + override def shouldContinue(query: PipelineQuery): Stitch[Boolean] = { + val softUser = query.features + .exists(_.getOrElse(UserTypeFeature, None).exists(_ == t.UserType.Soft)) + Stitch.value(!softUser) + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/gate/MinCachedTweetsGate.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/gate/MinCachedTweetsGate.scala new file mode 100644 index 0000000000..a160a5b5d2 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/gate/MinCachedTweetsGate.scala @@ -0,0 +1,34 @@ +package com.twitter.home_mixer.functional_component.gate + +import com.twitter.home_mixer.functional_component.gate.MinCachedTweetsGate.identifierSuffix +import com.twitter.home_mixer.util.CachedScoredTweetsHelper +import com.twitter.product_mixer.core.functional_component.gate.Gate +import com.twitter.product_mixer.core.model.common.identifier.CandidatePipelineIdentifier +import com.twitter.product_mixer.core.model.common.identifier.GateIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.stitch.Stitch +import com.twitter.timelines.configapi.Param + +case class MinCachedTweetsGate( + candidatePipelineIdentifier: CandidatePipelineIdentifier, + minCachedTweetsParam: Param[Int]) + extends Gate[PipelineQuery] { + + override val identifier: GateIdentifier = + GateIdentifier(candidatePipelineIdentifier + identifierSuffix) + + override def shouldContinue(query: PipelineQuery): Stitch[Boolean] = { + val minCachedTweets = query.params(minCachedTweetsParam) + val cachedScoredTweets = + query.features.map(CachedScoredTweetsHelper.unseenCachedScoredTweets).getOrElse(Seq.empty) + val numCachedTweets = cachedScoredTweets.count { tweet => + tweet.candidatePipelineIdentifier.exists( + CandidatePipelineIdentifier(_).equals(candidatePipelineIdentifier)) + } + Stitch.value(numCachedTweets < minCachedTweets) + } +} + +object MinCachedTweetsGate { + val identifierSuffix = "MinCachedTweets" +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/gate/NonEmptySeqFeatureGate.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/gate/NonEmptySeqFeatureGate.scala new file mode 100644 index 0000000000..ef3bc5cff0 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/gate/NonEmptySeqFeatureGate.scala @@ -0,0 +1,18 @@ +package com.twitter.home_mixer.functional_component.gate + +import com.twitter.product_mixer.core.feature.Feature +import com.twitter.product_mixer.core.functional_component.gate.Gate +import com.twitter.product_mixer.core.model.common.identifier.GateIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.stitch.Stitch +import scala.reflect.runtime.universe._ + +case class NonEmptySeqFeatureGate[T: TypeTag]( + feature: Feature[PipelineQuery, Seq[T]]) + extends Gate[PipelineQuery] { + + override val identifier: GateIdentifier = GateIdentifier(s"NonEmptySeq$feature") + + override def shouldContinue(query: PipelineQuery): Stitch[Boolean] = + Stitch.value(query.features.exists(_.get(feature).nonEmpty)) +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/gate/RequestContextGate.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/gate/RequestContextGate.scala new file mode 100644 index 0000000000..0ffe6793e2 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/gate/RequestContextGate.scala @@ -0,0 +1,22 @@ +package com.twitter.home_mixer.functional_component.gate + +import com.twitter.home_mixer.model.request.DeviceContext.RequestContext +import com.twitter.home_mixer.model.request.HasDeviceContext +import com.twitter.product_mixer.core.functional_component.gate.Gate +import com.twitter.product_mixer.core.model.common.identifier.GateIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.stitch.Stitch + +/** + * Gate that fetches the request context from the device context and + * continues if the request context matches *any* of the specified ones. + */ +case class RequestContextGate(requestContexts: Seq[RequestContext.Value]) + extends Gate[PipelineQuery with HasDeviceContext] { + + override val identifier: GateIdentifier = GateIdentifier("RequestContext") + + override def shouldContinue(query: PipelineQuery with HasDeviceContext): Stitch[Boolean] = + Stitch.value( + requestContexts.exists(query.deviceContext.flatMap(_.requestContextValue).contains)) +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/gate/RequestContextNotGate.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/gate/RequestContextNotGate.scala new file mode 100644 index 0000000000..c52f05f753 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/gate/RequestContextNotGate.scala @@ -0,0 +1,24 @@ +package com.twitter.home_mixer.functional_component.gate + +import com.twitter.home_mixer.model.request.DeviceContext.RequestContext +import com.twitter.home_mixer.model.request.HasDeviceContext +import com.twitter.product_mixer.core.functional_component.gate.Gate +import com.twitter.product_mixer.core.model.common.identifier.GateIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.stitch.Stitch + +/** + * Gate that fetches the request context from the device context and + * continues if the request context does not match any of the specified ones. + * + * If no input request context is specified, the gate continues + */ +case class RequestContextNotGate(requestContexts: Seq[RequestContext.Value]) + extends Gate[PipelineQuery with HasDeviceContext] { + + override val identifier: GateIdentifier = GateIdentifier("RequestContextNot") + + override def shouldContinue(query: PipelineQuery with HasDeviceContext): Stitch[Boolean] = + Stitch.value( + !requestContexts.exists(query.deviceContext.flatMap(_.requestContextValue).contains)) +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/gate/SupportedLanguagesGate.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/gate/SupportedLanguagesGate.scala new file mode 100644 index 0000000000..53b681236e --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/gate/SupportedLanguagesGate.scala @@ -0,0 +1,68 @@ +package com.twitter.home_mixer.functional_component.gate + +import com.twitter.product_mixer.core.functional_component.gate.Gate +import com.twitter.product_mixer.core.model.common.identifier.GateIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.stitch.Stitch + +object SupportedLanguagesGate extends Gate[PipelineQuery] { + + override val identifier: GateIdentifier = GateIdentifier("SupportedLanguages") + + // Production languages which have high translation coverage for strings used in Home Timeline. + private val supportedLanguages: Set[String] = Set( + "ar", // Arabic + "ar-x-fm", // Arabic (Female) + "bg", // Bulgarian + "bn", // Bengali + "ca", // Catalan + "cs", // Czech + "da", // Danish + "de", // German + "el", // Greek + "en", // English + "en-gb", // British English + "en-ss", // English Screen shot + "en-xx", // English Pseudo + "es", // Spanish + "eu", // Basque + "fa", // Farsi (Persian) + "fi", // Finnish + "fil", // Filipino + "fr", // French + "ga", // Irish + "gl", // Galician + "gu", // Gujarati + "he", // Hebrew + "hi", // Hindi + "hr", // Croatian + "hu", // Hungarian + "id", // Indonesian + "it", // Italian + "ja", // Japanese + "kn", // Kannada + "ko", // Korean + "mr", // Marathi + "msa", // Malay + "nl", // Dutch + "no", // Norwegian + "pl", // Polish + "pt", // Portuguese + "ro", // Romanian + "ru", // Russian + "sk", // Slovak + "sr", // Serbian + "sv", // Swedish + "ta", // Tamil + "th", // Thai + "tr", // Turkish + "uk", // Ukrainian + "ur", // Urdu + "vi", // Vietnamese + "zh-cn", // Simplified Chinese + "zh-tw" // Traditional Chinese + ) + + override def shouldContinue(query: PipelineQuery): Stitch[Boolean] = + Stitch.value(query.getLanguageCode.forall(supportedLanguages.contains)) +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/gate/TimelinesPersistenceStoreLastInjectionGate.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/gate/TimelinesPersistenceStoreLastInjectionGate.scala new file mode 100644 index 0000000000..31fff23069 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/gate/TimelinesPersistenceStoreLastInjectionGate.scala @@ -0,0 +1,51 @@ +package com.twitter.home_mixer.functional_component.gate + +import com.twitter.common_internal.analytics.twitter_client_user_agent_parser.UserAgent +import com.twitter.product_mixer.core.feature.Feature +import com.twitter.product_mixer.core.functional_component.gate.Gate +import com.twitter.product_mixer.core.model.common.identifier.GateIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.stitch.Stitch +import com.twitter.timelinemixer.clients.persistence.TimelineResponseV3 +import com.twitter.timelinemixer.injection.store.persistence.TimelinePersistenceUtils +import com.twitter.timelines.configapi.Param +import com.twitter.timelines.util.client_info.ClientPlatform +import com.twitter.timelineservice.model.rich.EntityIdType +import com.twitter.util.Duration +import com.twitter.util.Time + +/** + * Gate used to reduce the frequency of injections. Note that the actual interval between injections may be + * less than the specified minInjectionIntervalParam if data is unavailable or missing. For example, being deleted by + * the persistence store via a TTL or similar mechanism. + * + * @param minInjectionIntervalParam the desired minimum interval between injections + * @param persistenceEntriesFeature the feature for retrieving persisted timeline responses + */ +case class TimelinesPersistenceStoreLastInjectionGate( + minInjectionIntervalParam: Param[Duration], + persistenceEntriesFeature: Feature[PipelineQuery, Seq[TimelineResponseV3]], + entityIdType: EntityIdType.Value) + extends Gate[PipelineQuery] + with TimelinePersistenceUtils { + + override val identifier: GateIdentifier = GateIdentifier("TimelinesPersistenceStoreLastInjection") + + override def shouldContinue(query: PipelineQuery): Stitch[Boolean] = + Stitch( + query.queryTime.since(getLastInjectionTime(query)) > query.params(minInjectionIntervalParam)) + + private def getLastInjectionTime(query: PipelineQuery) = query.features + .flatMap { featureMap => + val timelineResponses = featureMap.getOrElse(persistenceEntriesFeature, Seq.empty) + val clientPlatform = ClientPlatform.fromQueryOptions( + clientAppId = query.clientContext.appId, + userAgent = query.clientContext.userAgent.flatMap(UserAgent.fromString) + ) + val sortedResponses = responseByClient(clientPlatform, timelineResponses) + val latestResponseWithEntityIdTypeEntry = + sortedResponses.find(_.entries.exists(_.entityIdType == entityIdType)) + + latestResponseWithEntityIdTypeEntry.map(_.servedTime) + }.getOrElse(Time.Bottom) +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/gate/ViewerIsListOwnerGate.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/gate/ViewerIsListOwnerGate.scala new file mode 100644 index 0000000000..487487e923 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/gate/ViewerIsListOwnerGate.scala @@ -0,0 +1,29 @@ +package com.twitter.home_mixer.functional_component.gate + +import com.twitter.home_mixer.model.request.HasListId +import com.twitter.product_mixer.core.functional_component.gate.Gate +import com.twitter.product_mixer.core.model.common.identifier.GateIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.socialgraph.{thriftscala => sg} +import com.twitter.stitch.Stitch +import com.twitter.stitch.socialgraph.SocialGraph + +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +case class ViewerIsListOwnerGate @Inject() (socialGraph: SocialGraph) + extends Gate[PipelineQuery with HasListId] { + + override val identifier: GateIdentifier = GateIdentifier("ViewerIsListOwner") + + private val relationship = sg.Relationship(relationshipType = sg.RelationshipType.ListOwning) + + override def shouldContinue(query: PipelineQuery with HasListId): Stitch[Boolean] = { + val request = sg.ExistsRequest( + source = query.getRequiredUserId, + target = query.listId, + relationships = Seq(relationship)) + socialGraph.exists(request).map(_.exists) + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/query_transformer/BUILD.bazel b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/query_transformer/BUILD.bazel new file mode 100644 index 0000000000..67763235be --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/query_transformer/BUILD.bazel @@ -0,0 +1,13 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + strict_deps = True, + tags = ["bazel-compatible"], + dependencies = [ + "common-internal/analytics/twitter-client-user-agent-parser/src/main/scala", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/model", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/transformer", + "timelinemixer/server/src/main/scala/com/twitter/timelinemixer/injection/store/persistence", + "timelineservice/common:model", + ], +) diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/query_transformer/EditedTweetsCandidatePipelineQueryTransformer.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/query_transformer/EditedTweetsCandidatePipelineQueryTransformer.scala new file mode 100644 index 0000000000..731f7166e7 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/query_transformer/EditedTweetsCandidatePipelineQueryTransformer.scala @@ -0,0 +1,85 @@ +package com.twitter.home_mixer.functional_component.query_transformer + +import com.twitter.common_internal.analytics.twitter_client_user_agent_parser.UserAgent +import com.twitter.conversions.DurationOps._ +import com.twitter.home_mixer.model.HomeFeatures.PersistenceEntriesFeature +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.functional_component.transformer.CandidatePipelineQueryTransformer +import com.twitter.product_mixer.core.model.common.identifier.TransformerIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.timelinemixer.clients.persistence.EntryWithItemIds +import com.twitter.timelines.persistence.thriftscala.RequestType +import com.twitter.timelines.util.client_info.ClientPlatform +import com.twitter.timelineservice.model.rich.EntityIdType +import com.twitter.util.Time + +object EditedTweetsCandidatePipelineQueryTransformer + extends CandidatePipelineQueryTransformer[PipelineQuery, Seq[Long]] { + + override val identifier: TransformerIdentifier = TransformerIdentifier("EditedTweets") + + // The time window for which a tweet remains editable after creation. + private val EditTimeWindow = 30.minutes + + override def transform(query: PipelineQuery): Seq[Long] = { + val applicableCandidates = getApplicableCandidates(query) + + if (applicableCandidates.nonEmpty) { + // Include the response corresponding with the Previous Timeline Load (PTL). + // Any tweets in it could have become stale since being served. + val previousTimelineLoadTime = applicableCandidates.head.servedTime + + // The time window for editing a tweet is 30 minutes, + // so we ignore responses older than (PTL Time - 30 mins). + val inWindowCandidates: Seq[PersistenceStoreEntry] = applicableCandidates + .takeWhile(_.servedTime.until(previousTimelineLoadTime) < EditTimeWindow) + + // Exclude the tweet IDs for which ReplaceEntry instructions have already been sent. + val (tweetsAlreadyReplaced, tweetsToCheck) = inWindowCandidates + .partition(_.entryWithItemIds.itemIds.exists(_.head.entryIdToReplace.nonEmpty)) + + val tweetIdFromEntry: PartialFunction[PersistenceStoreEntry, Long] = { + case entry if entry.tweetId.nonEmpty => entry.tweetId.get + } + + val tweetIdsAlreadyReplaced: Set[Long] = tweetsAlreadyReplaced.collect(tweetIdFromEntry).toSet + val tweetIdsToCheck: Seq[Long] = tweetsToCheck.collect(tweetIdFromEntry) + + tweetIdsToCheck.filterNot(tweetIdsAlreadyReplaced.contains).distinct + } else Seq.empty + } + + // The candidates here come from the Timelines Persistence Store, via a query feature + private def getApplicableCandidates(query: PipelineQuery): Seq[PersistenceStoreEntry] = { + val userAgent = UserAgent.fromString(query.clientContext.userAgent.getOrElse("")) + val clientPlatform = ClientPlatform.fromQueryOptions(query.clientContext.appId, userAgent) + + val sortedResponses = query.features + .getOrElse(FeatureMap.empty) + .getOrElse(PersistenceEntriesFeature, Seq.empty) + .filter(_.clientPlatform == clientPlatform) + .sortBy(-_.servedTime.inMilliseconds) + + val recentResponses = sortedResponses.indexWhere(_.requestType == RequestType.Initial) match { + case -1 => sortedResponses + case lastGetInitialIndex => sortedResponses.take(lastGetInitialIndex + 1) + } + + recentResponses.flatMap { r => + r.entries.collect { + case entry if entry.entityIdType == EntityIdType.Tweet => + PersistenceStoreEntry(entry, r.servedTime, r.clientPlatform, r.requestType) + } + }.distinct + } +} + +case class PersistenceStoreEntry( + entryWithItemIds: EntryWithItemIds, + servedTime: Time, + clientPlatform: ClientPlatform, + requestType: RequestType) { + + // Timelines Persistence Store currently includes 1 tweet ID per entryWithItemIds for tweets + val tweetId: Option[Long] = entryWithItemIds.itemIds.flatMap(_.head.tweetId) +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/scorer/BUILD.bazel b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/scorer/BUILD.bazel new file mode 100644 index 0000000000..94a648e3aa --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/scorer/BUILD.bazel @@ -0,0 +1,17 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + strict_deps = True, + dependencies = [ + "home-mixer/server/src/main/scala/com/twitter/home_mixer/model", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/model/request", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/param", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/product/for_you/model", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/util", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/product", + "src/thrift/com/twitter/timelines/common:thrift-scala", + "src/thrift/com/twitter/timelineservice/server/internal:thrift-scala", + "timelineservice/common:model", + ], +) diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/scorer/FeedbackFatigueScorer.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/scorer/FeedbackFatigueScorer.scala new file mode 100644 index 0000000000..98e7aeb644 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/scorer/FeedbackFatigueScorer.scala @@ -0,0 +1,130 @@ +package com.twitter.home_mixer.functional_component.scorer + +import com.twitter.conversions.DurationOps._ +import com.twitter.home_mixer.model.HomeFeatures.AuthorIdFeature +import com.twitter.home_mixer.model.HomeFeatures.FeedbackHistoryFeature +import com.twitter.home_mixer.model.HomeFeatures.IsRetweetFeature +import com.twitter.home_mixer.model.HomeFeatures.SGSValidFollowedByUserIdsFeature +import com.twitter.home_mixer.model.HomeFeatures.SGSValidLikedByUserIdsFeature +import com.twitter.home_mixer.model.HomeFeatures.ScoreFeature +import com.twitter.home_mixer.util.CandidatesUtil +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.feature.Feature +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.feature.featuremap.FeatureMapBuilder +import com.twitter.product_mixer.core.functional_component.scorer.Scorer +import com.twitter.product_mixer.core.model.common.CandidateWithFeatures +import com.twitter.product_mixer.core.model.common.Conditionally +import com.twitter.product_mixer.core.model.common.identifier.ScorerIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.stitch.Stitch +import com.twitter.timelines.common.{thriftscala => tl} +import com.twitter.timelineservice.model.FeedbackEntry +import com.twitter.timelineservice.{thriftscala => tls} +import com.twitter.util.Time +import scala.collection.mutable + +object FeedbackFatigueScorer + extends Scorer[PipelineQuery, TweetCandidate] + with Conditionally[PipelineQuery] { + + override val identifier: ScorerIdentifier = ScorerIdentifier("FeedbackFatigue") + + override def features: Set[Feature[_, _]] = Set(ScoreFeature) + + override def onlyIf(query: PipelineQuery): Boolean = + query.features.exists(_.getOrElse(FeedbackHistoryFeature, Seq.empty).nonEmpty) + + private val DurationForFiltering = 14.days + private val DurationForDiscounting = 140.days + private val ScoreMultiplierLowerBound = 0.2 + private val ScoreMultiplierUpperBound = 1.0 + private val ScoreMultiplierIncrementsCount = 4 + private val ScoreMultiplierIncrement = + (ScoreMultiplierUpperBound - ScoreMultiplierLowerBound) / ScoreMultiplierIncrementsCount + private val ScoreMultiplierIncrementDurationInDays = + DurationForDiscounting.inDays / ScoreMultiplierIncrementsCount.toDouble + + override def apply( + query: PipelineQuery, + candidates: Seq[CandidateWithFeatures[TweetCandidate]] + ): Stitch[Seq[FeatureMap]] = { + val feedbackEntriesByEngagementType = + query.features + .getOrElse(FeatureMap.empty).getOrElse(FeedbackHistoryFeature, Seq.empty) + .filter { entry => + val timeSinceFeedback = query.queryTime.minus(entry.timestamp) + timeSinceFeedback < DurationForFiltering + DurationForDiscounting && + entry.feedbackType == tls.FeedbackType.SeeFewer + }.groupBy(_.engagementType) + + val authorsToDiscount = + getUserDiscounts( + query.queryTime, + feedbackEntriesByEngagementType.getOrElse(tls.FeedbackEngagementType.Tweet, Seq.empty)) + val likersToDiscount = + getUserDiscounts( + query.queryTime, + feedbackEntriesByEngagementType.getOrElse(tls.FeedbackEngagementType.Like, Seq.empty)) + val followersToDiscount = + getUserDiscounts( + query.queryTime, + feedbackEntriesByEngagementType.getOrElse(tls.FeedbackEngagementType.Follow, Seq.empty)) + val retweetersToDiscount = + getUserDiscounts( + query.queryTime, + feedbackEntriesByEngagementType.getOrElse(tls.FeedbackEngagementType.Retweet, Seq.empty)) + + val featureMaps = candidates.map { candidate => + val score = candidate.features.getOrElse(ScoreFeature, None) + + val originalAuthorId = + CandidatesUtil.getOriginalAuthorId(candidate.features).getOrElse(0L) + val originalAuthorMultiplier = authorsToDiscount.getOrElse(originalAuthorId, 1.0) + + val likers = candidate.features.getOrElse(SGSValidLikedByUserIdsFeature, Seq.empty) + val likerMultipliers = likers.flatMap(likersToDiscount.get) + val likerMultiplier = + if (likerMultipliers.nonEmpty && likers.size == likerMultipliers.size) + likerMultipliers.max + else 1.0 + + val followers = candidate.features.getOrElse(SGSValidFollowedByUserIdsFeature, Seq.empty) + val followerMultipliers = followers.flatMap(followersToDiscount.get) + val followerMultiplier = + if (followerMultipliers.nonEmpty && followers.size == followerMultipliers.size) + followerMultipliers.max + else 1.0 + + val authorId = candidate.features.getOrElse(AuthorIdFeature, None).getOrElse(0L) + val retweeterMultiplier = + if (candidate.features.getOrElse(IsRetweetFeature, false)) + retweetersToDiscount.getOrElse(authorId, 1.0) + else 1.0 + + val multiplier = + originalAuthorMultiplier * likerMultiplier * followerMultiplier * retweeterMultiplier + + FeatureMapBuilder().add(ScoreFeature, score.map(_ * multiplier)).build() + } + + Stitch.value(featureMaps) + } + + private def getUserDiscounts( + queryTime: Time, + feedbackEntries: Seq[FeedbackEntry], + ): Map[Long, Double] = { + val userDiscounts = mutable.Map[Long, Double]() + feedbackEntries + .collect { + case FeedbackEntry(_, _, tl.FeedbackEntity.UserId(userId), timestamp, _) => + val timeSinceFeedback = queryTime.minus(timestamp) + val timeSinceDiscounting = timeSinceFeedback - DurationForFiltering + val multiplier = ((timeSinceDiscounting.inDays / ScoreMultiplierIncrementDurationInDays) + * ScoreMultiplierIncrement + ScoreMultiplierLowerBound) + userDiscounts.update(userId, multiplier) + } + userDiscounts.toMap + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/scorer/OONTweetScalingScorer.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/scorer/OONTweetScalingScorer.scala new file mode 100644 index 0000000000..4b1cec4a52 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/scorer/OONTweetScalingScorer.scala @@ -0,0 +1,48 @@ +package com.twitter.home_mixer.functional_component.scorer + +import com.twitter.home_mixer.model.HomeFeatures.InNetworkFeature +import com.twitter.home_mixer.model.HomeFeatures.IsRetweetFeature +import com.twitter.home_mixer.model.HomeFeatures.ScoreFeature +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.feature.Feature +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.feature.featuremap.FeatureMapBuilder +import com.twitter.product_mixer.core.functional_component.scorer.Scorer +import com.twitter.product_mixer.core.model.common.CandidateWithFeatures +import com.twitter.product_mixer.core.model.common.identifier.ScorerIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.stitch.Stitch + +/** + * Scales scores of each out-of-network tweet by the specified scale factor + */ +object OONTweetScalingScorer extends Scorer[PipelineQuery, TweetCandidate] { + + override val identifier: ScorerIdentifier = ScorerIdentifier("OONTweetScaling") + + override val features: Set[Feature[_, _]] = Set(ScoreFeature) + + private val ScaleFactor = 0.75 + + override def apply( + query: PipelineQuery, + candidates: Seq[CandidateWithFeatures[TweetCandidate]] + ): Stitch[Seq[FeatureMap]] = { + Stitch.value { + candidates.map { candidate => + val score = candidate.features.getOrElse(ScoreFeature, None) + val updatedScore = if (selector(candidate)) score.map(_ * ScaleFactor) else score + FeatureMapBuilder().add(ScoreFeature, updatedScore).build() + } + } + } + + /** + * We should only be applying this multiplier to Out-Of-Network tweets. + * In-Network Retweets of Out-Of-Network tweets should not have this multiplier applied + */ + private def selector(candidate: CandidateWithFeatures[TweetCandidate]): Boolean = { + !candidate.features.getOrElse(InNetworkFeature, false) && + !candidate.features.getOrElse(IsRetweetFeature, false) + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/scorer/VerifiedAuthorScalingScorer.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/scorer/VerifiedAuthorScalingScorer.scala new file mode 100644 index 0000000000..3644b7d36d --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/scorer/VerifiedAuthorScalingScorer.scala @@ -0,0 +1,61 @@ +package com.twitter.home_mixer.functional_component.scorer + +import com.twitter.home_mixer.model.HomeFeatures.AuthorIsBlueVerifiedFeature +import com.twitter.home_mixer.model.HomeFeatures.InNetworkFeature +import com.twitter.home_mixer.model.HomeFeatures.ScoreFeature +import com.twitter.home_mixer.param.HomeGlobalParams.BlueVerifiedAuthorInNetworkMultiplierParam +import com.twitter.home_mixer.param.HomeGlobalParams.BlueVerifiedAuthorOutOfNetworkMultiplierParam +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.feature.Feature +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.feature.featuremap.FeatureMapBuilder +import com.twitter.product_mixer.core.functional_component.scorer.Scorer +import com.twitter.product_mixer.core.model.common.CandidateWithFeatures +import com.twitter.product_mixer.core.model.common.identifier.ScorerIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.stitch.Stitch + +/** + * Scales scores of tweets whose author is Blue Verified by the provided scale factor + */ +object VerifiedAuthorScalingScorer extends Scorer[PipelineQuery, TweetCandidate] { + + override val identifier: ScorerIdentifier = ScorerIdentifier("VerifiedAuthorScaling") + + override val features: Set[Feature[_, _]] = Set(ScoreFeature) + + override def apply( + query: PipelineQuery, + candidates: Seq[CandidateWithFeatures[TweetCandidate]] + ): Stitch[Seq[FeatureMap]] = { + Stitch.value { + candidates.map { candidate => + val score = candidate.features.getOrElse(ScoreFeature, None) + val updatedScore = getUpdatedScore(score, candidate, query) + FeatureMapBuilder().add(ScoreFeature, updatedScore).build() + } + } + } + + /** + * We should only be applying this multiplier if the author of the candidate is Blue Verified. + * We also treat In-Network vs Out-of-Network differently. + */ + private def getUpdatedScore( + score: Option[Double], + candidate: CandidateWithFeatures[TweetCandidate], + query: PipelineQuery + ): Option[Double] = { + val isAuthorBlueVerified = candidate.features.getOrElse(AuthorIsBlueVerifiedFeature, false) + + if (isAuthorBlueVerified) { + val isCandidateInNetwork = candidate.features.getOrElse(InNetworkFeature, false) + + val scaleFactor = + if (isCandidateInNetwork) query.params(BlueVerifiedAuthorInNetworkMultiplierParam) + else query.params(BlueVerifiedAuthorOutOfNetworkMultiplierParam) + + score.map(_ * scaleFactor) + } else score + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/selector/BUILD.bazel b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/selector/BUILD.bazel new file mode 100644 index 0000000000..0f8ac903b8 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/selector/BUILD.bazel @@ -0,0 +1,24 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + strict_deps = True, + tags = ["bazel-compatible"], + dependencies = [ + "home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/model", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/model/request", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/param", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/util", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/candidate", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/presentation/urt", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/decorator/urt/builder", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/selector", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/common/identifier", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/common/presentation", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/item", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline", + "src/scala/com/twitter/suggests/controller_data", + "stringcenter/client", + "stringcenter/client/src/main/java", + ], +) diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/selector/DebunchCandidates.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/selector/DebunchCandidates.scala new file mode 100644 index 0000000000..6481742731 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/selector/DebunchCandidates.scala @@ -0,0 +1,83 @@ +package com.twitter.home_mixer.functional_component.selector + +import com.twitter.home_mixer.functional_component.selector.DebunchCandidates.TrailingTweetsMinSize +import com.twitter.home_mixer.functional_component.selector.DebunchCandidates.TrailingTweetsPortionToKeep +import com.twitter.home_mixer.model.HomeFeatures.GetNewerFeature +import com.twitter.product_mixer.core.functional_component.common.CandidateScope +import com.twitter.product_mixer.core.functional_component.common.CandidateScope.PartitionedCandidates +import com.twitter.product_mixer.core.functional_component.selector.Selector +import com.twitter.product_mixer.core.functional_component.selector.SelectorResult +import com.twitter.product_mixer.core.model.common.presentation.CandidateWithDetails +import com.twitter.product_mixer.core.pipeline.PipelineQuery + +trait MustDebunch { + def apply(candidate: CandidateWithDetails): Boolean +} + +object DebunchCandidates { + val TrailingTweetsMinSize = 5 + val TrailingTweetsPortionToKeep = 0.1 +} + +/** + * This selector rearranges the candidates to only allow bunches of size [[maxBunchSize]], where a + * bunch is a consecutive sequence of candidates that meet [[mustDebunch]]. + */ +case class DebunchCandidates( + override val pipelineScope: CandidateScope, + mustDebunch: MustDebunch, + maxBunchSize: Int) + extends Selector[PipelineQuery] { + + override def apply( + query: PipelineQuery, + remainingCandidates: Seq[CandidateWithDetails], + result: Seq[CandidateWithDetails] + ): SelectorResult = { + val PartitionedCandidates(selectedCandidates, otherCandidates) = + pipelineScope.partition(remainingCandidates) + val mutableCandidates = collection.mutable.ListBuffer(selectedCandidates: _*) + + var candidatePointer = 0 + var nonDebunchPointer = 0 + var bunchSize = 0 + var finalNonDebunch = -1 + + while (candidatePointer < mutableCandidates.size) { + if (mustDebunch(mutableCandidates(candidatePointer))) bunchSize += 1 + else { + bunchSize = 0 + finalNonDebunch = candidatePointer + } + + if (bunchSize > maxBunchSize) { + nonDebunchPointer = Math.max(candidatePointer, nonDebunchPointer) + while (nonDebunchPointer < mutableCandidates.size && + mustDebunch(mutableCandidates(nonDebunchPointer))) { + nonDebunchPointer += 1 + } + if (nonDebunchPointer == mutableCandidates.size) + candidatePointer = mutableCandidates.size + else { + val nextNonDebunch = mutableCandidates(nonDebunchPointer) + mutableCandidates.remove(nonDebunchPointer) + mutableCandidates.insert(candidatePointer, nextNonDebunch) + bunchSize = 0 + finalNonDebunch = candidatePointer + } + } + + candidatePointer += 1 + } + + val debunchedCandidates = if (query.features.exists(_.getOrElse(GetNewerFeature, false))) { + val trailingTweetsSize = mutableCandidates.size - finalNonDebunch - 1 + val keepCandidates = finalNonDebunch + 1 + + Math.max(TrailingTweetsMinSize, TrailingTweetsPortionToKeep * trailingTweetsSize).toInt + mutableCandidates.toList.take(keepCandidates) + } else mutableCandidates.toList + + val updatedCandidates = otherCandidates ++ debunchedCandidates + SelectorResult(remainingCandidates = updatedCandidates, result = result) + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/selector/UpdateConversationModuleId.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/selector/UpdateConversationModuleId.scala new file mode 100644 index 0000000000..bbde0ed0d6 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/selector/UpdateConversationModuleId.scala @@ -0,0 +1,40 @@ +package com.twitter.home_mixer.functional_component.selector + +import com.twitter.product_mixer.component_library.model.presentation.urt.UrtModulePresentation +import com.twitter.product_mixer.core.functional_component.common.CandidateScope +import com.twitter.product_mixer.core.functional_component.common.CandidateScope.PartitionedCandidates +import com.twitter.product_mixer.core.functional_component.selector.Selector +import com.twitter.product_mixer.core.functional_component.selector.SelectorResult +import com.twitter.product_mixer.core.model.common.presentation.CandidateWithDetails +import com.twitter.product_mixer.core.model.common.presentation.ModuleCandidateWithDetails +import com.twitter.product_mixer.core.pipeline.PipelineQuery + +/** + * This selector updates the id of the conversation modules to be the head of the module's id. + */ +case class UpdateConversationModuleId( + override val pipelineScope: CandidateScope) + extends Selector[PipelineQuery] { + + override def apply( + query: PipelineQuery, + remainingCandidates: Seq[CandidateWithDetails], + result: Seq[CandidateWithDetails] + ): SelectorResult = { + val PartitionedCandidates(selectedCandidates, otherCandidates) = + pipelineScope.partition(remainingCandidates) + + val updatedCandidates = selectedCandidates.map { + case module @ ModuleCandidateWithDetails(candidates, presentationOpt, _) => + val updatedPresentation = presentationOpt.map { + case urtModule @ UrtModulePresentation(timelineModule) => + urtModule.copy(timelineModule = + timelineModule.copy(id = candidates.head.candidateIdLong)) + } + module.copy(presentation = updatedPresentation) + case candidate => candidate + } + + SelectorResult(remainingCandidates = updatedCandidates ++ otherCandidates, result = result) + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/selector/UpdateHomeClientEventDetails.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/selector/UpdateHomeClientEventDetails.scala new file mode 100644 index 0000000000..28968e64d4 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/selector/UpdateHomeClientEventDetails.scala @@ -0,0 +1,137 @@ +package com.twitter.home_mixer.functional_component.selector + +import com.twitter.home_mixer.functional_component.decorator.HomeClientEventDetailsBuilder +import com.twitter.home_mixer.model.HomeFeatures.AncestorsFeature +import com.twitter.home_mixer.model.HomeFeatures.ConversationModule2DisplayedTweetsFeature +import com.twitter.home_mixer.model.HomeFeatures.ConversationModuleHasGapFeature +import com.twitter.home_mixer.model.HomeFeatures.HasRandomTweetFeature +import com.twitter.home_mixer.model.HomeFeatures.IsRandomTweetAboveFeature +import com.twitter.home_mixer.model.HomeFeatures.IsRandomTweetFeature +import com.twitter.home_mixer.model.HomeFeatures.PositionFeature +import com.twitter.home_mixer.model.HomeFeatures.ServedInConversationModuleFeature +import com.twitter.home_mixer.model.HomeFeatures.ServedSizeFeature +import com.twitter.product_mixer.component_library.model.presentation.urt.UrtItemPresentation +import com.twitter.product_mixer.component_library.model.presentation.urt.UrtModulePresentation +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.feature.featuremap.FeatureMapBuilder +import com.twitter.product_mixer.core.functional_component.common.CandidateScope +import com.twitter.product_mixer.core.functional_component.common.SpecificPipelines +import com.twitter.product_mixer.core.functional_component.selector.Selector +import com.twitter.product_mixer.core.functional_component.selector.SelectorResult +import com.twitter.product_mixer.core.model.common.identifier.CandidatePipelineIdentifier +import com.twitter.product_mixer.core.model.common.presentation.CandidateWithDetails +import com.twitter.product_mixer.core.model.common.presentation.ItemCandidateWithDetails +import com.twitter.product_mixer.core.model.common.presentation.ModuleCandidateWithDetails +import com.twitter.product_mixer.core.model.marshalling.response.urt.item.tweet.TweetItem +import com.twitter.product_mixer.core.pipeline.PipelineQuery + +/** + * Builds serialized tweet type metrics controller data and updates Client Event Details + * and Candidate Presentations with this info. + * + * Currently only updates presentation of Item Candidates. This needs to be updated + * when modules are added. + * + * This is implemented as a Selector instead of a Decorator in the Candidate Pipeline + * because we need to add controller data that looks at the final timeline as a whole + * (e.g. served size, final candidate positions). + * + * @param candidatePipelines - only candidates from the specified pipeline will be updated + */ +case class UpdateHomeClientEventDetails(candidatePipelines: Set[CandidatePipelineIdentifier]) + extends Selector[PipelineQuery] { + + override val pipelineScope: CandidateScope = SpecificPipelines(candidatePipelines) + + private val detailsBuilder = HomeClientEventDetailsBuilder() + + override def apply( + query: PipelineQuery, + remainingCandidates: Seq[CandidateWithDetails], + result: Seq[CandidateWithDetails] + ): SelectorResult = { + val selectedCandidates = result.filter(pipelineScope.contains) + + val randomTweetsByPosition = result + .map(_.features.getOrElse(IsRandomTweetFeature, false)) + .zipWithIndex.map(_.swap).toMap + + val resultFeatures = FeatureMapBuilder() + .add(ServedSizeFeature, Some(selectedCandidates.size)) + .add(HasRandomTweetFeature, randomTweetsByPosition.valuesIterator.contains(true)) + .build() + + val updatedResult = result.zipWithIndex.map { + case (item @ ItemCandidateWithDetails(candidate, _, _), position) + if pipelineScope.contains(item) => + val resultCandidateFeatures = FeatureMapBuilder() + .add(PositionFeature, Some(position)) + .add(IsRandomTweetAboveFeature, randomTweetsByPosition.getOrElse(position - 1, false)) + .build() + + updateItemPresentation(query, item, resultFeatures, resultCandidateFeatures) + + case (module @ ModuleCandidateWithDetails(candidates, presentation, features), position) + if pipelineScope.contains(module) => + val resultCandidateFeatures = FeatureMapBuilder() + .add(PositionFeature, Some(position)) + .add(IsRandomTweetAboveFeature, randomTweetsByPosition.getOrElse(position - 1, false)) + .add(ServedInConversationModuleFeature, true) + .add(ConversationModule2DisplayedTweetsFeature, module.candidates.size == 2) + .add( + ConversationModuleHasGapFeature, + module.candidates.last.features.getOrElse(AncestorsFeature, Seq.empty).size > 2) + .build() + + val updatedItemCandidates = + candidates.map(updateItemPresentation(query, _, resultFeatures, resultCandidateFeatures)) + + val updatedCandidateFeatures = features ++ resultFeatures ++ resultCandidateFeatures + + val updatedPresentation = presentation.map { + case urtModule @ UrtModulePresentation(timelineModule) => + val clientEventDetails = + detailsBuilder( + query, + candidates.last.candidate, + query.features.get ++ updatedCandidateFeatures) + val updatedClientEventInfo = + timelineModule.clientEventInfo.map(_.copy(details = clientEventDetails)) + val updatedTimelineModule = + timelineModule.copy(clientEventInfo = updatedClientEventInfo) + urtModule.copy(timelineModule = updatedTimelineModule) + } + + module.copy( + candidates = updatedItemCandidates, + presentation = updatedPresentation, + features = updatedCandidateFeatures + ) + + case (any, position) => any + } + + SelectorResult(remainingCandidates = remainingCandidates, result = updatedResult) + } + + private def updateItemPresentation( + query: PipelineQuery, + item: ItemCandidateWithDetails, + resultCandidateFeatures: FeatureMap, + resultFeatures: FeatureMap, + ): ItemCandidateWithDetails = { + val updatedItemCandidateFeatures = item.features ++ resultFeatures ++ resultCandidateFeatures + + val updatedPresentation = item.presentation.map { + case urtItem @ UrtItemPresentation(timelineItem: TweetItem, _) => + val clientEventDetails = + detailsBuilder(query, item.candidate, query.features.get ++ updatedItemCandidateFeatures) + val updatedClientEventInfo = + timelineItem.clientEventInfo.map(_.copy(details = clientEventDetails)) + val updatedTimelineItem = timelineItem.copy(clientEventInfo = updatedClientEventInfo) + urtItem.copy(timelineItem = updatedTimelineItem) + case any => any + } + item.copy(presentation = updatedPresentation, features = updatedItemCandidateFeatures) + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/selector/UpdateNewTweetsPillDecoration.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/selector/UpdateNewTweetsPillDecoration.scala new file mode 100644 index 0000000000..a805ae1a7c --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/selector/UpdateNewTweetsPillDecoration.scala @@ -0,0 +1,80 @@ +package com.twitter.home_mixer.functional_component.selector + +import com.twitter.home_mixer.functional_component.selector.UpdateNewTweetsPillDecoration.NumAvatars +import com.twitter.home_mixer.model.HomeFeatures.AuthorIdFeature +import com.twitter.home_mixer.model.HomeFeatures.IsRetweetFeature +import com.twitter.home_mixer.model.request.HasDeviceContext +import com.twitter.home_mixer.param.HomeGlobalParams.EnableNewTweetsPillAvatarsParam +import com.twitter.home_mixer.util.CandidatesUtil +import com.twitter.product_mixer.component_library.model.candidate.ShowAlertCandidate +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.component_library.model.presentation.urt.UrtItemPresentation +import com.twitter.product_mixer.core.functional_component.common.CandidateScope +import com.twitter.product_mixer.core.functional_component.selector.Selector +import com.twitter.product_mixer.core.functional_component.selector.SelectorResult +import com.twitter.product_mixer.core.model.common.presentation.CandidateWithDetails +import com.twitter.product_mixer.core.model.common.presentation.ItemCandidateWithDetails +import com.twitter.product_mixer.core.model.marshalling.response.urt.ShowAlert +import com.twitter.product_mixer.core.model.marshalling.response.urt.richtext.RichText +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.stringcenter.client.StringCenter +import com.twitter.stringcenter.client.core.ExternalString + +object UpdateNewTweetsPillDecoration { + val NumAvatars = 3 +} + +case class UpdateNewTweetsPillDecoration[Query <: PipelineQuery with HasDeviceContext]( + override val pipelineScope: CandidateScope, + stringCenter: StringCenter, + seeNewTweetsString: ExternalString, + tweetedString: ExternalString) + extends Selector[Query] { + + override def apply( + query: Query, + remainingCandidates: Seq[CandidateWithDetails], + result: Seq[CandidateWithDetails] + ): SelectorResult = { + val (alerts, otherCandidates) = + remainingCandidates.partition(candidate => + candidate.isCandidateType[ShowAlertCandidate]() && pipelineScope.contains(candidate)) + val updatedCandidates = alerts + .collectFirst { + case newTweetsPill: ItemCandidateWithDetails => + val userIds = CandidatesUtil + .getItemCandidatesWithOnlyModuleLast(result) + .filter(candidate => + candidate.isCandidateType[TweetCandidate]() && pipelineScope.contains(candidate)) + .filterNot(_.features.getOrElse(IsRetweetFeature, false)) + .flatMap(_.features.getOrElse(AuthorIdFeature, None)) + .filterNot(_ == query.getRequiredUserId) + .distinct + + val updatedPresentation = newTweetsPill.presentation.map { + case presentation: UrtItemPresentation => + presentation.timelineItem match { + case alert: ShowAlert => + val text = if (useAvatars(query, userIds)) tweetedString else seeNewTweetsString + val richText = RichText( + text = stringCenter.prepare(text), + entities = List.empty, + rtl = None, + alignment = None) + + val updatedAlert = + alert.copy(userIds = Some(userIds.take(NumAvatars)), richText = Some(richText)) + presentation.copy(timelineItem = updatedAlert) + } + } + otherCandidates :+ newTweetsPill.copy(presentation = updatedPresentation) + }.getOrElse(remainingCandidates) + + SelectorResult(remainingCandidates = updatedCandidates, result = result) + } + + private def useAvatars(query: Query, userIds: Seq[Long]): Boolean = { + val enableAvatars = query.params(EnableNewTweetsPillAvatarsParam) + enableAvatars && userIds.size >= NumAvatars + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/side_effect/BUILD.bazel b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/side_effect/BUILD.bazel new file mode 100644 index 0000000000..c9c0b343cf --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/side_effect/BUILD.bazel @@ -0,0 +1,49 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + strict_deps = True, + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/javax/inject:javax.inject", + "eventbus/client/src/main/scala/com/twitter/eventbus/client", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/marshaller/timeline_logging", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/marshaller/timelines", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/model", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/model/request", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/param", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/product/following/model", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/product/following/param", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/model", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/service", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/util", + "kafka/finagle-kafka/finatra-kafka/src/main/scala", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/candidate", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/presentation/urt", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/pipeline/candidate/who_to_follow_module", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/side_effect", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/product", + "src/scala/com/twitter/timelines/prediction/common/adapters", + "src/scala/com/twitter/timelines/prediction/features/common", + "src/thrift/com/twitter/timelines/impression:thrift-scala", + "src/thrift/com/twitter/timelines/impression_bloom_filter:thrift-scala", + "src/thrift/com/twitter/timelines/impression_store:thrift-scala", + "src/thrift/com/twitter/timelines/served_candidates_logging:served_candidates_logging-scala", + "src/thrift/com/twitter/timelines/suggests/common:poly_data_record-java", + "src/thrift/com/twitter/timelines/timeline_logging:thrift-scala", + "src/thrift/com/twitter/user_session_store:thrift-scala", + "timelinemixer/server/src/main/scala/com/twitter/timelinemixer/injection/core", + "timelinemixer/server/src/main/scala/com/twitter/timelinemixer/injection/store/persistence", + "timelinemixer/server/src/main/scala/com/twitter/timelinemixer/injection/store/uss", + "timelines/ml:kafka", + "timelines/ml/cont_train/common/client/src/main/scala/com/twitter/timelines/ml/cont_train/common/client/kafka", + "timelines/ml/cont_train/common/domain/src/main/scala/com/twitter/timelines/ml/cont_train/common/domain/non_scalding", + "timelines/src/main/scala/com/twitter/timelines/clientconfig", + "timelines/src/main/scala/com/twitter/timelines/impressionstore/impressionbloomfilter", + "timelines/src/main/scala/com/twitter/timelines/impressionstore/store", + "timelines/src/main/scala/com/twitter/timelines/injection/scribe", + "timelineservice/common:model", + "user_session_store/src/main/scala/com/twitter/user_session_store", + ], +) diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/side_effect/ClientEventsBuilder.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/side_effect/ClientEventsBuilder.scala new file mode 100644 index 0000000000..a0233e64eb --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/side_effect/ClientEventsBuilder.scala @@ -0,0 +1,177 @@ +package com.twitter.home_mixer.functional_component.side_effect + +import com.twitter.conversions.DurationOps._ +import com.twitter.home_mixer.functional_component.decorator.HomeQueryTypePredicates +import com.twitter.home_mixer.functional_component.decorator.HomeTweetTypePredicates +import com.twitter.home_mixer.model.HomeFeatures.AccountAgeFeature +import com.twitter.home_mixer.model.HomeFeatures.SuggestTypeFeature +import com.twitter.home_mixer.model.HomeFeatures.VideoDurationMsFeature +import com.twitter.home_mixer.model.request.FollowingProduct +import com.twitter.home_mixer.model.request.ForYouProduct +import com.twitter.home_mixer.model.request.ListTweetsProduct +import com.twitter.product_mixer.component_library.side_effect.ScribeClientEventSideEffect.ClientEvent +import com.twitter.product_mixer.component_library.side_effect.ScribeClientEventSideEffect.EventNamespace +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.model.common.presentation.CandidateWithDetails +import com.twitter.product_mixer.core.model.common.presentation.ItemCandidateWithDetails +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.timelines.injection.scribe.InjectionScribeUtil + +private[side_effect] sealed trait ClientEventsBuilder { + private val FollowingSection = Some("home_latest") + private val ForYouSection = Some("home") + private val ListTweetsSection = Some("list") + + protected def section(query: PipelineQuery): Option[String] = { + query.product match { + case FollowingProduct => FollowingSection + case ForYouProduct => ForYouSection + case ListTweetsProduct => ListTweetsSection + case other => throw new UnsupportedOperationException(s"Unknown product: $other") + } + } + + protected def count( + candidates: Seq[CandidateWithDetails], + predicate: FeatureMap => Boolean = _ => true, + queryFeatures: FeatureMap = FeatureMap.empty + ): Option[Long] = Some(candidates.view.count(item => predicate(item.features ++ queryFeatures))) + + protected def sum( + candidates: Seq[CandidateWithDetails], + predicate: FeatureMap => Option[Int], + queryFeatures: FeatureMap = FeatureMap.empty + ): Option[Long] = + Some(candidates.view.flatMap(item => predicate(item.features ++ queryFeatures)).sum) +} + +private[side_effect] object ServedEventsBuilder extends ClientEventsBuilder { + + private val ServedTweetsAction = Some("served_tweets") + private val ServedUsersAction = Some("served_users") + private val InjectedComponent = Some("injected") + private val PromotedComponent = Some("promoted") + private val WhoToFollowComponent = Some("who_to_follow") + private val WithVideoDurationComponent = Some("with_video_duration") + private val VideoDurationSumElement = Some("video_duration_sum") + private val NumVideosElement = Some("num_videos") + + def build( + query: PipelineQuery, + injectedTweets: Seq[ItemCandidateWithDetails], + promotedTweets: Seq[ItemCandidateWithDetails], + whoToFollowUsers: Seq[ItemCandidateWithDetails] + ): Seq[ClientEvent] = { + val baseEventNamespace = EventNamespace( + section = section(query), + action = ServedTweetsAction + ) + val overallServedEvents = Seq( + ClientEvent(baseEventNamespace, eventValue = count(injectedTweets ++ promotedTweets)), + ClientEvent( + baseEventNamespace.copy(component = InjectedComponent), + eventValue = count(injectedTweets)), + ClientEvent( + baseEventNamespace.copy(component = PromotedComponent), + eventValue = count(promotedTweets)), + ClientEvent( + baseEventNamespace.copy(component = WhoToFollowComponent, action = ServedUsersAction), + eventValue = count(whoToFollowUsers)), + ) + + val tweetTypeServedEvents = HomeTweetTypePredicates.PredicateMap.map { + case (tweetType, predicate) => + ClientEvent( + baseEventNamespace.copy(component = InjectedComponent, element = Some(tweetType)), + eventValue = count(injectedTweets, predicate, query.features.getOrElse(FeatureMap.empty)) + ) + }.toSeq + + val suggestTypeServedEvents = injectedTweets + .flatMap(_.features.getOrElse(SuggestTypeFeature, None)) + .map { + InjectionScribeUtil.scribeComponent + } + .groupBy(identity).map { + case (suggestType, group) => + ClientEvent( + baseEventNamespace.copy(component = suggestType), + eventValue = Some(group.size.toLong)) + }.toSeq + + // Video duration events + val numVideosEvent = ClientEvent( + baseEventNamespace.copy(component = WithVideoDurationComponent, element = NumVideosElement), + eventValue = count(injectedTweets, _.getOrElse(VideoDurationMsFeature, None).nonEmpty) + ) + val videoDurationSumEvent = ClientEvent( + baseEventNamespace + .copy(component = WithVideoDurationComponent, element = VideoDurationSumElement), + eventValue = sum(injectedTweets, _.getOrElse(VideoDurationMsFeature, None)) + ) + val videoEvents = Seq(numVideosEvent, videoDurationSumEvent) + + overallServedEvents ++ tweetTypeServedEvents ++ suggestTypeServedEvents ++ videoEvents + } +} + +private[side_effect] object EmptyTimelineEventsBuilder extends ClientEventsBuilder { + private val EmptyAction = Some("empty") + private val AccountAgeLessThan30MinutesComponent = Some("account_age_less_than_30_minutes") + private val ServedNonPromotedTweetElement = Some("served_non_promoted_tweet") + + def build( + query: PipelineQuery, + injectedTweets: Seq[ItemCandidateWithDetails] + ): Seq[ClientEvent] = { + val baseEventNamespace = EventNamespace( + section = section(query), + action = EmptyAction + ) + + // Empty timeline events + val accountAgeLessThan30Minutes = query.features + .flatMap(_.getOrElse(AccountAgeFeature, None)) + .exists(_.untilNow < 30.minutes) + val isEmptyTimeline = count(injectedTweets).contains(0L) + val predicates = Seq( + None -> isEmptyTimeline, + AccountAgeLessThan30MinutesComponent -> (isEmptyTimeline && accountAgeLessThan30Minutes) + ) + for { + (component, predicate) <- predicates + if predicate + } yield ClientEvent( + baseEventNamespace.copy(component = component, element = ServedNonPromotedTweetElement)) + } +} + +private[side_effect] object QueryEventsBuilder extends ClientEventsBuilder { + + private val ServedSizePredicateMap: Map[String, Int => Boolean] = Map( + ("size_is_empty", _ <= 0), + ("size_at_most_5", _ <= 5), + ("size_at_most_10", _ <= 10), + ("size_at_most_35", _ <= 35) + ) + + def build( + query: PipelineQuery, + injectedTweets: Seq[ItemCandidateWithDetails] + ): Seq[ClientEvent] = { + val baseEventNamespace = EventNamespace( + section = section(query) + ) + val queryFeatureMap = query.features.getOrElse(FeatureMap.empty) + val servedSizeQueryEvents = + for { + (queryPredicateName, queryPredicate) <- HomeQueryTypePredicates.PredicateMap + if queryPredicate(queryFeatureMap) + (servedSizePredicateName, servedSizePredicate) <- ServedSizePredicateMap + if servedSizePredicate(injectedTweets.size) + } yield ClientEvent( + baseEventNamespace + .copy(component = Some(servedSizePredicateName), action = Some(queryPredicateName))) + servedSizeQueryEvents.toSeq + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/side_effect/HomeScribeClientEventSideEffect.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/side_effect/HomeScribeClientEventSideEffect.scala new file mode 100644 index 0000000000..83f434addb --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/side_effect/HomeScribeClientEventSideEffect.scala @@ -0,0 +1,58 @@ +package com.twitter.home_mixer.functional_component.side_effect + +import com.twitter.clientapp.thriftscala.LogEvent +import com.twitter.home_mixer.service.HomeMixerAlertConfig +import com.twitter.home_mixer.util.CandidatesUtil +import com.twitter.logpipeline.client.common.EventPublisher +import com.twitter.product_mixer.component_library.side_effect.ScribeClientEventSideEffect +import com.twitter.product_mixer.core.model.common.identifier.CandidatePipelineIdentifier +import com.twitter.product_mixer.core.model.common.identifier.SideEffectIdentifier +import com.twitter.product_mixer.core.model.common.presentation.CandidateWithDetails +import com.twitter.product_mixer.core.model.marshalling.response.urt.Timeline +import com.twitter.product_mixer.core.pipeline.PipelineQuery + +/** + * Side effect that logs served tweet metrics to Scribe as client events. + */ +case class HomeScribeClientEventSideEffect( + override val logPipelinePublisher: EventPublisher[LogEvent], + injectedTweetsCandidatePipelineIdentifiers: Seq[CandidatePipelineIdentifier], + adsCandidatePipelineIdentifier: CandidatePipelineIdentifier, + whoToFollowCandidatePipelineIdentifier: Option[CandidatePipelineIdentifier] = None, +) extends ScribeClientEventSideEffect[PipelineQuery, Timeline] { + + override val identifier: SideEffectIdentifier = SideEffectIdentifier("HomeScribeClientEvent") + + override val page = "timelinemixer" + + override def buildClientEvents( + query: PipelineQuery, + selectedCandidates: Seq[CandidateWithDetails], + remainingCandidates: Seq[CandidateWithDetails], + droppedCandidates: Seq[CandidateWithDetails], + response: Timeline + ): Seq[ScribeClientEventSideEffect.ClientEvent] = { + + val itemCandidates = CandidatesUtil.getItemCandidates(selectedCandidates) + val sources = itemCandidates.groupBy(_.source) + val injectedTweets = + injectedTweetsCandidatePipelineIdentifiers.flatMap(sources.getOrElse(_, Seq.empty)) + val promotedTweets = sources.getOrElse(adsCandidatePipelineIdentifier, Seq.empty) + + // WhoToFollow module is not required for all home-mixer products, e.g. list tweets timeline. + val whoToFollowUsers = whoToFollowCandidatePipelineIdentifier.flatMap(sources.get).toSeq.flatten + + val servedEvents = ServedEventsBuilder + .build(query, injectedTweets, promotedTweets, whoToFollowUsers) + + val emptyTimelineEvents = EmptyTimelineEventsBuilder.build(query, injectedTweets) + + val queryEvents = QueryEventsBuilder.build(query, injectedTweets) + + (servedEvents ++ emptyTimelineEvents ++ queryEvents).filter(_.eventValue.forall(_ > 0)) + } + + override val alerts = Seq( + HomeMixerAlertConfig.BusinessHours.defaultSuccessRateAlert(99.9) + ) +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/side_effect/HomeScribeServedEntriesSideEffect.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/side_effect/HomeScribeServedEntriesSideEffect.scala new file mode 100644 index 0000000000..ab74360321 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/side_effect/HomeScribeServedEntriesSideEffect.scala @@ -0,0 +1,212 @@ +package com.twitter.home_mixer.functional_component.side_effect + +import com.twitter.finagle.tracing.Trace +import com.twitter.home_mixer.marshaller.timeline_logging.ConversationEntryMarshaller +import com.twitter.home_mixer.marshaller.timeline_logging.PromotedTweetEntryMarshaller +import com.twitter.home_mixer.marshaller.timeline_logging.TweetEntryMarshaller +import com.twitter.home_mixer.marshaller.timeline_logging.WhoToFollowEntryMarshaller +import com.twitter.home_mixer.model.HomeFeatures.GetInitialFeature +import com.twitter.home_mixer.model.HomeFeatures.GetMiddleFeature +import com.twitter.home_mixer.model.HomeFeatures.GetNewerFeature +import com.twitter.home_mixer.model.HomeFeatures.GetOlderFeature +import com.twitter.home_mixer.model.HomeFeatures.HasDarkRequestFeature +import com.twitter.home_mixer.model.HomeFeatures.RequestJoinIdFeature +import com.twitter.home_mixer.model.HomeFeatures.ServedRequestIdFeature +import com.twitter.home_mixer.model.request.DeviceContext.RequestContext +import com.twitter.home_mixer.model.request.HasDeviceContext +import com.twitter.home_mixer.model.request.HasSeenTweetIds +import com.twitter.home_mixer.model.request.FollowingProduct +import com.twitter.home_mixer.model.request.ForYouProduct +import com.twitter.home_mixer.service.HomeMixerAlertConfig +import com.twitter.logpipeline.client.common.EventPublisher +import com.twitter.product_mixer.component_library.model.candidate.BaseTweetCandidate +import com.twitter.product_mixer.component_library.model.candidate.BaseUserCandidate +import com.twitter.product_mixer.component_library.pipeline.candidate.who_to_follow_module.WhoToFollowCandidateDecorator +import com.twitter.product_mixer.core.functional_component.side_effect.PipelineResultSideEffect +import com.twitter.product_mixer.core.model.common.identifier.SideEffectIdentifier +import com.twitter.product_mixer.core.model.common.presentation.ItemCandidateWithDetails +import com.twitter.product_mixer.core.model.common.presentation.ModuleCandidateWithDetails +import com.twitter.product_mixer.core.model.marshalling.response.urt.AddEntriesTimelineInstruction +import com.twitter.product_mixer.core.model.marshalling.response.urt.ModuleItem +import com.twitter.product_mixer.core.model.marshalling.response.urt.Timeline +import com.twitter.product_mixer.core.model.marshalling.response.urt.TimelineModule +import com.twitter.product_mixer.core.model.marshalling.response.urt.item.tweet.TweetItem +import com.twitter.product_mixer.core.model.marshalling.response.urt.item.user.UserItem +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.stitch.Stitch +import com.twitter.timelines.timeline_logging.{thriftscala => thrift} +import com.twitter.util.Time +import javax.inject.Inject +import javax.inject.Singleton + +/** + * Side effect that logs home timeline served entries to Scribe. + */ +@Singleton +class HomeScribeServedEntriesSideEffect @Inject() ( + scribeEventPublisher: EventPublisher[thrift.Timeline]) + extends PipelineResultSideEffect[ + PipelineQuery with HasSeenTweetIds with HasDeviceContext, + Timeline + ] { + + override val identifier: SideEffectIdentifier = SideEffectIdentifier("HomeScribeServedEntries") + + final override def apply( + inputs: PipelineResultSideEffect.Inputs[ + PipelineQuery with HasSeenTweetIds with HasDeviceContext, + Timeline + ] + ): Stitch[Unit] = { + val timelineThrift = buildTimeline(inputs) + Stitch.callFuture(scribeEventPublisher.publish(timelineThrift)).unit + } + + def buildTimeline( + inputs: PipelineResultSideEffect.Inputs[ + PipelineQuery with HasSeenTweetIds with HasDeviceContext, + Timeline + ] + ): thrift.Timeline = { + val timelineType = inputs.query.product match { + case FollowingProduct => thrift.TimelineType.HomeLatest + case ForYouProduct => thrift.TimelineType.Home + case other => throw new UnsupportedOperationException(s"Unknown product: $other") + } + val requestProvenance = inputs.query.deviceContext.map { deviceContext => + deviceContext.requestContextValue match { + case RequestContext.Foreground => thrift.RequestProvenance.Foreground + case RequestContext.Launch => thrift.RequestProvenance.Launch + case RequestContext.PullToRefresh => thrift.RequestProvenance.Ptr + case _ => thrift.RequestProvenance.Other + } + } + val queryType = inputs.query.features.map { featureMap => + if (featureMap.getOrElse(GetOlderFeature, false)) thrift.QueryType.GetOlder + else if (featureMap.getOrElse(GetNewerFeature, false)) thrift.QueryType.GetNewer + else if (featureMap.getOrElse(GetMiddleFeature, false)) thrift.QueryType.GetMiddle + else if (featureMap.getOrElse(GetInitialFeature, false)) thrift.QueryType.GetInitial + else thrift.QueryType.Other + } + + val tweetIdToItemCandidateMap: Map[Long, ItemCandidateWithDetails] = + inputs.selectedCandidates.flatMap { + case item: ItemCandidateWithDetails if item.candidate.isInstanceOf[BaseTweetCandidate] => + Seq((item.candidateIdLong, item)) + case module: ModuleCandidateWithDetails + if module.candidates.headOption.exists(_.candidate.isInstanceOf[BaseTweetCandidate]) => + module.candidates.map(item => (item.candidateIdLong, item)) + case _ => Seq.empty + }.toMap + + val userIdToItemCandidateMap: Map[Long, ItemCandidateWithDetails] = + inputs.selectedCandidates.flatMap { + case module: ModuleCandidateWithDetails + if module.candidates.forall(_.candidate.isInstanceOf[BaseUserCandidate]) => + module.candidates.map { item => + (item.candidateIdLong, item) + } + case _ => Seq.empty + }.toMap + + val timelineEntries = inputs.response.instructions.zipWithIndex.collect { + case (AddEntriesTimelineInstruction(entries), index) => + entries.collect { + case entry: TweetItem if entry.promotedMetadata.isDefined => + val promotedTweetEntry = PromotedTweetEntryMarshaller(entry, index) + Seq( + thrift.TimelineEntry( + content = thrift.Content.PromotedTweetEntry(promotedTweetEntry), + position = index.shortValue(), + entryId = entry.entryIdentifier, + entryType = thrift.EntryType.PromotedTweet, + sortIndex = entry.sortIndex, + verticalSize = Some(1) + ) + ) + case entry: TweetItem => + val candidate = tweetIdToItemCandidateMap(entry.id) + val tweetEntry = TweetEntryMarshaller(entry, candidate) + Seq( + thrift.TimelineEntry( + content = thrift.Content.TweetEntry(tweetEntry), + position = index.shortValue(), + entryId = entry.entryIdentifier, + entryType = thrift.EntryType.Tweet, + sortIndex = entry.sortIndex, + verticalSize = Some(1) + ) + ) + case module: TimelineModule + if module.entryNamespace.toString == WhoToFollowCandidateDecorator.EntryNamespaceString => + val whoToFollowEntries = module.items.collect { + case ModuleItem(entry: UserItem, _, _) => + val candidate = userIdToItemCandidateMap(entry.id) + val whoToFollowEntry = WhoToFollowEntryMarshaller(entry, candidate) + thrift.AtomicEntry.WtfEntry(whoToFollowEntry) + } + Seq( + thrift.TimelineEntry( + content = thrift.Content.Entries(whoToFollowEntries), + position = index.shortValue(), + entryId = module.entryIdentifier, + entryType = thrift.EntryType.WhoToFollowModule, + sortIndex = module.sortIndex + ) + ) + case module: TimelineModule + if module.sortIndex.isDefined && module.items.headOption.exists( + _.item.isInstanceOf[TweetItem]) => + val conversationTweetEntries = module.items.collect { + case ModuleItem(entry: TweetItem, _, _) => + val candidate = tweetIdToItemCandidateMap(entry.id) + val conversationEntry = ConversationEntryMarshaller(entry, candidate) + thrift.AtomicEntry.ConversationEntry(conversationEntry) + } + Seq( + thrift.TimelineEntry( + content = thrift.Content.Entries(conversationTweetEntries), + position = index.shortValue(), + entryId = module.entryIdentifier, + entryType = thrift.EntryType.ConversationModule, + sortIndex = module.sortIndex + ) + ) + case _ => Seq.empty + }.flatten + // Other instructions + case _ => Seq.empty[thrift.TimelineEntry] + }.flatten + + thrift.Timeline( + timelineEntries = timelineEntries, + requestTimeMs = inputs.query.queryTime.inMilliseconds, + traceId = Trace.id.traceId.toLong, + userId = inputs.query.getOptionalUserId, + clientAppId = inputs.query.clientContext.appId, + sourceJobInstance = None, + hasDarkRequest = inputs.query.features.flatMap(_.getOrElse(HasDarkRequestFeature, None)), + parentId = Some(Trace.id.parentId.toLong), + spanId = Some(Trace.id.spanId.toLong), + timelineType = Some(timelineType), + ipAddress = inputs.query.clientContext.ipAddress, + userAgent = inputs.query.clientContext.userAgent, + queryType = queryType, + requestProvenance = requestProvenance, + sessionId = None, + timeZone = None, + browserNotificationPermission = None, + lastNonePollingTimeMs = None, + languageCode = inputs.query.clientContext.languageCode, + countryCode = inputs.query.clientContext.countryCode, + requestEndTimeMs = Some(Time.now.inMilliseconds), + servedRequestId = inputs.query.features.flatMap(_.getOrElse(ServedRequestIdFeature, None)), + requestJoinId = inputs.query.features.flatMap(_.getOrElse(RequestJoinIdFeature, None)), + requestSeenTweetIds = inputs.query.seenTweetIds + ) + } + + override val alerts = Seq( + HomeMixerAlertConfig.BusinessHours.defaultSuccessRateAlert() + ) +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/side_effect/PublishClientSentImpressionsEventBusSideEffect.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/side_effect/PublishClientSentImpressionsEventBusSideEffect.scala new file mode 100644 index 0000000000..3d85d11373 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/side_effect/PublishClientSentImpressionsEventBusSideEffect.scala @@ -0,0 +1,92 @@ +package com.twitter.home_mixer.functional_component.side_effect + +import com.twitter.eventbus.client.EventBusPublisher +import com.twitter.home_mixer.model.request.FollowingProduct +import com.twitter.home_mixer.model.request.ForYouProduct +import com.twitter.home_mixer.model.request.HasSeenTweetIds +import com.twitter.home_mixer.service.HomeMixerAlertConfig +import com.twitter.product_mixer.core.functional_component.side_effect.PipelineResultSideEffect +import com.twitter.product_mixer.core.model.common.identifier.SideEffectIdentifier +import com.twitter.product_mixer.core.model.common.presentation.CandidateWithDetails +import com.twitter.product_mixer.core.model.marshalling.HasMarshalling +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.stitch.Stitch +import com.twitter.timelines.impressionstore.thriftscala.Impression +import com.twitter.timelines.impressionstore.thriftscala.ImpressionList +import com.twitter.timelines.impressionstore.thriftscala.PublishedImpressionList +import com.twitter.timelines.impressionstore.thriftscala.SurfaceArea +import com.twitter.util.Time +import javax.inject.Inject +import javax.inject.Singleton + +object PublishClientSentImpressionsEventBusSideEffect { + val HomeSurfaceArea: Option[Set[SurfaceArea]] = Some(Set(SurfaceArea.HomeTimeline)) + val HomeLatestSurfaceArea: Option[Set[SurfaceArea]] = Some(Set(SurfaceArea.HomeLatestTimeline)) +} + +/** + * Side effect that publishes seen tweet IDs sent from clients. The seen tweet IDs are sent to a + * heron topology which writes to a memcache dataset. + */ +@Singleton +class PublishClientSentImpressionsEventBusSideEffect @Inject() ( + eventBusPublisher: EventBusPublisher[PublishedImpressionList]) + extends PipelineResultSideEffect[PipelineQuery with HasSeenTweetIds, HasMarshalling] + with PipelineResultSideEffect.Conditionally[ + PipelineQuery with HasSeenTweetIds, + HasMarshalling + ] { + import PublishClientSentImpressionsEventBusSideEffect._ + + override val identifier: SideEffectIdentifier = + SideEffectIdentifier("PublishClientSentImpressionsEventBus") + + override def onlyIf( + query: PipelineQuery with HasSeenTweetIds, + selectedCandidates: Seq[CandidateWithDetails], + remainingCandidates: Seq[CandidateWithDetails], + droppedCandidates: Seq[CandidateWithDetails], + response: HasMarshalling + ): Boolean = query.seenTweetIds.exists(_.nonEmpty) + + def buildEvents( + query: PipelineQuery with HasSeenTweetIds, + currentTime: Long + ): Option[Seq[Impression]] = { + val surfaceArea = query.product match { + case ForYouProduct => HomeSurfaceArea + case FollowingProduct => HomeLatestSurfaceArea + case _ => None + } + query.seenTweetIds.map { seenTweetIds => + seenTweetIds.map { tweetId => + Impression( + tweetId = tweetId, + impressionTime = Some(currentTime), + surfaceAreas = surfaceArea + ) + } + } + } + + final override def apply( + inputs: PipelineResultSideEffect.Inputs[PipelineQuery with HasSeenTweetIds, HasMarshalling] + ): Stitch[Unit] = { + val currentTime = Time.now.inMilliseconds + val impressions = buildEvents(inputs.query, currentTime) + + Stitch.callFuture( + eventBusPublisher.publish( + PublishedImpressionList( + inputs.query.getRequiredUserId, + ImpressionList(impressions), + currentTime + ) + ) + ) + } + + override val alerts = Seq( + HomeMixerAlertConfig.BusinessHours.defaultSuccessRateAlert(99.4) + ) +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/side_effect/PublishClientSentImpressionsManhattanSideEffect.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/side_effect/PublishClientSentImpressionsManhattanSideEffect.scala new file mode 100644 index 0000000000..bf365f1a63 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/side_effect/PublishClientSentImpressionsManhattanSideEffect.scala @@ -0,0 +1,65 @@ +package com.twitter.home_mixer.functional_component.side_effect + +import com.twitter.home_mixer.model.HomeFeatures.TweetImpressionsFeature +import com.twitter.home_mixer.model.request.HasSeenTweetIds +import com.twitter.home_mixer.service.HomeMixerAlertConfig +import com.twitter.product_mixer.core.functional_component.side_effect.PipelineResultSideEffect +import com.twitter.product_mixer.core.model.common.identifier.SideEffectIdentifier +import com.twitter.product_mixer.core.model.common.presentation.CandidateWithDetails +import com.twitter.product_mixer.core.model.marshalling.HasMarshalling +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.stitch.Stitch +import com.twitter.timelines.impression.{thriftscala => t} +import com.twitter.timelines.impressionstore.store.ManhattanTweetImpressionStoreClient +import javax.inject.Inject +import javax.inject.Singleton + +/** + * Side effect that updates the timelines tweet impression + * store (Manhattan) with seen tweet IDs sent from clients + */ +@Singleton +class PublishClientSentImpressionsManhattanSideEffect @Inject() ( + manhattanTweetImpressionStoreClient: ManhattanTweetImpressionStoreClient) + extends PipelineResultSideEffect[PipelineQuery with HasSeenTweetIds, HasMarshalling] + with PipelineResultSideEffect.Conditionally[ + PipelineQuery with HasSeenTweetIds, + HasMarshalling + ] { + + override val identifier: SideEffectIdentifier = + SideEffectIdentifier("PublishClientSentImpressionsManhattan") + + override def onlyIf( + query: PipelineQuery with HasSeenTweetIds, + selectedCandidates: Seq[CandidateWithDetails], + remainingCandidates: Seq[CandidateWithDetails], + droppedCandidates: Seq[CandidateWithDetails], + response: HasMarshalling + ): Boolean = query.seenTweetIds.exists(_.nonEmpty) + + def buildEvents(query: PipelineQuery): Option[(Long, t.TweetImpressionsEntries)] = { + query.features.flatMap { featureMap => + val impressions = featureMap.getOrElse(TweetImpressionsFeature, Seq.empty) + if (impressions.nonEmpty) + Some((query.getRequiredUserId, t.TweetImpressionsEntries(impressions))) + else None + } + } + + final override def apply( + inputs: PipelineResultSideEffect.Inputs[PipelineQuery with HasSeenTweetIds, HasMarshalling] + ): Stitch[Unit] = { + val events = buildEvents(inputs.query) + + Stitch + .traverse(events) { + case (key, value) => manhattanTweetImpressionStoreClient.write(key, value) + } + .unit + } + + override val alerts = Seq( + HomeMixerAlertConfig.BusinessHours.defaultSuccessRateAlert(99.4) + ) +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/side_effect/ServedCandidateFeatureKeysKafkaSideEffect.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/side_effect/ServedCandidateFeatureKeysKafkaSideEffect.scala new file mode 100644 index 0000000000..52cf0d7200 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/side_effect/ServedCandidateFeatureKeysKafkaSideEffect.scala @@ -0,0 +1,112 @@ +package com.twitter.home_mixer.functional_component.side_effect + +import com.twitter.home_mixer.model.HomeFeatures.CandidateSourceIdFeature +import com.twitter.home_mixer.model.HomeFeatures.InNetworkFeature +import com.twitter.home_mixer.model.HomeFeatures.IsReadFromCacheFeature +import com.twitter.home_mixer.model.HomeFeatures.PredictionRequestIdFeature +import com.twitter.home_mixer.model.HomeFeatures.ServedIdFeature +import com.twitter.home_mixer.model.HomeFeatures.ServedRequestIdFeature +import com.twitter.home_mixer.model.HomeFeatures.SuggestTypeFeature +import com.twitter.home_mixer.param.HomeGlobalParams.EnableServedCandidateKafkaPublishingParam +import com.twitter.home_mixer.service.HomeMixerAlertConfig +import com.twitter.product_mixer.component_library.side_effect.KafkaPublishingSideEffect +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.functional_component.side_effect.PipelineResultSideEffect +import com.twitter.product_mixer.core.model.common.identifier +import com.twitter.product_mixer.core.model.common.identifier.SideEffectIdentifier +import com.twitter.product_mixer.core.model.common.presentation.CandidateWithDetails +import com.twitter.product_mixer.core.model.marshalling.response.urt.Timeline +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.timelines.ml.cont_train.common.domain.non_scalding.ServedCandidateFeatureKeysAdapter +import com.twitter.timelines.ml.cont_train.common.domain.non_scalding.ServedCandidateFeatureKeysFields +import com.twitter.timelines.ml.kafka.serde.CandidateFeatureKeySerde +import com.twitter.timelines.ml.kafka.serde.TBaseSerde +import com.twitter.timelines.served_candidates_logging.{thriftscala => sc} +import com.twitter.timelines.suggests.common.poly_data_record.{thriftjava => pldr} +import com.twitter.timelineservice.suggests.{thriftscala => tls} +import org.apache.kafka.clients.producer.ProducerRecord +import org.apache.kafka.common.serialization.Serializer +import scala.collection.JavaConverters._ + +/** + * Pipeline side-effect that publishes candidate keys to a Kafka topic. + */ +class ServedCandidateFeatureKeysKafkaSideEffect( + topic: String, + sourceIdentifiers: Set[identifier.CandidatePipelineIdentifier]) + extends KafkaPublishingSideEffect[ + sc.CandidateFeatureKey, + pldr.PolyDataRecord, + PipelineQuery, + Timeline + ] + with PipelineResultSideEffect.Conditionally[PipelineQuery, Timeline] { + + import ServedCandidateKafkaSideEffect._ + + override val identifier: SideEffectIdentifier = SideEffectIdentifier("ServedCandidateFeatureKeys") + + override def onlyIf( + query: PipelineQuery, + selectedCandidates: Seq[CandidateWithDetails], + remainingCandidates: Seq[CandidateWithDetails], + droppedCandidates: Seq[CandidateWithDetails], + response: Timeline + ): Boolean = query.params.getBoolean(EnableServedCandidateKafkaPublishingParam) + + override val bootstrapServer: String = "/s/kafka/timeline:kafka-tls" + + override val keySerde: Serializer[sc.CandidateFeatureKey] = + CandidateFeatureKeySerde().serializer() + + override val valueSerde: Serializer[pldr.PolyDataRecord] = + TBaseSerde.Thrift[pldr.PolyDataRecord]().serializer + + override val clientId: String = "home_mixer_served_candidate_feature_keys_producer" + + override def buildRecords( + query: PipelineQuery, + selectedCandidates: Seq[CandidateWithDetails], + remainingCandidates: Seq[CandidateWithDetails], + droppedCandidates: Seq[CandidateWithDetails], + response: Timeline + ): Seq[ProducerRecord[sc.CandidateFeatureKey, pldr.PolyDataRecord]] = { + val servedRequestIdOpt = + query.features.getOrElse(FeatureMap.empty).getOrElse(ServedRequestIdFeature, None) + + extractCandidates(query, selectedCandidates, sourceIdentifiers).map { candidate => + val isReadFromCache = candidate.features.getOrElse(IsReadFromCacheFeature, false) + val servedId = candidate.features.get(ServedIdFeature).get + + val key = sc.CandidateFeatureKey( + tweetId = candidate.candidateIdLong, + viewerId = query.getRequiredUserId, + servedId = servedId) + + val record = + ServedCandidateFeatureKeysAdapter + .adaptToDataRecords( + ServedCandidateFeatureKeysFields( + candidateTweetSourceId = candidate.features + .getOrElse(CandidateSourceIdFeature, None).map(_.value.toLong).getOrElse(2L), + predictionRequestId = + candidate.features.getOrElse(PredictionRequestIdFeature, None).get, + servedRequestIdOpt = if (isReadFromCache) servedRequestIdOpt else None, + servedId = servedId, + injectionModuleName = candidate.getClass.getSimpleName, + viewerFollowsOriginalAuthor = + Some(candidate.features.getOrElse(InNetworkFeature, true)), + suggestType = candidate.features + .getOrElse(SuggestTypeFeature, None).getOrElse(tls.SuggestType.RankedOrganicTweet), + finalPositionIndex = Some(candidate.sourcePosition), + isReadFromCache = isReadFromCache + )).asScala.head + + new ProducerRecord(topic, key, pldr.PolyDataRecord.dataRecord(record)) + } + } + + override val alerts = Seq( + HomeMixerAlertConfig.BusinessHours.defaultSuccessRateAlert(98.5) + ) +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/side_effect/ServedCandidateFeatureKeysKafkaSideEffectBuilder.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/side_effect/ServedCandidateFeatureKeysKafkaSideEffectBuilder.scala new file mode 100644 index 0000000000..c840719864 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/side_effect/ServedCandidateFeatureKeysKafkaSideEffectBuilder.scala @@ -0,0 +1,20 @@ +package com.twitter.home_mixer.functional_component.side_effect + +import com.twitter.finagle.mtls.authentication.ServiceIdentifier +import com.twitter.product_mixer.core.model.common.identifier.CandidatePipelineIdentifier +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +case class ServedCandidateFeatureKeysKafkaSideEffectBuilder @Inject() ( + injectedServiceIdentifier: ServiceIdentifier) { + def build( + sourceIdentifiers: Set[CandidatePipelineIdentifier] + ): ServedCandidateFeatureKeysKafkaSideEffect = { + val topic = injectedServiceIdentifier.environment.toLowerCase match { + case "prod" => "tq_ct_served_candidate_feature_keys" + case _ => "tq_ct_served_candidate_feature_keys_staging" + } + new ServedCandidateFeatureKeysKafkaSideEffect(topic, sourceIdentifiers) + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/side_effect/ServedCandidateKafkaSideEffect.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/side_effect/ServedCandidateKafkaSideEffect.scala new file mode 100644 index 0000000000..a59a2219a7 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/side_effect/ServedCandidateKafkaSideEffect.scala @@ -0,0 +1,50 @@ +package com.twitter.home_mixer.functional_component.side_effect + +import com.twitter.home_mixer.model.HomeFeatures.IsReadFromCacheFeature +import com.twitter.home_mixer.model.HomeFeatures.PredictionRequestIdFeature +import com.twitter.home_mixer.model.HomeFeatures.ServedIdFeature +import com.twitter.home_mixer.model.HomeFeatures.ServedRequestIdFeature +import com.twitter.home_mixer.model.HomeFeatures.StreamToKafkaFeature +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.model.common.identifier.CandidatePipelineIdentifier +import com.twitter.product_mixer.core.model.common.presentation.CandidateWithDetails +import com.twitter.product_mixer.core.model.common.presentation.ItemCandidateWithDetails +import com.twitter.product_mixer.core.model.common.presentation.ModuleCandidateWithDetails +import com.twitter.product_mixer.core.pipeline.PipelineQuery + +object ServedCandidateKafkaSideEffect { + + def extractCandidates( + query: PipelineQuery, + selectedCandidates: Seq[CandidateWithDetails], + sourceIdentifiers: Set[CandidatePipelineIdentifier] + ): Seq[ItemCandidateWithDetails] = { + val servedRequestIdOpt = + query.features.getOrElse(FeatureMap.empty).getOrElse(ServedRequestIdFeature, None) + + selectedCandidates.iterator + .filter(candidate => sourceIdentifiers.contains(candidate.source)) + .flatMap { + case item: ItemCandidateWithDetails => Seq(item) + case module: ModuleCandidateWithDetails => module.candidates + } + .filter(candidate => candidate.features.getOrElse(StreamToKafkaFeature, false)) + .map { candidate => + val servedId = + if (candidate.features.getOrElse(IsReadFromCacheFeature, false) && + servedRequestIdOpt.nonEmpty) + servedRequestIdOpt + else + candidate.features.getOrElse(PredictionRequestIdFeature, None) + + candidate.copy(features = candidate.features + (ServedIdFeature, servedId)) + }.toSeq + // deduplicate by (tweetId, userId, servedId) + .groupBy { candidate => + ( + candidate.candidateIdLong, + query.getRequiredUserId, + candidate.features.getOrElse(ServedIdFeature, None)) + }.values.map(_.head).toSeq + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/side_effect/ServedCandidateKeysKafkaSideEffect.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/side_effect/ServedCandidateKeysKafkaSideEffect.scala new file mode 100644 index 0000000000..28db059c84 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/side_effect/ServedCandidateKeysKafkaSideEffect.scala @@ -0,0 +1,111 @@ +package com.twitter.home_mixer.functional_component.side_effect + +import com.twitter.home_mixer.model.HomeFeatures.IsReadFromCacheFeature +import com.twitter.home_mixer.model.HomeFeatures.PredictionRequestIdFeature +import com.twitter.home_mixer.model.HomeFeatures.ServedIdFeature +import com.twitter.home_mixer.model.HomeFeatures.ServedRequestIdFeature +import com.twitter.home_mixer.param.HomeGlobalParams.EnableServedCandidateKafkaPublishingParam +import com.twitter.home_mixer.service.HomeMixerAlertConfig +import com.twitter.ml.api.DataRecord +import com.twitter.ml.api.util.SRichDataRecord +import com.twitter.product_mixer.component_library.side_effect.KafkaPublishingSideEffect +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.functional_component.side_effect.PipelineResultSideEffect +import com.twitter.product_mixer.core.model.common.identifier.CandidatePipelineIdentifier +import com.twitter.product_mixer.core.model.common.identifier.SideEffectIdentifier +import com.twitter.product_mixer.core.model.common.presentation.CandidateWithDetails +import com.twitter.product_mixer.core.model.marshalling.response.urt.Timeline +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.timelines.ml.cont_train.common.domain.non_scalding.DataRecordLoggingRelatedFeatures.tlmServedKeysFeatureContext +import com.twitter.timelines.ml.kafka.serde.ServedCandidateKeySerde +import com.twitter.timelines.ml.kafka.serde.TBaseSerde +import com.twitter.timelines.prediction.features.common.TimelinesSharedFeatures +import com.twitter.timelines.served_candidates_logging.{thriftscala => sc} +import com.twitter.timelines.suggests.common.poly_data_record.{thriftjava => pldr} +import com.twitter.util.Time +import org.apache.kafka.clients.producer.ProducerRecord +import org.apache.kafka.common.serialization.Serializer + +/** + * Pipeline side-effect that publishes candidate keys to a Kafka topic. + */ +class ServedCandidateKeysKafkaSideEffect( + topic: String, + sourceIdentifiers: Set[CandidatePipelineIdentifier]) + extends KafkaPublishingSideEffect[ + sc.ServedCandidateKey, + pldr.PolyDataRecord, + PipelineQuery, + Timeline + ] + with PipelineResultSideEffect.Conditionally[PipelineQuery, Timeline] { + + import ServedCandidateKafkaSideEffect._ + + override val identifier: SideEffectIdentifier = SideEffectIdentifier("ServedCandidateKeys") + + override def onlyIf( + query: PipelineQuery, + selectedCandidates: Seq[CandidateWithDetails], + remainingCandidates: Seq[CandidateWithDetails], + droppedCandidates: Seq[CandidateWithDetails], + response: Timeline + ): Boolean = query.params.getBoolean(EnableServedCandidateKafkaPublishingParam) + + override val bootstrapServer: String = "/s/kafka/timeline:kafka-tls" + + override val keySerde: Serializer[sc.ServedCandidateKey] = ServedCandidateKeySerde.serializer() + + override val valueSerde: Serializer[pldr.PolyDataRecord] = + TBaseSerde.Thrift[pldr.PolyDataRecord]().serializer + + override val clientId: String = "home_mixer_served_candidate_keys_producer" + + override def buildRecords( + query: PipelineQuery, + selectedCandidates: Seq[CandidateWithDetails], + remainingCandidates: Seq[CandidateWithDetails], + droppedCandidates: Seq[CandidateWithDetails], + response: Timeline + ): Seq[ProducerRecord[sc.ServedCandidateKey, pldr.PolyDataRecord]] = { + val servedTimestamp = Time.now.inMilliseconds + val servedRequestIdOpt = + query.features.getOrElse(FeatureMap.empty).getOrElse(ServedRequestIdFeature, None) + + extractCandidates(query, selectedCandidates, sourceIdentifiers).collect { + // Only publish non-cached tweets to the ServedCandidateKey topic + case candidate if !candidate.features.getOrElse(IsReadFromCacheFeature, false) => + val key = sc.ServedCandidateKey( + tweetId = candidate.candidateIdLong, + viewerId = query.getRequiredUserId, + servedId = -1L + ) + + val record = SRichDataRecord(new DataRecord, tlmServedKeysFeatureContext) + record.setFeatureValueFromOption( + TimelinesSharedFeatures.PREDICTION_REQUEST_ID, + candidate.features.getOrElse(PredictionRequestIdFeature, None) + ) + record + .setFeatureValueFromOption(TimelinesSharedFeatures.SERVED_REQUEST_ID, servedRequestIdOpt) + record.setFeatureValueFromOption( + TimelinesSharedFeatures.SERVED_ID, + candidate.features.getOrElse(ServedIdFeature, None) + ) + record.setFeatureValueFromOption( + TimelinesSharedFeatures.INJECTION_TYPE, + record.getFeatureValueOpt(TimelinesSharedFeatures.INJECTION_TYPE)) + record.setFeatureValue( + TimelinesSharedFeatures.SERVED_TIMESTAMP, + servedTimestamp + ) + record.record.dropUnknownFeatures() + + new ProducerRecord(topic, key, pldr.PolyDataRecord.dataRecord(record.getRecord)) + } + } + + override val alerts = Seq( + HomeMixerAlertConfig.BusinessHours.defaultSuccessRateAlert(98.5) + ) +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/side_effect/ServedCandidateKeysKafkaSideEffectBuilder.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/side_effect/ServedCandidateKeysKafkaSideEffectBuilder.scala new file mode 100644 index 0000000000..5e86fdadab --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/side_effect/ServedCandidateKeysKafkaSideEffectBuilder.scala @@ -0,0 +1,20 @@ +package com.twitter.home_mixer.functional_component.side_effect + +import com.twitter.finagle.mtls.authentication.ServiceIdentifier +import com.twitter.product_mixer.core.model.common.identifier.CandidatePipelineIdentifier +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +case class ServedCandidateKeysKafkaSideEffectBuilder @Inject() ( + injectedServiceIdentifier: ServiceIdentifier) { + def build( + sourceIdentifiers: Set[CandidatePipelineIdentifier] + ): ServedCandidateKeysKafkaSideEffect = { + val topic = injectedServiceIdentifier.environment.toLowerCase match { + case "prod" => "tq_ct_served_candidate_keys" + case _ => "tq_ct_served_candidate_keys_staging" + } + new ServedCandidateKeysKafkaSideEffect(topic, sourceIdentifiers) + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/side_effect/ServedStatsSideEffect.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/side_effect/ServedStatsSideEffect.scala new file mode 100644 index 0000000000..3a24a59270 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/side_effect/ServedStatsSideEffect.scala @@ -0,0 +1,80 @@ +package com.twitter.home_mixer.functional_component.side_effect + +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.home_mixer.model.HomeFeatures.AuthorIdFeature +import com.twitter.home_mixer.model.HomeFeatures.CandidateSourceIdFeature +import com.twitter.home_mixer.model.HomeFeatures.InNetworkFeature +import com.twitter.home_mixer.model.HomeFeatures.InReplyToTweetIdFeature +import com.twitter.home_mixer.model.HomeFeatures.IsRetweetFeature +import com.twitter.home_mixer.param.HomeGlobalParams.AuthorListForStatsParam +import com.twitter.home_mixer.util.CandidatesUtil +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.functional_component.side_effect.PipelineResultSideEffect +import com.twitter.product_mixer.core.model.common.identifier.SideEffectIdentifier +import com.twitter.product_mixer.core.model.common.presentation.CandidateWithDetails +import com.twitter.product_mixer.core.model.common.presentation.ItemCandidateWithDetails +import com.twitter.product_mixer.core.model.marshalling.response.urt.Timeline +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.stitch.Stitch +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class ServedStatsSideEffect @Inject() (statsReceiver: StatsReceiver) + extends PipelineResultSideEffect[PipelineQuery, Timeline] { + + override val identifier: SideEffectIdentifier = SideEffectIdentifier("ServedStats") + + private val baseStatsReceiver = statsReceiver.scope(identifier.toString) + private val authorStatsReceiver = baseStatsReceiver.scope("Author") + private val candidateSourceStatsReceiver = baseStatsReceiver.scope("CandidateSource") + private val contentBalanceStatsReceiver = baseStatsReceiver.scope("ContentBalance") + private val inNetworkStatsCounter = contentBalanceStatsReceiver.counter("InNetwork") + private val outOfNetworkStatsCounter = contentBalanceStatsReceiver.counter("OutOfNetwork") + + override def apply( + inputs: PipelineResultSideEffect.Inputs[PipelineQuery, Timeline] + ): Stitch[Unit] = { + val tweetCandidates = CandidatesUtil + .getItemCandidates(inputs.selectedCandidates).filter(_.isCandidateType[TweetCandidate]()) + + recordAuthorStats(tweetCandidates, inputs.query.params(AuthorListForStatsParam)) + recordCandidateSourceStats(tweetCandidates) + recordContentBalanceStats(tweetCandidates) + Stitch.Unit + } + + def recordAuthorStats(candidates: Seq[CandidateWithDetails], authors: Set[Long]): Unit = { + candidates + .filter { candidate => + candidate.features.getOrElse(AuthorIdFeature, None).exists(authors.contains) && + // Only include original tweets + (!candidate.features.getOrElse(IsRetweetFeature, false)) && + candidate.features.getOrElse(InReplyToTweetIdFeature, None).isEmpty + } + .groupBy { candidate => + (getCandidateSourceId(candidate), candidate.features.get(AuthorIdFeature).get) + } + .foreach { + case ((candidateSourceId, authorId), authorCandidates) => + authorStatsReceiver + .scope(authorId.toString).counter(candidateSourceId).incr(authorCandidates.size) + } + } + + def recordCandidateSourceStats(candidates: Seq[ItemCandidateWithDetails]): Unit = { + candidates.groupBy(getCandidateSourceId).foreach { + case (candidateSourceId, candidateSourceCandidates) => + candidateSourceStatsReceiver.counter(candidateSourceId).incr(candidateSourceCandidates.size) + } + } + + def recordContentBalanceStats(candidates: Seq[ItemCandidateWithDetails]): Unit = { + val (in, oon) = candidates.partition(_.features.getOrElse(InNetworkFeature, true)) + inNetworkStatsCounter.incr(in.size) + outOfNetworkStatsCounter.incr(oon.size) + } + + private def getCandidateSourceId(candidate: CandidateWithDetails): String = + candidate.features.getOrElse(CandidateSourceIdFeature, None).map(_.name).getOrElse("None") +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/side_effect/TruncateTimelinesPersistenceStoreSideEffect.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/side_effect/TruncateTimelinesPersistenceStoreSideEffect.scala new file mode 100644 index 0000000000..63855d3b8f --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/side_effect/TruncateTimelinesPersistenceStoreSideEffect.scala @@ -0,0 +1,68 @@ +package com.twitter.home_mixer.functional_component.side_effect + +import com.twitter.home_mixer.model.HomeFeatures.PersistenceEntriesFeature +import com.twitter.home_mixer.model.request.FollowingProduct +import com.twitter.home_mixer.model.request.ForYouProduct +import com.twitter.home_mixer.param.HomeGlobalParams.TimelinesPersistenceStoreMaxEntriesPerClient +import com.twitter.home_mixer.service.HomeMixerAlertConfig +import com.twitter.product_mixer.core.functional_component.side_effect.PipelineResultSideEffect +import com.twitter.product_mixer.core.model.common.identifier.SideEffectIdentifier +import com.twitter.product_mixer.core.model.marshalling.response.urt.Timeline +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.stitch.Stitch +import com.twitter.timelinemixer.clients.persistence.TimelineResponseBatchesClient +import com.twitter.timelinemixer.clients.persistence.TimelineResponseV3 +import com.twitter.timelineservice.model.TimelineQuery +import com.twitter.timelineservice.model.core.TimelineKind +import javax.inject.Inject +import javax.inject.Singleton + +/** + * Side effect that truncates entries in the Timelines Persistence store + * based on the number of entries per client. + */ +@Singleton +class TruncateTimelinesPersistenceStoreSideEffect @Inject() ( + timelineResponseBatchesClient: TimelineResponseBatchesClient[TimelineResponseV3]) + extends PipelineResultSideEffect[PipelineQuery, Timeline] { + + override val identifier: SideEffectIdentifier = + SideEffectIdentifier("TruncateTimelinesPersistenceStore") + + def getResponsesToDelete(query: PipelineQuery): Seq[TimelineResponseV3] = { + val responses = + query.features.map(_.getOrElse(PersistenceEntriesFeature, Seq.empty)).toSeq.flatten + val responsesByClient = responses.groupBy(_.clientPlatform).values.toSeq + val maxEntriesPerClient = query.params(TimelinesPersistenceStoreMaxEntriesPerClient) + + responsesByClient.flatMap { + _.sortBy(_.servedTime.inMilliseconds) + .foldRight((Seq.empty[TimelineResponseV3], maxEntriesPerClient)) { + case (response, (responsesToDelete, remainingCap)) => + if (remainingCap > 0) (responsesToDelete, remainingCap - response.entries.size) + else (response +: responsesToDelete, remainingCap) + } match { case (responsesToDelete, _) => responsesToDelete } + } + } + + final override def apply( + inputs: PipelineResultSideEffect.Inputs[PipelineQuery, Timeline] + ): Stitch[Unit] = { + val timelineKind = inputs.query.product match { + case FollowingProduct => TimelineKind.homeLatest + case ForYouProduct => TimelineKind.home + case other => throw new UnsupportedOperationException(s"Unknown product: $other") + } + val timelineQuery = TimelineQuery(id = inputs.query.getRequiredUserId, kind = timelineKind) + + val responsesToDelete = getResponsesToDelete(inputs.query) + + if (responsesToDelete.nonEmpty) + Stitch.callFuture(timelineResponseBatchesClient.delete(timelineQuery, responsesToDelete)) + else Stitch.Unit + } + + override val alerts = Seq( + HomeMixerAlertConfig.BusinessHours.defaultSuccessRateAlert(99.8) + ) +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/side_effect/UpdateImpressionBloomFilterSideEffect.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/side_effect/UpdateImpressionBloomFilterSideEffect.scala new file mode 100644 index 0000000000..957fbcd372 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/side_effect/UpdateImpressionBloomFilterSideEffect.scala @@ -0,0 +1,60 @@ +package com.twitter.home_mixer.functional_component.side_effect + +import com.twitter.conversions.DurationOps._ +import com.twitter.home_mixer.model.HomeFeatures.ImpressionBloomFilterFeature +import com.twitter.home_mixer.model.request.HasSeenTweetIds +import com.twitter.home_mixer.service.HomeMixerAlertConfig +import com.twitter.product_mixer.core.functional_component.side_effect.PipelineResultSideEffect +import com.twitter.product_mixer.core.model.common.identifier.SideEffectIdentifier +import com.twitter.product_mixer.core.model.common.presentation.CandidateWithDetails +import com.twitter.product_mixer.core.model.marshalling.response.urt.Timeline +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.stitch.Stitch +import com.twitter.timelines.impressionbloomfilter.{thriftscala => t} +import com.twitter.timelines.impressionstore.impressionbloomfilter.ImpressionBloomFilter +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class UpdateImpressionBloomFilterSideEffect @Inject() (bloomFilter: ImpressionBloomFilter) + extends PipelineResultSideEffect[PipelineQuery with HasSeenTweetIds, Timeline] + with PipelineResultSideEffect.Conditionally[PipelineQuery with HasSeenTweetIds, Timeline] { + + private val SurfaceArea = t.SurfaceArea.HomeTimeline + + override val identifier: SideEffectIdentifier = + SideEffectIdentifier("UpdateImpressionBloomFilter") + + override def onlyIf( + query: PipelineQuery with HasSeenTweetIds, + selectedCandidates: Seq[CandidateWithDetails], + remainingCandidates: Seq[CandidateWithDetails], + droppedCandidates: Seq[CandidateWithDetails], + response: Timeline + ): Boolean = query.seenTweetIds.exists(_.nonEmpty) + + def buildEvents(query: PipelineQuery): Option[t.ImpressionBloomFilterSeq] = { + query.features.flatMap { featureMap => + val impressionBloomFilterSeq = featureMap.get(ImpressionBloomFilterFeature) + if (impressionBloomFilterSeq.entries.nonEmpty) Some(impressionBloomFilterSeq) + else None + } + } + + override def apply( + inputs: PipelineResultSideEffect.Inputs[PipelineQuery with HasSeenTweetIds, Timeline] + ): Stitch[Unit] = { + buildEvents(inputs.query) + .map { updatedBloomFilter => + bloomFilter.writeBloomFilterSeq( + userId = inputs.query.getRequiredUserId, + surfaceArea = SurfaceArea, + impressionBloomFilterSeq = updatedBloomFilter) + }.getOrElse(Stitch.Unit) + } + + override val alerts = Seq( + HomeMixerAlertConfig.BusinessHours.defaultSuccessRateAlert(99.8), + HomeMixerAlertConfig.BusinessHours.defaultLatencyAlert(30.millis) + ) +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/side_effect/UpdateLastNonPollingTimeSideEffect.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/side_effect/UpdateLastNonPollingTimeSideEffect.scala new file mode 100644 index 0000000000..24199a79c0 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/side_effect/UpdateLastNonPollingTimeSideEffect.scala @@ -0,0 +1,78 @@ +package com.twitter.home_mixer.functional_component.side_effect + +import com.twitter.home_mixer.model.HomeFeatures.FollowingLastNonPollingTimeFeature +import com.twitter.home_mixer.model.HomeFeatures.NonPollingTimesFeature +import com.twitter.home_mixer.model.HomeFeatures.PollingFeature +import com.twitter.home_mixer.model.request.DeviceContext +import com.twitter.home_mixer.model.request.HasDeviceContext +import com.twitter.home_mixer.model.request.FollowingProduct +import com.twitter.home_mixer.service.HomeMixerAlertConfig +import com.twitter.product_mixer.component_library.side_effect.UserSessionStoreUpdateSideEffect +import com.twitter.product_mixer.core.model.common.identifier.SideEffectIdentifier +import com.twitter.product_mixer.core.model.marshalling.HasMarshalling +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.timelineservice.model.util.FinagleRequestContext +import com.twitter.user_session_store.ReadWriteUserSessionStore +import com.twitter.user_session_store.WriteRequest +import com.twitter.user_session_store.thriftscala.NonPollingTimestamps +import com.twitter.user_session_store.thriftscala.UserSessionField +import com.twitter.util.Time + +import javax.inject.Inject +import javax.inject.Singleton + +/** + * Side effect that updates the User Session Store (Manhattan) with the timestamps of non polling requests. + */ +@Singleton +class UpdateLastNonPollingTimeSideEffect[ + Query <: PipelineQuery with HasDeviceContext, + ResponseType <: HasMarshalling] @Inject() ( + override val userSessionStore: ReadWriteUserSessionStore) + extends UserSessionStoreUpdateSideEffect[ + WriteRequest, + Query, + ResponseType + ] { + private val MaxNonPollingTimes = 10 + + override val identifier: SideEffectIdentifier = SideEffectIdentifier("UpdateLastNonPollingTime") + + /** + * When the request is non polling and is not a background fetch request, update + * the list of non polling timestamps with the timestamp of the current request + */ + override def buildWriteRequest(query: Query): Option[WriteRequest] = { + val isBackgroundFetch = query.deviceContext + .exists(_.requestContextValue.contains(DeviceContext.RequestContext.BackgroundFetch)) + + if (!query.features.exists(_.getOrElse(PollingFeature, false)) && !isBackgroundFetch) { + val fields = Seq(UserSessionField.NonPollingTimestamps(makeLastNonPollingTimestamps(query))) + Some(WriteRequest(query.getRequiredUserId, fields)) + } else None + } + + override val alerts = Seq( + HomeMixerAlertConfig.BusinessHours.defaultSuccessRateAlert(99.96) + ) + + private def makeLastNonPollingTimestamps(query: Query): NonPollingTimestamps = { + val priorNonPollingTimestamps = + query.features.map(_.getOrElse(NonPollingTimesFeature, Seq.empty)).toSeq.flatten + + val lastNonPollingTimeMs = + FinagleRequestContext.default.requestStartTime.get.getOrElse(Time.now).inMillis + + val followingLastNonPollingTime = query.features + .flatMap(features => features.getOrElse(FollowingLastNonPollingTimeFeature, None)) + .map(_.inMillis) + + NonPollingTimestamps( + nonPollingTimestampsMs = + (lastNonPollingTimeMs +: priorNonPollingTimestamps).take(MaxNonPollingTimes), + mostRecentHomeLatestNonPollingTimestampMs = + if (query.product == FollowingProduct) Some(lastNonPollingTimeMs) + else followingLastNonPollingTime + ) + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/side_effect/UpdateTimelinesPersistenceStoreSideEffect.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/side_effect/UpdateTimelinesPersistenceStoreSideEffect.scala new file mode 100644 index 0000000000..ef8a737a8c --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/side_effect/UpdateTimelinesPersistenceStoreSideEffect.scala @@ -0,0 +1,243 @@ +package com.twitter.home_mixer.functional_component.side_effect + +import com.twitter.home_mixer.model.HomeFeatures._ +import com.twitter.home_mixer.model.request.FollowingProduct +import com.twitter.home_mixer.model.request.ForYouProduct +import com.twitter.home_mixer.service.HomeMixerAlertConfig +import com.twitter.product_mixer.component_library.pipeline.candidate.who_to_follow_module.WhoToFollowCandidateDecorator +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.functional_component.side_effect.PipelineResultSideEffect +import com.twitter.product_mixer.core.model.common.identifier.SideEffectIdentifier +import com.twitter.product_mixer.core.model.common.presentation.ItemCandidateWithDetails +import com.twitter.product_mixer.core.model.common.presentation.ModuleCandidateWithDetails +import com.twitter.product_mixer.core.model.marshalling.response.urt.AddEntriesTimelineInstruction +import com.twitter.product_mixer.core.model.marshalling.response.urt.ReplaceEntryTimelineInstruction +import com.twitter.product_mixer.core.model.marshalling.response.urt.ShowCoverInstruction +import com.twitter.product_mixer.core.model.marshalling.response.urt.Timeline +import com.twitter.product_mixer.core.model.marshalling.response.urt.TimelineModule +import com.twitter.product_mixer.core.model.marshalling.response.urt.item.tweet.TweetItem +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.stitch.Stitch +import com.twitter.timelinemixer.clients.persistence.EntryWithItemIds +import com.twitter.timelinemixer.clients.persistence.ItemIds +import com.twitter.timelinemixer.clients.persistence.TimelineResponseBatchesClient +import com.twitter.timelinemixer.clients.persistence.TimelineResponseV3 +import com.twitter.timelines.persistence.thriftscala.TweetScoreV1 +import com.twitter.timelines.persistence.{thriftscala => persistence} +import com.twitter.timelineservice.model.TimelineQuery +import com.twitter.timelineservice.model.TimelineQueryOptions +import com.twitter.timelineservice.model.TweetScore +import com.twitter.timelineservice.model.core.TimelineKind +import com.twitter.timelineservice.model.rich.EntityIdType +import com.twitter.util.Time +import com.twitter.{timelineservice => tls} +import javax.inject.Inject +import javax.inject.Singleton + +object UpdateTimelinesPersistenceStoreSideEffect { + val EmptyItemIds = ItemIds( + None, + None, + None, + None, + None, + None, + None, + None, + None, + None, + None, + None, + None, + None, + None) +} + +/** + * Side effect that updates the Timelines Persistence Store (Manhattan) with the entries being returned. + */ +@Singleton +class UpdateTimelinesPersistenceStoreSideEffect @Inject() ( + timelineResponseBatchesClient: TimelineResponseBatchesClient[TimelineResponseV3]) + extends PipelineResultSideEffect[PipelineQuery, Timeline] { + + override val identifier: SideEffectIdentifier = + SideEffectIdentifier("UpdateTimelinesPersistenceStore") + + final override def apply( + inputs: PipelineResultSideEffect.Inputs[PipelineQuery, Timeline] + ): Stitch[Unit] = { + if (inputs.response.instructions.nonEmpty) { + val timelineKind = inputs.query.product match { + case FollowingProduct => TimelineKind.homeLatest + case ForYouProduct => TimelineKind.home + case other => throw new UnsupportedOperationException(s"Unknown product: $other") + } + val timelineQuery = TimelineQuery( + id = inputs.query.getRequiredUserId, + kind = timelineKind, + options = TimelineQueryOptions( + contextualUserId = inputs.query.getOptionalUserId, + deviceContext = tls.DeviceContext.empty.copy( + userAgent = inputs.query.clientContext.userAgent, + clientAppId = inputs.query.clientContext.appId) + ) + ) + + val tweetIdToItemCandidateMap: Map[Long, ItemCandidateWithDetails] = + inputs.selectedCandidates.flatMap { + case item: ItemCandidateWithDetails if item.candidate.id.isInstanceOf[Long] => + Seq((item.candidateIdLong, item)) + case module: ModuleCandidateWithDetails + if module.candidates.headOption.exists(_.candidate.id.isInstanceOf[Long]) => + module.candidates.map(item => (item.candidateIdLong, item)) + case _ => Seq.empty + }.toMap + + val entries = inputs.response.instructions.collect { + case AddEntriesTimelineInstruction(entries) => + entries.collect { + // includes both tweets and promoted tweets + case entry: TweetItem if entry.sortIndex.isDefined => + Seq( + buildTweetEntryWithItemIds( + tweetIdToItemCandidateMap(entry.id), + entry.sortIndex.get)) + // tweet conversation modules are flattened to individual tweets in the persistence store + case module: TimelineModule + if module.sortIndex.isDefined && module.items.headOption.exists( + _.item.isInstanceOf[TweetItem]) => + module.items.map { item => + buildTweetEntryWithItemIds( + tweetIdToItemCandidateMap(item.item.id.asInstanceOf[Long]), + module.sortIndex.get) + } + case module: TimelineModule + if module.sortIndex.isDefined && module.entryNamespace.toString == WhoToFollowCandidateDecorator.EntryNamespaceString => + val userIds = module.items + .map(item => + UpdateTimelinesPersistenceStoreSideEffect.EmptyItemIds.copy(userId = + Some(item.item.id.asInstanceOf[Long]))) + Seq( + EntryWithItemIds( + entityIdType = EntityIdType.WhoToFollow, + sortIndex = module.sortIndex.get, + size = module.items.size.toShort, + itemIds = Some(userIds) + )) + }.flatten + case ShowCoverInstruction(cover) => + Seq( + EntryWithItemIds( + entityIdType = EntityIdType.Prompt, + sortIndex = cover.sortIndex.get, + size = 1, + itemIds = None + ) + ) + case ReplaceEntryTimelineInstruction(entry) => + val namespaceLength = TweetItem.TweetEntryNamespace.toString.length + Seq( + EntryWithItemIds( + entityIdType = EntityIdType.Tweet, + sortIndex = entry.sortIndex.get, + size = 1, + itemIds = Some( + Seq( + ItemIds( + tweetId = + entry.entryIdToReplace.map(e => e.substring(namespaceLength + 1).toLong), + sourceTweetId = None, + quoteTweetId = None, + sourceAuthorId = None, + quoteAuthorId = None, + inReplyToTweetId = None, + inReplyToAuthorId = None, + semanticCoreId = None, + articleId = None, + hasRelevancePrompt = None, + promptData = None, + tweetScore = None, + entryIdToReplace = entry.entryIdToReplace, + tweetReactiveData = None, + userId = None + ) + )) + ) + ) + + }.flatten + + val response = TimelineResponseV3( + clientPlatform = timelineQuery.clientPlatform, + servedTime = Time.now, + requestType = requestTypeFromQuery(inputs.query), + entries = entries) + + Stitch.callFuture(timelineResponseBatchesClient.insertResponse(timelineQuery, response)) + } else Stitch.Unit + } + + override val alerts = Seq( + HomeMixerAlertConfig.BusinessHours.defaultSuccessRateAlert(99.8) + ) + + private def buildTweetEntryWithItemIds( + candidate: ItemCandidateWithDetails, + sortIndex: Long + ): EntryWithItemIds = { + val features = candidate.features + val sourceAuthorId = + if (features.getOrElse(IsRetweetFeature, false)) features.getOrElse(SourceUserIdFeature, None) + else features.getOrElse(AuthorIdFeature, None) + val quoteAuthorId = + if (features.getOrElse(QuotedTweetIdFeature, None).nonEmpty) + features.getOrElse(SourceUserIdFeature, None) + else None + val tweetScore = features.getOrElse(ScoreFeature, None).map { score => + TweetScore.fromThrift(persistence.TweetScore.TweetScoreV1(TweetScoreV1(score))) + } + + val itemIds = ItemIds( + tweetId = Some(candidate.candidateIdLong), + sourceTweetId = features.getOrElse(SourceTweetIdFeature, None), + quoteTweetId = features.getOrElse(QuotedTweetIdFeature, None), + sourceAuthorId = sourceAuthorId, + quoteAuthorId = quoteAuthorId, + inReplyToTweetId = features.getOrElse(InReplyToTweetIdFeature, None), + inReplyToAuthorId = features.getOrElse(DirectedAtUserIdFeature, None), + semanticCoreId = features.getOrElse(SemanticCoreIdFeature, None), + articleId = None, + hasRelevancePrompt = None, + promptData = None, + tweetScore = tweetScore, + entryIdToReplace = None, + tweetReactiveData = None, + userId = None + ) + + EntryWithItemIds( + entityIdType = EntityIdType.Tweet, + sortIndex = sortIndex, + size = 1.toShort, + itemIds = Some(Seq(itemIds)) + ) + } + + private def requestTypeFromQuery(query: PipelineQuery): persistence.RequestType = { + val features = query.features.getOrElse(FeatureMap.empty) + + val featureToRequestType = Seq( + (PollingFeature, persistence.RequestType.Polling), + (GetInitialFeature, persistence.RequestType.Initial), + (GetNewerFeature, persistence.RequestType.Newer), + (GetMiddleFeature, persistence.RequestType.Middle), + (GetOlderFeature, persistence.RequestType.Older) + ) + + featureToRequestType + .collectFirst { + case (feature, requestType) if features.getOrElse(feature, false) => requestType + }.getOrElse(persistence.RequestType.Other) + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/marshaller/request/BUILD.bazel b/home-mixer/server/src/main/scala/com/twitter/home_mixer/marshaller/request/BUILD.bazel new file mode 100644 index 0000000000..fe1ee71901 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/marshaller/request/BUILD.bazel @@ -0,0 +1,20 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + strict_deps = True, + tags = ["bazel-compatible"], + dependencies = [ + "dspbidder/thrift/src/main/thrift/com/twitter/dspbidder/commons:thrift-scala", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/model/request", + "home-mixer/thrift/src/main/thrift:thrift-scala", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/request", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/common", + ], + exports = [ + "dspbidder/thrift/src/main/thrift/com/twitter/dspbidder/commons:thrift-scala", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/model/request", + "home-mixer/thrift/src/main/thrift:thrift-scala", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/request", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/common", + ], +) diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/marshaller/request/DeviceContextUnmarshaller.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/marshaller/request/DeviceContextUnmarshaller.scala new file mode 100644 index 0000000000..b2c6e63475 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/marshaller/request/DeviceContextUnmarshaller.scala @@ -0,0 +1,19 @@ +package com.twitter.home_mixer.marshaller.request + +import com.twitter.home_mixer.model.request.DeviceContext +import com.twitter.home_mixer.{thriftscala => t} +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class DeviceContextUnmarshaller @Inject() () { + + def apply(deviceContext: t.DeviceContext): DeviceContext = { + DeviceContext( + isPolling = deviceContext.isPolling, + requestContext = deviceContext.requestContext, + latestControlAvailable = deviceContext.latestControlAvailable, + autoplayEnabled = deviceContext.autoplayEnabled + ) + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/marshaller/request/HomeMixerDebugParamsUnmarshaller.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/marshaller/request/HomeMixerDebugParamsUnmarshaller.scala new file mode 100644 index 0000000000..81a9abf2b9 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/marshaller/request/HomeMixerDebugParamsUnmarshaller.scala @@ -0,0 +1,27 @@ +package com.twitter.home_mixer.marshaller.request + +import com.twitter.home_mixer.model.request.HomeMixerDebugOptions +import com.twitter.home_mixer.{thriftscala => t} +import com.twitter.product_mixer.core.functional_component.marshaller.request.FeatureValueUnmarshaller +import com.twitter.product_mixer.core.model.marshalling.request.DebugParams +import com.twitter.util.Time +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class HomeMixerDebugParamsUnmarshaller @Inject() ( + featureValueUnmarshaller: FeatureValueUnmarshaller) { + + def apply(debugParams: t.DebugParams): DebugParams = { + DebugParams( + featureOverrides = debugParams.featureOverrides.map { map => + map.mapValues(featureValueUnmarshaller(_)).toMap + }, + debugOptions = debugParams.debugOptions.map { options => + HomeMixerDebugOptions( + requestTimeOverride = options.requestTimeOverrideMillis.map(Time.fromMilliseconds) + ) + } + ) + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/marshaller/request/HomeMixerProductContextUnmarshaller.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/marshaller/request/HomeMixerProductContextUnmarshaller.scala new file mode 100644 index 0000000000..bbc93389ce --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/marshaller/request/HomeMixerProductContextUnmarshaller.scala @@ -0,0 +1,54 @@ +package com.twitter.home_mixer.marshaller.request + +import com.twitter.home_mixer.model.request.FollowingProductContext +import com.twitter.home_mixer.model.request.ForYouProductContext +import com.twitter.home_mixer.model.request.ListRecommendedUsersProductContext +import com.twitter.home_mixer.model.request.ListTweetsProductContext +import com.twitter.home_mixer.model.request.ScoredTweetsProductContext +import com.twitter.home_mixer.{thriftscala => t} +import com.twitter.product_mixer.core.model.marshalling.request.ProductContext + +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class HomeMixerProductContextUnmarshaller @Inject() ( + deviceContextUnmarshaller: DeviceContextUnmarshaller) { + + def apply(productContext: t.ProductContext): ProductContext = productContext match { + case t.ProductContext.Following(p) => + FollowingProductContext( + deviceContext = p.deviceContext.map(deviceContextUnmarshaller(_)), + seenTweetIds = p.seenTweetIds, + dspClientContext = p.dspClientContext + ) + case t.ProductContext.ForYou(p) => + ForYouProductContext( + deviceContext = p.deviceContext.map(deviceContextUnmarshaller(_)), + seenTweetIds = p.seenTweetIds, + dspClientContext = p.dspClientContext + ) + case t.ProductContext.Realtime(p) => + throw new UnsupportedOperationException(s"This product is no longer used") + case t.ProductContext.ScoredTweets(p) => + ScoredTweetsProductContext( + deviceContext = p.deviceContext.map(deviceContextUnmarshaller(_)), + seenTweetIds = p.seenTweetIds, + servedTweetIds = p.servedTweetIds + ) + case t.ProductContext.ListTweets(p) => + ListTweetsProductContext( + listId = p.listId, + deviceContext = p.deviceContext.map(deviceContextUnmarshaller(_)), + dspClientContext = p.dspClientContext + ) + case t.ProductContext.ListRecommendedUsers(p) => + ListRecommendedUsersProductContext( + listId = p.listId, + selectedUserIds = p.selectedUserIds, + excludedUserIds = p.excludedUserIds + ) + case t.ProductContext.UnknownUnionField(field) => + throw new UnsupportedOperationException(s"Unknown display context: ${field.field.name}") + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/marshaller/request/HomeMixerProductUnmarshaller.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/marshaller/request/HomeMixerProductUnmarshaller.scala new file mode 100644 index 0000000000..0089c5efb5 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/marshaller/request/HomeMixerProductUnmarshaller.scala @@ -0,0 +1,28 @@ +package com.twitter.home_mixer.marshaller.request + +import com.twitter.home_mixer.model.request.FollowingProduct +import com.twitter.home_mixer.model.request.ForYouProduct +import com.twitter.home_mixer.model.request.ListRecommendedUsersProduct +import com.twitter.home_mixer.model.request.ListTweetsProduct +import com.twitter.home_mixer.model.request.ScoredTweetsProduct +import com.twitter.home_mixer.{thriftscala => t} +import com.twitter.product_mixer.core.model.marshalling.request.Product + +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class HomeMixerProductUnmarshaller @Inject() () { + + def apply(product: t.Product): Product = product match { + case t.Product.Following => FollowingProduct + case t.Product.ForYou => ForYouProduct + case t.Product.Realtime => + throw new UnsupportedOperationException(s"This product is no longer used") + case t.Product.ScoredTweets => ScoredTweetsProduct + case t.Product.ListTweets => ListTweetsProduct + case t.Product.ListRecommendedUsers => ListRecommendedUsersProduct + case t.Product.EnumUnknownProduct(value) => + throw new UnsupportedOperationException(s"Unknown product: $value") + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/marshaller/request/HomeMixerRequestUnmarshaller.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/marshaller/request/HomeMixerRequestUnmarshaller.scala new file mode 100644 index 0000000000..b8894c8b06 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/marshaller/request/HomeMixerRequestUnmarshaller.scala @@ -0,0 +1,30 @@ +package com.twitter.home_mixer.marshaller.request + +import com.twitter.home_mixer.model.request.HomeMixerRequest +import com.twitter.home_mixer.{thriftscala => t} +import com.twitter.product_mixer.core.functional_component.marshaller.request.ClientContextUnmarshaller +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class HomeMixerRequestUnmarshaller @Inject() ( + clientContextUnmarshaller: ClientContextUnmarshaller, + homeProductUnmarshaller: HomeMixerProductUnmarshaller, + homeProductContextUnmarshaller: HomeMixerProductContextUnmarshaller, + homeDebugParamsUnmarshaller: HomeMixerDebugParamsUnmarshaller) { + + def apply(homeRequest: t.HomeMixerRequest): HomeMixerRequest = { + HomeMixerRequest( + clientContext = clientContextUnmarshaller(homeRequest.clientContext), + product = homeProductUnmarshaller(homeRequest.product), + productContext = homeRequest.productContext.map(homeProductContextUnmarshaller(_)), + // Avoid de-serializing cursors in the request unmarshaller. The unmarshaller should never + // fail, which is often a possibility when trying to de-serialize a cursor. Cursors can also + // be product-specific and more appropriately handled in individual product pipelines. + serializedRequestCursor = homeRequest.cursor, + maxResults = homeRequest.maxResults, + debugParams = homeRequest.debugParams.map(homeDebugParamsUnmarshaller(_)), + homeRequestParam = false + ) + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/marshaller/timeline_logging/BUILD.bazel b/home-mixer/server/src/main/scala/com/twitter/home_mixer/marshaller/timeline_logging/BUILD.bazel new file mode 100644 index 0000000000..efcad840b5 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/marshaller/timeline_logging/BUILD.bazel @@ -0,0 +1,13 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + strict_deps = True, + tags = ["bazel-compatible"], + dependencies = [ + "home-mixer/server/src/main/scala/com/twitter/home_mixer/model", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/presentation/urt", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/pipeline/candidate/who_to_follow_module", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/item", + "src/thrift/com/twitter/timelines/timeline_logging:thrift-scala", + ], +) diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/marshaller/timeline_logging/ConversationEntryMarshaller.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/marshaller/timeline_logging/ConversationEntryMarshaller.scala new file mode 100644 index 0000000000..8123f35974 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/marshaller/timeline_logging/ConversationEntryMarshaller.scala @@ -0,0 +1,16 @@ +package com.twitter.home_mixer.marshaller.timeline_logging + +import com.twitter.home_mixer.model.HomeFeatures.ScoreFeature +import com.twitter.product_mixer.core.model.common.presentation.ItemCandidateWithDetails +import com.twitter.product_mixer.core.model.marshalling.response.urt.item.tweet.TweetItem +import com.twitter.timelines.timeline_logging.{thriftscala => thriftlog} + +object ConversationEntryMarshaller { + + def apply(entry: TweetItem, candidate: ItemCandidateWithDetails): thriftlog.ConversationEntry = + thriftlog.ConversationEntry( + displayedTweetId = entry.id, + displayType = Some(entry.displayType.toString), + score = candidate.features.getOrElse(ScoreFeature, None) + ) +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/marshaller/timeline_logging/PromotedTweetEntryMarshaller.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/marshaller/timeline_logging/PromotedTweetEntryMarshaller.scala new file mode 100644 index 0000000000..b96bb38a2c --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/marshaller/timeline_logging/PromotedTweetEntryMarshaller.scala @@ -0,0 +1,17 @@ +package com.twitter.home_mixer.marshaller.timeline_logging + +import com.twitter.product_mixer.core.model.marshalling.response.urt.item.tweet.TweetItem +import com.twitter.timelines.timeline_logging.{thriftscala => thriftlog} + +object PromotedTweetEntryMarshaller { + + def apply(entry: TweetItem, position: Int): thriftlog.PromotedTweetEntry = { + thriftlog.PromotedTweetEntry( + id = entry.id, + advertiserId = entry.promotedMetadata.map(_.advertiserId).getOrElse(0L), + insertPosition = position, + impressionId = entry.promotedMetadata.flatMap(_.impressionString), + displayType = Some(entry.displayType.toString) + ) + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/marshaller/timeline_logging/TweetEntryMarshaller.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/marshaller/timeline_logging/TweetEntryMarshaller.scala new file mode 100644 index 0000000000..ca6b08778b --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/marshaller/timeline_logging/TweetEntryMarshaller.scala @@ -0,0 +1,29 @@ +package com.twitter.home_mixer.marshaller.timeline_logging + +import com.twitter.home_mixer.model.HomeFeatures.ScoreFeature +import com.twitter.home_mixer.model.HomeFeatures.SocialContextFeature +import com.twitter.home_mixer.model.HomeFeatures.SourceTweetIdFeature +import com.twitter.product_mixer.core.model.common.presentation.CandidateWithDetails +import com.twitter.product_mixer.core.model.marshalling.response.urt.item.tweet.TweetItem +import com.twitter.timelines.service.{thriftscala => tst} +import com.twitter.timelines.timeline_logging.{thriftscala => thriftlog} + +object TweetEntryMarshaller { + + def apply(entry: TweetItem, candidate: CandidateWithDetails): thriftlog.TweetEntry = { + val socialContextType = candidate.features.getOrElse(SocialContextFeature, None) match { + case Some(tst.SocialContext.GeneralContext(tst.GeneralContext(contextType, _, _, _, _))) => + Some(contextType.value.toShort) + case Some(tst.SocialContext.TopicContext(_)) => + Some(tst.ContextType.Topic.value.toShort) + case _ => None + } + thriftlog.TweetEntry( + id = candidate.candidateIdLong, + sourceTweetId = candidate.features.getOrElse(SourceTweetIdFeature, None), + displayType = Some(entry.displayType.toString), + score = candidate.features.getOrElse(ScoreFeature, None), + socialContextType = socialContextType + ) + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/marshaller/timeline_logging/WhoToFollowEntryMarshaller.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/marshaller/timeline_logging/WhoToFollowEntryMarshaller.scala new file mode 100644 index 0000000000..9a253b7263 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/marshaller/timeline_logging/WhoToFollowEntryMarshaller.scala @@ -0,0 +1,19 @@ +package com.twitter.home_mixer.marshaller.timeline_logging + +import com.twitter.product_mixer.component_library.pipeline.candidate.who_to_follow_module.ScoreFeature +import com.twitter.product_mixer.core.model.common.presentation.ItemCandidateWithDetails +import com.twitter.product_mixer.core.model.marshalling.response.urt.item.user.UserItem +import com.twitter.timelines.timeline_logging.{thriftscala => thriftlog} + +object WhoToFollowEntryMarshaller { + + def apply(entry: UserItem, candidate: ItemCandidateWithDetails): thriftlog.WhoToFollowEntry = + thriftlog.WhoToFollowEntry( + userId = entry.id, + displayType = Some(entry.displayType.toString), + score = candidate.features.getOrElse(ScoreFeature, None), + enableReactiveBlending = entry.enableReactiveBlending, + impressionId = entry.promotedMetadata.flatMap(_.impressionString), + advertiserId = entry.promotedMetadata.map(_.advertiserId) + ) +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/marshaller/timelines/BUILD.bazel b/home-mixer/server/src/main/scala/com/twitter/home_mixer/marshaller/timelines/BUILD.bazel new file mode 100644 index 0000000000..e424298f0b --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/marshaller/timelines/BUILD.bazel @@ -0,0 +1,16 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + strict_deps = True, + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/javax/inject:javax.inject", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/model/request", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/cursor", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/request", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt", + "src/thrift/com/twitter/timelines/render:thrift-scala", + "src/thrift/com/twitter/timelineservice:thrift-scala", + "timelineservice/common:model", + ], +) diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/marshaller/timelines/ChronologicalCursorMarshaller.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/marshaller/timelines/ChronologicalCursorMarshaller.scala new file mode 100644 index 0000000000..40e25a8abd --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/marshaller/timelines/ChronologicalCursorMarshaller.scala @@ -0,0 +1,20 @@ +package com.twitter.home_mixer.marshaller.timelines + +import com.twitter.product_mixer.component_library.model.cursor.UrtOrderedCursor +import com.twitter.product_mixer.core.model.marshalling.response.urt.operation.BottomCursor +import com.twitter.product_mixer.core.model.marshalling.response.urt.operation.GapCursor +import com.twitter.product_mixer.core.model.marshalling.response.urt.operation.TopCursor +import com.twitter.timelines.service.{thriftscala => t} + +object ChronologicalCursorMarshaller { + + def apply(cursor: UrtOrderedCursor): Option[t.ChronologicalCursor] = { + cursor.cursorType match { + case Some(TopCursor) => Some(t.ChronologicalCursor(bottom = cursor.id)) + case Some(BottomCursor) => Some(t.ChronologicalCursor(top = cursor.id)) + case Some(GapCursor) => + Some(t.ChronologicalCursor(top = cursor.id, bottom = cursor.gapBoundaryId)) + case _ => None + } + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/marshaller/timelines/ChronologicalCursorUnmarshaller.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/marshaller/timelines/ChronologicalCursorUnmarshaller.scala new file mode 100644 index 0000000000..739c490ea5 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/marshaller/timelines/ChronologicalCursorUnmarshaller.scala @@ -0,0 +1,26 @@ +package com.twitter.home_mixer.marshaller.timelines + +import com.twitter.product_mixer.component_library.model.cursor.UrtOrderedCursor +import com.twitter.product_mixer.core.model.marshalling.response.urt.operation.BottomCursor +import com.twitter.product_mixer.core.model.marshalling.response.urt.operation.GapCursor +import com.twitter.product_mixer.core.model.marshalling.response.urt.operation.TopCursor +import com.twitter.timelines.service.{thriftscala => t} + +object ChronologicalCursorUnmarshaller { + + def apply(requestCursor: t.RequestCursor): Option[UrtOrderedCursor] = { + requestCursor match { + case t.RequestCursor.ChronologicalCursor(cursor) => + (cursor.top, cursor.bottom) match { + case (Some(top), None) => + Some(UrtOrderedCursor(top, cursor.top, Some(BottomCursor))) + case (None, Some(bottom)) => + Some(UrtOrderedCursor(bottom, cursor.bottom, Some(TopCursor))) + case (Some(top), Some(bottom)) => + Some(UrtOrderedCursor(top, cursor.top, Some(GapCursor), cursor.bottom)) + case _ => None + } + case _ => None + } + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/marshaller/timelines/DeviceContextMarshaller.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/marshaller/timelines/DeviceContextMarshaller.scala new file mode 100644 index 0000000000..097d8dea48 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/marshaller/timelines/DeviceContextMarshaller.scala @@ -0,0 +1,34 @@ +package com.twitter.home_mixer.marshaller.timelines + +import com.twitter.home_mixer.model.request.DeviceContext +import com.twitter.product_mixer.core.model.marshalling.request.ClientContext +import com.twitter.timelineservice.{thriftscala => t} +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class DeviceContextMarshaller @Inject() () { + + def apply(deviceContext: DeviceContext, clientContext: ClientContext): t.DeviceContext = { + t.DeviceContext( + countryCode = clientContext.countryCode, + languageCode = clientContext.languageCode, + clientAppId = clientContext.appId, + ipAddress = clientContext.ipAddress, + guestId = clientContext.guestId, + userAgent = clientContext.userAgent, + deviceId = clientContext.deviceId, + isPolling = deviceContext.isPolling, + requestContext = deviceContext.requestContext, + referrer = None, + tfeAuthHeader = None, + mobileDeviceId = clientContext.mobileDeviceId, + isSessionStart = None, + latestControlAvailable = deviceContext.latestControlAvailable, + guestIdMarketing = clientContext.guestIdMarketing, + isInternalOrTwoffice = clientContext.isTwoffice, + guestIdAds = clientContext.guestIdAds, + isUrtRequest = Some(true) + ) + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/marshaller/timelines/RecommendedUsersCursorUnmarshaller.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/marshaller/timelines/RecommendedUsersCursorUnmarshaller.scala new file mode 100644 index 0000000000..693684558a --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/marshaller/timelines/RecommendedUsersCursorUnmarshaller.scala @@ -0,0 +1,20 @@ +package com.twitter.home_mixer.marshaller.timelines + +import com.twitter.product_mixer.component_library.model.cursor.UrtUnorderedExcludeIdsCursor +import com.twitter.timelines.service.{thriftscala => t} +import com.twitter.util.Time + +object RecommendedUsersCursorUnmarshaller { + + def apply(requestCursor: t.RequestCursor): Option[UrtUnorderedExcludeIdsCursor] = { + requestCursor match { + case t.RequestCursor.RecommendedUsersCursor(cursor) => + Some( + UrtUnorderedExcludeIdsCursor( + initialSortIndex = cursor.minSortIndex.getOrElse(Time.now.inMilliseconds), + excludedIds = cursor.previouslyRecommendedUserIds + )) + case _ => None + } + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/marshaller/timelines/TimelineServiceCursorMarshaller.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/marshaller/timelines/TimelineServiceCursorMarshaller.scala new file mode 100644 index 0000000000..cbc956ad42 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/marshaller/timelines/TimelineServiceCursorMarshaller.scala @@ -0,0 +1,21 @@ +package com.twitter.home_mixer.marshaller.timelines + +import com.twitter.product_mixer.component_library.model.cursor.UrtOrderedCursor +import com.twitter.product_mixer.core.model.marshalling.response.urt.operation.BottomCursor +import com.twitter.product_mixer.core.model.marshalling.response.urt.operation.GapCursor +import com.twitter.product_mixer.core.model.marshalling.response.urt.operation.TopCursor +import com.twitter.timelineservice.{thriftscala => t} + +object TimelineServiceCursorMarshaller { + + def apply(cursor: UrtOrderedCursor): Option[t.Cursor2] = { + val id = cursor.id.map(_.toString) + val gapBoundaryId = cursor.gapBoundaryId.map(_.toString) + cursor.cursorType match { + case Some(TopCursor) => Some(t.Cursor2(bottom = id)) + case Some(BottomCursor) => Some(t.Cursor2(top = id)) + case Some(GapCursor) => Some(t.Cursor2(top = id, bottom = gapBoundaryId)) + case _ => None + } + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/marshaller/timelines/TopicContextFunctionalityTypeUnmarshaller.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/marshaller/timelines/TopicContextFunctionalityTypeUnmarshaller.scala new file mode 100644 index 0000000000..f542cd5358 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/marshaller/timelines/TopicContextFunctionalityTypeUnmarshaller.scala @@ -0,0 +1,22 @@ +package com.twitter.home_mixer.marshaller.timelines + +import com.twitter.product_mixer.core.model.marshalling.response.urt.metadata.BasicTopicContextFunctionalityType +import com.twitter.product_mixer.core.model.marshalling.response.urt.metadata.RecWithEducationTopicContextFunctionalityType +import com.twitter.product_mixer.core.model.marshalling.response.urt.metadata.RecommendationTopicContextFunctionalityType +import com.twitter.product_mixer.core.model.marshalling.response.urt.metadata.TopicContextFunctionalityType +import com.twitter.timelines.render.{thriftscala => urt} + +object TopicContextFunctionalityTypeUnmarshaller { + + def apply( + topicContextFunctionalityType: urt.TopicContextFunctionalityType + ): TopicContextFunctionalityType = topicContextFunctionalityType match { + case urt.TopicContextFunctionalityType.Basic => BasicTopicContextFunctionalityType + case urt.TopicContextFunctionalityType.Recommendation => + RecommendationTopicContextFunctionalityType + case urt.TopicContextFunctionalityType.RecWithEducation => + RecWithEducationTopicContextFunctionalityType + case urt.TopicContextFunctionalityType.EnumUnknownTopicContextFunctionalityType(field) => + throw new UnsupportedOperationException(s"Unknown topic context functionality type: $field") + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/model/BUILD.bazel b/home-mixer/server/src/main/scala/com/twitter/home_mixer/model/BUILD.bazel new file mode 100644 index 0000000000..c5c46d2d12 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/model/BUILD.bazel @@ -0,0 +1,49 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + strict_deps = True, + tags = ["bazel-compatible"], + dependencies = [ + "home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/candidate_source", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/model/request", + "home-mixer/thrift/src/main/thrift:thrift-scala", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/candidate", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/query/ads", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/premarshaller/urt/builder", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/feature/datarecord", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline", + "src/java/com/twitter/ml/api:api-base", + "src/java/com/twitter/ml/api/constant", + "src/scala/com/twitter/ml/api:api-base", + "src/scala/com/twitter/timelines/prediction/features/common", + "src/scala/com/twitter/timelines/prediction/features/recap", + "src/scala/com/twitter/timelines/prediction/features/request_context", + "src/thrift/com/twitter/dal/personal_data:personal_data-java", + "src/thrift/com/twitter/escherbird:tweet-annotation-scala", + "src/thrift/com/twitter/gizmoduck:thrift-scala", + "src/thrift/com/twitter/timelines/author_features:thrift-java", + "src/thrift/com/twitter/timelines/conversation_features:conversation_features-scala", + "src/thrift/com/twitter/timelines/impression:thrift-scala", + "src/thrift/com/twitter/timelines/impression_bloom_filter:thrift-scala", + "src/thrift/com/twitter/timelinescorer/common/scoredtweetcandidate:thrift-scala", + "src/thrift/com/twitter/timelineservice/server/suggests/logging:thrift-scala", + "src/thrift/com/twitter/tweetypie:media-entity-scala", + "src/thrift/com/twitter/tweetypie:tweet-scala", + "src/thrift/com/twitter/user_session_store:thrift-java", + "timelinemixer/common/src/main/scala/com/twitter/timelinemixer/clients/manhattan", + "timelinemixer/common/src/main/scala/com/twitter/timelinemixer/clients/persistence", + "timelinemixer/server/src/main/scala/com/twitter/timelinemixer/injection/model/candidate", + "timelines/src/main/scala/com/twitter/timelines/model/types", + "topic-social-proof/server/src/main/thrift:thrift-scala", + "tweetconvosvc/common/src/main/thrift/com/twitter/tweetconvosvc/tweet_ancestor:thrift-scala", + ], + exports = [ + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/cursor", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/query/ads", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/feature/datarecord", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline", + "src/thrift/com/twitter/timelines/impression:thrift-scala", + "src/thrift/com/twitter/timelinescorer/common/scoredtweetcandidate:thrift-scala", + "tweetconvosvc/common/src/main/thrift/com/twitter/tweetconvosvc/tweet_ancestor:thrift-scala", + ], +) diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/model/ClearCacheIncludeInstruction.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/model/ClearCacheIncludeInstruction.scala new file mode 100644 index 0000000000..85154e55bb --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/model/ClearCacheIncludeInstruction.scala @@ -0,0 +1,43 @@ +package com.twitter.home_mixer.model + +import com.twitter.home_mixer.model.request.DeviceContext.RequestContext +import com.twitter.home_mixer.model.request.HasDeviceContext +import com.twitter.product_mixer.component_library.premarshaller.urt.builder.IncludeInstruction +import com.twitter.product_mixer.core.model.marshalling.response.urt.TimelineEntry +import com.twitter.product_mixer.core.model.marshalling.response.urt.TimelineModule +import com.twitter.product_mixer.core.model.marshalling.response.urt.item.tweet.TweetItem +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.timelines.configapi.FSBoundedParam +import com.twitter.timelines.configapi.FSParam + +/** + * Include a clear cache timeline instruction when we satisfy these criteria: + * - Request Provenance is "pull to refresh" + * - Atleast N non-ad tweet entries in the response + * + * This is to ensure that we have sufficient new content to justify jumping users to the + * top of the new timelines response and don't add unnecessary load to backend systems + */ +case class ClearCacheIncludeInstruction( + enableParam: FSParam[Boolean], + minEntriesParam: FSBoundedParam[Int]) + extends IncludeInstruction[PipelineQuery with HasDeviceContext] { + + override def apply( + query: PipelineQuery with HasDeviceContext, + entries: Seq[TimelineEntry] + ): Boolean = { + val enabled = query.params(enableParam) + + val ptr = + query.deviceContext.flatMap(_.requestContextValue).contains(RequestContext.PullToRefresh) + + val minTweets = query.params(minEntriesParam) <= entries.collect { + case item: TweetItem if item.promotedMetadata.isEmpty => 1 + case module: TimelineModule if module.items.head.item.isInstanceOf[TweetItem] => + module.items.size + }.sum + + enabled && ptr && minTweets + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/model/ContentFeatures.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/model/ContentFeatures.scala new file mode 100644 index 0000000000..44e8cf25e5 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/model/ContentFeatures.scala @@ -0,0 +1,96 @@ +package com.twitter.home_mixer.model + +import com.twitter.escherbird.{thriftscala => esb} +import com.twitter.tweetypie.{thriftscala => tp} + +object ContentFeatures { + val Empty: ContentFeatures = ContentFeatures( + 0.toShort, + false, + 0.toShort, + 0.toShort, + None, + None, + None, + None, + None, + None, + None, + None, + None, + None, + None, + None, + None, + None, + None, + None, + None, + None, + None, + None, + None, + None, + None, + None, + None, + None, + None, + None, + None, + None, + None, + None, + None, + None, + None, + None, + None, + None + ) +} + +case class ContentFeatures( + length: Short, + hasQuestion: Boolean, + numCaps: Short, + numWhiteSpaces: Short, + numNewlines: Option[Short], + videoDurationMs: Option[Int], + bitRate: Option[Int], + aspectRatioNum: Option[Short], + aspectRatioDen: Option[Short], + widths: Option[Seq[Short]], + heights: Option[Seq[Short]], + resizeMethods: Option[Seq[Short]], + numMediaTags: Option[Short], + mediaTagScreenNames: Option[Seq[String]], + emojiTokens: Option[Set[String]], + emoticonTokens: Option[Set[String]], + faceAreas: Option[Seq[Int]], + dominantColorRed: Option[Short], + dominantColorBlue: Option[Short], + dominantColorGreen: Option[Short], + numColors: Option[Short], + stickerIds: Option[Seq[Long]], + mediaOriginProviders: Option[Seq[String]], + isManaged: Option[Boolean], + is360: Option[Boolean], + viewCount: Option[Long], + isMonetizable: Option[Boolean], + isEmbeddable: Option[Boolean], + hasSelectedPreviewImage: Option[Boolean], + hasTitle: Option[Boolean], + hasDescription: Option[Boolean], + hasVisitSiteCallToAction: Option[Boolean], + hasAppInstallCallToAction: Option[Boolean], + hasWatchNowCallToAction: Option[Boolean], + media: Option[Seq[tp.MediaEntity]], + dominantColorPercentage: Option[Double], + posUnigrams: Option[Set[String]], + posBigrams: Option[Set[String]], + semanticCoreAnnotations: Option[Seq[esb.TweetEntityAnnotation]], + selfThreadMetadata: Option[tp.SelfThreadMetadata], + tokens: Option[Seq[String]], + conversationControl: Option[tp.ConversationControl], +) diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/model/GapIncludeInstruction.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/model/GapIncludeInstruction.scala new file mode 100644 index 0000000000..c11631fc89 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/model/GapIncludeInstruction.scala @@ -0,0 +1,63 @@ +package com.twitter.home_mixer.model + +import com.twitter.home_mixer.functional_component.candidate_source.EarlybirdBottomTweetFeature +import com.twitter.home_mixer.functional_component.candidate_source.EarlybirdResponseTruncatedFeature +import com.twitter.product_mixer.component_library.model.cursor.UrtOrderedCursor +import com.twitter.product_mixer.component_library.premarshaller.urt.builder.IncludeInstruction +import com.twitter.product_mixer.core.model.marshalling.response.urt.TimelineEntry +import com.twitter.product_mixer.core.model.marshalling.response.urt.TimelineModule +import com.twitter.product_mixer.core.model.marshalling.response.urt.item.tweet.TweetItem +import com.twitter.product_mixer.core.model.marshalling.response.urt.operation.GapCursor +import com.twitter.product_mixer.core.model.marshalling.response.urt.operation.TopCursor +import com.twitter.product_mixer.core.pipeline.HasPipelineCursor +import com.twitter.product_mixer.core.pipeline.PipelineQuery + +/** + * Determine whether to include a Gap Cursor in the response based on whether a timeline + * is truncated because it has more entries than the max response size. + * There are two ways this can happen: + * 1) There are unused entries in Earlybird. This is determined by a flag returned from Earlybird. + * We respect the Earlybird flag only if there are some entries after deduping and filtering + * to ensure that we do not get stuck repeatedly serving gaps which lead to no tweets. + * 2) Ads injection can take the response size over the max count. Goldfinch truncates tweet + * entries in this case. We can check if the bottom tweet from Earlybird is in the response to + * determine if all Earlybird tweets have been used. + * + * While scrolling down to get older tweets (BottomCursor), responses will generally be + * truncated, but we don't want to render a gap cursor there, so we need to ensure we only + * apply the truncation check to newer (TopCursor) or middle (GapCursor) requests. + * + * We return either a Gap Cursor or a Bottom Cursor, but not both, so the include instruction + * for Bottom should be the inverse of Gap. + */ +object GapIncludeInstruction + extends IncludeInstruction[PipelineQuery with HasPipelineCursor[UrtOrderedCursor]] { + + override def apply( + query: PipelineQuery with HasPipelineCursor[UrtOrderedCursor], + entries: Seq[TimelineEntry] + ): Boolean = { + val wasTruncated = query.features.exists(_.getOrElse(EarlybirdResponseTruncatedFeature, false)) + + // Get oldest tweet or tweets within oldest conversation module + val tweetEntries = entries.view.reverse + .collectFirst { + case item: TweetItem if item.promotedMetadata.isEmpty => Seq(item.id.toString) + case module: TimelineModule if module.items.head.item.isInstanceOf[TweetItem] => + module.items.map(_.item.id.toString) + }.toSeq.flatten + + val bottomCursor = + query.features.flatMap(_.getOrElse(EarlybirdBottomTweetFeature, None)).map(_.toString) + + // Ads truncation happened if we have at least max count entries and bottom tweet is missing + val adsTruncation = query.requestedMaxResults.exists(_ <= entries.size) && + !bottomCursor.exists(tweetEntries.contains) + + query.pipelineCursor.exists(_.cursorType match { + case Some(TopCursor) | Some(GapCursor) => + (wasTruncated && tweetEntries.nonEmpty) || adsTruncation + case _ => false + }) + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/model/HomeAdsQuery.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/model/HomeAdsQuery.scala new file mode 100644 index 0000000000..bfbe722fcb --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/model/HomeAdsQuery.scala @@ -0,0 +1,42 @@ +package com.twitter.home_mixer.model + +import com.twitter.adserver.thriftscala.RequestTriggerType +import com.twitter.home_mixer.model.HomeFeatures.GetInitialFeature +import com.twitter.home_mixer.model.HomeFeatures.GetNewerFeature +import com.twitter.home_mixer.model.HomeFeatures.GetOlderFeature +import com.twitter.home_mixer.model.HomeFeatures.PollingFeature +import com.twitter.home_mixer.model.request.HasDeviceContext +import com.twitter.product_mixer.component_library.model.cursor.UrtOrderedCursor +import com.twitter.product_mixer.component_library.model.query.ads.AdsQuery +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.pipeline.HasPipelineCursor +import com.twitter.product_mixer.core.pipeline.PipelineQuery + +/** + * These are for feeds needed for ads only. + */ +trait HomeAdsQuery + extends AdsQuery + with PipelineQuery + with HasDeviceContext + with HasPipelineCursor[UrtOrderedCursor] { + + private val featureToRequestTriggerType = Seq( + (GetInitialFeature, RequestTriggerType.Initial), + (GetNewerFeature, RequestTriggerType.Scroll), + (GetOlderFeature, RequestTriggerType.Scroll), + (PollingFeature, RequestTriggerType.AutoRefresh) + ) + + override val autoplayEnabled: Option[Boolean] = deviceContext.flatMap(_.autoplayEnabled) + + override def requestTriggerType: Option[RequestTriggerType] = { + val features = this.features.getOrElse(FeatureMap.empty) + + featureToRequestTriggerType.collectFirst { + case (feature, requestType) if features.get(feature) => Some(requestType) + }.flatten + } + + override val disableNsfwAvoidance: Option[Boolean] = Some(true) +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/model/HomeFeatures.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/model/HomeFeatures.scala new file mode 100644 index 0000000000..0a19230d35 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/model/HomeFeatures.scala @@ -0,0 +1,266 @@ +package com.twitter.home_mixer.model + +import com.twitter.core_workflows.user_model.{thriftscala => um} +import com.twitter.dal.personal_data.{thriftjava => pd} +import com.twitter.escherbird.{thriftscala => esb} +import com.twitter.gizmoduck.{thriftscala => gt} +import com.twitter.home_mixer.{thriftscala => hmt} +import com.twitter.ml.api.constant.SharedFeatures +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.feature.Feature +import com.twitter.product_mixer.core.feature.FeatureWithDefaultOnFailure +import com.twitter.product_mixer.core.feature.datarecord.BoolDataRecordCompatible +import com.twitter.product_mixer.core.feature.datarecord.DataRecordFeature +import com.twitter.product_mixer.core.feature.datarecord.LongDiscreteDataRecordCompatible +import com.twitter.product_mixer.core.model.marshalling.response.urt.metadata.TopicContextFunctionalityType +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.search.common.features.{thriftscala => sc} +import com.twitter.timelinemixer.clients.manhattan.DismissInfo +import com.twitter.timelinemixer.clients.persistence.TimelineResponseV3 +import com.twitter.timelinemixer.injection.model.candidate.AudioSpaceMetaData +import com.twitter.timelines.conversation_features.v1.thriftscala.ConversationFeatures +import com.twitter.timelines.impression.{thriftscala => imp} +import com.twitter.timelines.impressionbloomfilter.{thriftscala => blm} +import com.twitter.timelines.model.UserId +import com.twitter.timelines.prediction.features.common.TimelinesSharedFeatures +import com.twitter.timelines.prediction.features.recap.RecapFeatures +import com.twitter.timelines.prediction.features.request_context.RequestContextFeatures +import com.twitter.timelines.service.{thriftscala => tst} +import com.twitter.timelineservice.model.FeedbackEntry +import com.twitter.timelineservice.suggests.logging.candidate_tweet_source_id.{thriftscala => cts} +import com.twitter.timelineservice.suggests.{thriftscala => st} +import com.twitter.tsp.{thriftscala => tsp} +import com.twitter.tweetconvosvc.tweet_ancestor.{thriftscala => ta} +import com.twitter.util.Time + +object HomeFeatures { + // Candidate Features + object AncestorsFeature extends Feature[TweetCandidate, Seq[ta.TweetAncestor]] + object AudioSpaceMetaDataFeature extends Feature[TweetCandidate, Option[AudioSpaceMetaData]] + object TwitterListIdFeature extends Feature[TweetCandidate, Option[Long]] + + /** + * For Retweets, this should refer to the retweeting user. Use [[SourceUserIdFeature]] if you want to know + * who created the Tweet that was retweeted. + */ + object AuthorIdFeature + extends Feature[TweetCandidate, Option[Long]] + with LongDiscreteDataRecordCompatible { + override val featureName: String = SharedFeatures.AUTHOR_ID.getFeatureName + override val personalDataTypes: Set[pd.PersonalDataType] = Set(pd.PersonalDataType.UserId) + } + object AuthorIsEligibleForConnectBoostFeature extends Feature[TweetCandidate, Boolean] + object AuthorIsBlueVerifiedFeature extends Feature[TweetCandidate, Boolean] + object AuthoredByContextualUserFeature extends Feature[TweetCandidate, Boolean] + object CachedCandidatePipelineIdentifierFeature extends Feature[TweetCandidate, Option[String]] + object CandidateSourceIdFeature + extends Feature[TweetCandidate, Option[cts.CandidateTweetSourceId]] + object ConversationFeature extends Feature[TweetCandidate, Option[ConversationFeatures]] + + /** + * This field should be set to the focal Tweet's tweetId for all tweets which are expected to + * be rendered in the same convo module. For non-convo module Tweets, this will be + * set to None. Note this is different from how TweetyPie defines ConversationId which is defined + * on all Tweets and points to the root tweet. This feature is used for grouping convo modules together. + */ + object ConversationModuleFocalTweetIdFeature extends Feature[TweetCandidate, Option[Long]] + + /** + * This field should always be set to the root Tweet in a conversation for all Tweets. For replies, this will + * point back to the root Tweet. For non-replies, this will be the candidate's Tweet id. This is consistent with + * the TweetyPie definition of ConversationModuleId. + */ + object ConversationModuleIdFeature extends Feature[TweetCandidate, Option[Long]] + object DirectedAtUserIdFeature extends Feature[TweetCandidate, Option[Long]] + object EarlybirdFeature extends Feature[TweetCandidate, Option[sc.ThriftTweetFeatures]] + object EarlybirdScoreFeature extends Feature[TweetCandidate, Option[Double]] + object EntityTokenFeature extends Feature[TweetCandidate, Option[String]] + object ExclusiveConversationAuthorIdFeature extends Feature[TweetCandidate, Option[Long]] + object FavoritedByUserIdsFeature extends Feature[TweetCandidate, Seq[Long]] + object FeedbackHistoryFeature extends Feature[TweetCandidate, Seq[FeedbackEntry]] + object RetweetedByEngagerIdsFeature extends Feature[TweetCandidate, Seq[Long]] + object RepliedByEngagerIdsFeature extends Feature[TweetCandidate, Seq[Long]] + object FollowedByUserIdsFeature extends Feature[TweetCandidate, Seq[Long]] + + object TopicIdSocialContextFeature extends Feature[TweetCandidate, Option[Long]] + object TopicContextFunctionalityTypeFeature + extends Feature[TweetCandidate, Option[TopicContextFunctionalityType]] + object FromInNetworkSourceFeature extends Feature[TweetCandidate, Boolean] + + object FullScoringSucceededFeature extends Feature[TweetCandidate, Boolean] + object HasDisplayedTextFeature extends Feature[TweetCandidate, Boolean] + object InReplyToTweetIdFeature extends Feature[TweetCandidate, Option[Long]] + object InReplyToUserIdFeature extends Feature[TweetCandidate, Option[Long]] + object IsAncestorCandidateFeature extends Feature[TweetCandidate, Boolean] + object IsExtendedReplyFeature + extends DataRecordFeature[TweetCandidate, Boolean] + with BoolDataRecordCompatible { + override val featureName: String = RecapFeatures.IS_EXTENDED_REPLY.getFeatureName + override val personalDataTypes: Set[pd.PersonalDataType] = Set.empty + } + object IsRandomTweetFeature + extends Feature[TweetCandidate, Boolean] + with BoolDataRecordCompatible { + override val featureName: String = TimelinesSharedFeatures.IS_RANDOM_TWEET.getFeatureName + override val personalDataTypes: Set[pd.PersonalDataType] = Set.empty + } + object IsReadFromCacheFeature extends Feature[TweetCandidate, Boolean] + object IsRetweetFeature extends Feature[TweetCandidate, Boolean] + object IsRetweetedReplyFeature extends Feature[TweetCandidate, Boolean] + object LastScoredTimestampMsFeature extends Feature[TweetCandidate, Option[Long]] + object NonSelfFavoritedByUserIdsFeature extends Feature[TweetCandidate, Seq[Long]] + object NumImagesFeature extends Feature[TweetCandidate, Option[Int]] + object OriginalTweetCreationTimeFromSnowflakeFeature extends Feature[TweetCandidate, Option[Time]] + object PositionFeature extends Feature[TweetCandidate, Option[Int]] + // Internal id generated per prediction service request + object PredictionRequestIdFeature extends Feature[TweetCandidate, Option[Long]] + object QuotedTweetIdFeature extends Feature[TweetCandidate, Option[Long]] + object QuotedUserIdFeature extends Feature[TweetCandidate, Option[Long]] + object ScoreFeature extends Feature[TweetCandidate, Option[Double]] + object SemanticCoreIdFeature extends Feature[TweetCandidate, Option[Long]] + // Key for kafka logging + object ServedIdFeature extends Feature[TweetCandidate, Option[Long]] + object SimclustersTweetTopKClustersWithScoresFeature + extends Feature[TweetCandidate, Map[String, Double]] + object SocialContextFeature extends Feature[TweetCandidate, Option[tst.SocialContext]] + object SourceTweetIdFeature + extends Feature[TweetCandidate, Option[Long]] + with LongDiscreteDataRecordCompatible { + override val featureName: String = TimelinesSharedFeatures.SOURCE_TWEET_ID.getFeatureName + override val personalDataTypes: Set[pd.PersonalDataType] = Set(pd.PersonalDataType.TweetId) + } + object SourceUserIdFeature extends Feature[TweetCandidate, Option[Long]] + object StreamToKafkaFeature extends Feature[TweetCandidate, Boolean] + object SuggestTypeFeature extends Feature[TweetCandidate, Option[st.SuggestType]] + object TSPMetricTagFeature extends Feature[TweetCandidate, Set[tsp.MetricTag]] + object TweetLanguageFeature extends Feature[TweetCandidate, Option[String]] + object TweetUrlsFeature extends Feature[TweetCandidate, Seq[String]] + object VideoDurationMsFeature extends Feature[TweetCandidate, Option[Int]] + object ViewerIdFeature + extends Feature[TweetCandidate, Long] + with LongDiscreteDataRecordCompatible { + override def featureName: String = SharedFeatures.USER_ID.getFeatureName + override def personalDataTypes: Set[pd.PersonalDataType] = Set(pd.PersonalDataType.UserId) + } + object WeightedModelScoreFeature extends Feature[TweetCandidate, Option[Double]] + object MentionUserIdFeature extends Feature[TweetCandidate, Seq[Long]] + object MentionScreenNameFeature extends Feature[TweetCandidate, Seq[String]] + object SemanticAnnotationFeature extends Feature[TweetCandidate, Seq[esb.TweetEntityAnnotation]] + object HasImageFeature extends Feature[TweetCandidate, Boolean] + object HasVideoFeature extends Feature[TweetCandidate, Boolean] + + // Tweetypie VF Features + object IsHydratedFeature extends FeatureWithDefaultOnFailure[TweetCandidate, Boolean] { + override val defaultValue: Boolean = true + } + object IsNsfwFeature extends Feature[TweetCandidate, Boolean] + object QuotedTweetDroppedFeature extends Feature[TweetCandidate, Boolean] + // Raw Tweet Text from Tweetypie + object TweetTextFeature extends Feature[TweetCandidate, Option[String]] + + // SGS Features + /** + * By convention, this is set to true for retweets of non-followed authors + * E.g. where somebody the viewer follows retweets a Tweet from somebody the viewer doesn't follow + */ + object InNetworkFeature extends FeatureWithDefaultOnFailure[TweetCandidate, Boolean] { + override val defaultValue: Boolean = true + } + + // Query Features + object AccountAgeFeature extends Feature[PipelineQuery, Option[Time]] + object ClientIdFeature + extends Feature[PipelineQuery, Option[Long]] + with LongDiscreteDataRecordCompatible { + override def featureName: String = SharedFeatures.CLIENT_ID.getFeatureName + override def personalDataTypes: Set[pd.PersonalDataType] = Set(pd.PersonalDataType.ClientType) + } + object CachedScoredTweetsFeature extends Feature[PipelineQuery, Seq[hmt.CachedScoredTweet]] + object DDGStatsElonFeature extends Feature[PipelineQuery, Long] + object DDGStatsVitsFeature extends Feature[PipelineQuery, Set[Long]] + object DDGStatsDemocratsFeature extends Feature[PipelineQuery, Set[Long]] + object DDGStatsRepublicansFeature extends Feature[PipelineQuery, Set[Long]] + object DeviceLanguageFeature extends Feature[PipelineQuery, Option[String]] + object DismissInfoFeature + extends FeatureWithDefaultOnFailure[PipelineQuery, Map[st.SuggestType, Option[DismissInfo]]] { + override def defaultValue: Map[st.SuggestType, Option[DismissInfo]] = Map.empty + } + object FollowingLastNonPollingTimeFeature extends Feature[PipelineQuery, Option[Time]] + object GetInitialFeature extends Feature[PipelineQuery, Boolean] with BoolDataRecordCompatible { + override def featureName: String = RequestContextFeatures.IS_GET_INITIAL.getFeatureName + override def personalDataTypes: Set[pd.PersonalDataType] = Set.empty + } + object GetMiddleFeature extends Feature[PipelineQuery, Boolean] with BoolDataRecordCompatible { + override def featureName: String = RequestContextFeatures.IS_GET_MIDDLE.getFeatureName + override def personalDataTypes: Set[pd.PersonalDataType] = Set.empty + } + object GetNewerFeature extends Feature[PipelineQuery, Boolean] with BoolDataRecordCompatible { + override def featureName: String = RequestContextFeatures.IS_GET_NEWER.getFeatureName + override def personalDataTypes: Set[pd.PersonalDataType] = Set.empty + } + object GetOlderFeature extends Feature[PipelineQuery, Boolean] with BoolDataRecordCompatible { + override def featureName: String = RequestContextFeatures.IS_GET_OLDER.getFeatureName + override def personalDataTypes: Set[pd.PersonalDataType] = Set.empty + } + object GuestIdFeature + extends Feature[PipelineQuery, Option[Long]] + with LongDiscreteDataRecordCompatible { + override def featureName: String = SharedFeatures.GUEST_ID.getFeatureName + override def personalDataTypes: Set[pd.PersonalDataType] = Set(pd.PersonalDataType.GuestId) + } + object HasDarkRequestFeature extends Feature[TweetCandidate, Option[Boolean]] + object ImpressionBloomFilterFeature + extends FeatureWithDefaultOnFailure[PipelineQuery, blm.ImpressionBloomFilterSeq] { + override def defaultValue: blm.ImpressionBloomFilterSeq = + blm.ImpressionBloomFilterSeq(Seq.empty) + } + object IsForegroundRequestFeature extends Feature[PipelineQuery, Boolean] + object IsLaunchRequestFeature extends Feature[PipelineQuery, Boolean] + object LastNonPollingTimeFeature extends Feature[PipelineQuery, Option[Time]] + object NonPollingTimesFeature extends Feature[PipelineQuery, Seq[Long]] + object PersistenceEntriesFeature extends Feature[PipelineQuery, Seq[TimelineResponseV3]] + object PollingFeature extends Feature[PipelineQuery, Boolean] + object PullToRefreshFeature extends Feature[PipelineQuery, Boolean] + // Scores from Real Graph representing the relationship between the viewer and another user + object RealGraphInNetworkScoresFeature extends Feature[PipelineQuery, Map[UserId, Double]] + object RequestJoinIdFeature extends Feature[TweetCandidate, Option[Long]] + // Internal id generated per request, mainly to deduplicate re-served cached tweets in logging + object ServedRequestIdFeature extends Feature[PipelineQuery, Option[Long]] + object ServedTweetIdsFeature extends Feature[PipelineQuery, Seq[Long]] + object TweetImpressionsFeature extends Feature[PipelineQuery, Seq[imp.TweetImpressionsEntry]] + object UserFollowedTopicsCountFeature extends Feature[PipelineQuery, Option[Int]] + object UserFollowingCountFeature extends Feature[PipelineQuery, Option[Int]] + object UserScreenNameFeature extends Feature[PipelineQuery, Option[String]] + object UserStateFeature extends Feature[PipelineQuery, Option[um.UserState]] + object UserTypeFeature extends Feature[PipelineQuery, Option[gt.UserType]] + object WhoToFollowExcludedUserIdsFeature + extends FeatureWithDefaultOnFailure[PipelineQuery, Seq[Long]] { + override def defaultValue = Seq.empty + } + + // Result Features + object ServedSizeFeature extends Feature[PipelineQuery, Option[Int]] + object HasRandomTweetFeature extends Feature[PipelineQuery, Boolean] + object IsRandomTweetAboveFeature extends Feature[TweetCandidate, Boolean] + object ServedInConversationModuleFeature extends Feature[TweetCandidate, Boolean] + object ConversationModule2DisplayedTweetsFeature extends Feature[TweetCandidate, Boolean] + object ConversationModuleHasGapFeature extends Feature[TweetCandidate, Boolean] + object SGSValidLikedByUserIdsFeature extends Feature[TweetCandidate, Seq[Long]] + object SGSValidFollowedByUserIdsFeature extends Feature[TweetCandidate, Seq[Long]] + object PerspectiveFilteredLikedByUserIdsFeature extends Feature[TweetCandidate, Seq[Long]] + object ScreenNamesFeature extends Feature[TweetCandidate, Map[Long, String]] + object RealNamesFeature extends Feature[TweetCandidate, Map[Long, String]] + + /** + * Features around the focal Tweet for Tweets which should be rendered in convo modules. + * These are needed in order to render social context above the root tweet in a convo modules. + * For example if we have a convo module A-B-C (A Tweets, B replies to A, C replies to B), the descendant features are + * for the Tweet C. These features are None except for the root Tweet for Tweets which should render into + * convo modules. + */ + object FocalTweetAuthorIdFeature extends Feature[TweetCandidate, Option[Long]] + object FocalTweetInNetworkFeature extends Feature[TweetCandidate, Option[Boolean]] + object FocalTweetRealNamesFeature extends Feature[TweetCandidate, Option[Map[Long, String]]] + object FocalTweetScreenNamesFeature extends Feature[TweetCandidate, Option[Map[Long, String]]] + object MediaUnderstandingAnnotationIdsFeature extends Feature[TweetCandidate, Seq[Long]] +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/model/request/BUILD.bazel b/home-mixer/server/src/main/scala/com/twitter/home_mixer/model/request/BUILD.bazel new file mode 100644 index 0000000000..2212b81590 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/model/request/BUILD.bazel @@ -0,0 +1,16 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + strict_deps = True, + tags = ["bazel-compatible"], + dependencies = [ + "dspbidder/thrift/src/main/thrift/com/twitter/dspbidder/commons:thrift-scala", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/common", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/common/identifier", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/request", + "timelineservice/common:model", + ], + exports = [ + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/request", + ], +) diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/model/request/DeviceContext.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/model/request/DeviceContext.scala new file mode 100644 index 0000000000..ef96865ca4 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/model/request/DeviceContext.scala @@ -0,0 +1,74 @@ +package com.twitter.home_mixer.model.request + +import com.twitter.product_mixer.core.model.marshalling.request.ClientContext +import com.twitter.{timelineservice => tls} + +case class DeviceContext( + isPolling: Option[Boolean], + requestContext: Option[String], + latestControlAvailable: Option[Boolean], + autoplayEnabled: Option[Boolean]) { + + lazy val requestContextValue: Option[DeviceContext.RequestContext.Value] = + requestContext.flatMap { value => + val normalizedValue = value.trim.toLowerCase() + DeviceContext.RequestContext.values.find(_.toString == normalizedValue) + } + + def toTimelineServiceDeviceContext(clientContext: ClientContext): tls.DeviceContext = + tls.DeviceContext( + countryCode = clientContext.countryCode, + languageCode = clientContext.languageCode, + clientAppId = clientContext.appId, + ipAddress = clientContext.ipAddress, + guestId = clientContext.guestId, + sessionId = None, + timezone = None, + userAgent = clientContext.userAgent, + deviceId = clientContext.deviceId, + isPolling = isPolling, + requestProvenance = requestContext, + referrer = None, + tfeAuthHeader = None, + mobileDeviceId = clientContext.mobileDeviceId, + isSessionStart = None, + displaySize = None, + isURTRequest = Some(true), + latestControlAvailable = latestControlAvailable, + guestIdMarketing = clientContext.guestIdMarketing, + isInternalOrTwoffice = clientContext.isTwoffice, + browserNotificationPermission = None, + guestIdAds = clientContext.guestIdAds, + ) +} + +object DeviceContext { + val Empty: DeviceContext = DeviceContext( + isPolling = None, + requestContext = None, + latestControlAvailable = None, + autoplayEnabled = None + ) + + /** + * Constants which reflect valid client request provenances (why a request was initiated, encoded + * by the "request_context" HTTP parameter). + */ + object RequestContext extends Enumeration { + val Auto = Value("auto") + val Foreground = Value("foreground") + val Gap = Value("gap") + val Launch = Value("launch") + val ManualRefresh = Value("manual_refresh") + val Navigate = Value("navigate") + val Polling = Value("polling") + val PullToRefresh = Value("ptr") + val Signup = Value("signup") + val TweetSelfThread = Value("tweet_self_thread") + val BackgroundFetch = Value("background_fetch") + } +} + +trait HasDeviceContext { + def deviceContext: Option[DeviceContext] +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/model/request/HasListId.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/model/request/HasListId.scala new file mode 100644 index 0000000000..ece09a0d4c --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/model/request/HasListId.scala @@ -0,0 +1,8 @@ +package com.twitter.home_mixer.model.request + +/** + * [[HasListId]] enables shared components to access the list id shared by all list timeline products. + */ +trait HasListId { + def listId: Long +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/model/request/HasSeenTweetIds.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/model/request/HasSeenTweetIds.scala new file mode 100644 index 0000000000..0ec6573847 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/model/request/HasSeenTweetIds.scala @@ -0,0 +1,9 @@ +package com.twitter.home_mixer.model.request + +/** + * [[HasSeenTweetIds]] enables shared components to access the list of impressed tweet IDs + * sent by clients across different Home Mixer query types (e.g. FollowingQuery, ForYouQuery) + */ +trait HasSeenTweetIds { + def seenTweetIds: Option[Seq[Long]] +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/model/request/HomeMixerDebugOptions.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/model/request/HomeMixerDebugOptions.scala new file mode 100644 index 0000000000..8183809462 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/model/request/HomeMixerDebugOptions.scala @@ -0,0 +1,8 @@ +package com.twitter.home_mixer.model.request + +import com.twitter.product_mixer.core.model.marshalling.request.DebugOptions +import com.twitter.util.Time + +case class HomeMixerDebugOptions( + override val requestTimeOverride: Option[Time]) + extends DebugOptions diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/model/request/HomeMixerProduct.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/model/request/HomeMixerProduct.scala new file mode 100644 index 0000000000..107f4c2435 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/model/request/HomeMixerProduct.scala @@ -0,0 +1,35 @@ +package com.twitter.home_mixer.model.request + +import com.twitter.product_mixer.core.model.common.identifier.ProductIdentifier +import com.twitter.product_mixer.core.model.marshalling.request.Product + +/** + * Identifier names on products can be used to create Feature Switch rules by product, + * which useful if bucketing occurs in a component shared by multiple products. + * @see [[Product.identifier]] + */ + +case object FollowingProduct extends Product { + override val identifier: ProductIdentifier = ProductIdentifier("Following") + override val stringCenterProject: Option[String] = Some("timelinemixer") +} + +case object ForYouProduct extends Product { + override val identifier: ProductIdentifier = ProductIdentifier("ForYou") + override val stringCenterProject: Option[String] = Some("timelinemixer") +} + +case object ScoredTweetsProduct extends Product { + override val identifier: ProductIdentifier = ProductIdentifier("ScoredTweets") + override val stringCenterProject: Option[String] = Some("timelinemixer") +} + +case object ListTweetsProduct extends Product { + override val identifier: ProductIdentifier = ProductIdentifier("ListTweets") + override val stringCenterProject: Option[String] = Some("timelinemixer") +} + +case object ListRecommendedUsersProduct extends Product { + override val identifier: ProductIdentifier = ProductIdentifier("ListRecommendedUsers") + override val stringCenterProject: Option[String] = Some("timelinemixer") +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/model/request/HomeMixerProductContext.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/model/request/HomeMixerProductContext.scala new file mode 100644 index 0000000000..9f3ec4cb78 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/model/request/HomeMixerProductContext.scala @@ -0,0 +1,34 @@ +package com.twitter.home_mixer.model.request + +import com.twitter.dspbidder.commons.thriftscala.DspClientContext +import com.twitter.product_mixer.core.model.marshalling.request.ProductContext + +case class FollowingProductContext( + deviceContext: Option[DeviceContext], + seenTweetIds: Option[Seq[Long]], + dspClientContext: Option[DspClientContext]) + extends ProductContext + +case class ForYouProductContext( + deviceContext: Option[DeviceContext], + seenTweetIds: Option[Seq[Long]], + dspClientContext: Option[DspClientContext]) + extends ProductContext + +case class ScoredTweetsProductContext( + deviceContext: Option[DeviceContext], + seenTweetIds: Option[Seq[Long]], + servedTweetIds: Option[Seq[Long]]) + extends ProductContext + +case class ListTweetsProductContext( + listId: Long, + deviceContext: Option[DeviceContext], + dspClientContext: Option[DspClientContext]) + extends ProductContext + +case class ListRecommendedUsersProductContext( + listId: Long, + selectedUserIds: Option[Seq[Long]], + excludedUserIds: Option[Seq[Long]]) + extends ProductContext diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/model/request/HomeMixerRequest.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/model/request/HomeMixerRequest.scala new file mode 100644 index 0000000000..34595c34ce --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/model/request/HomeMixerRequest.scala @@ -0,0 +1,19 @@ +package com.twitter.home_mixer.model.request + +import com.twitter.product_mixer.core.model.marshalling.request.ClientContext +import com.twitter.product_mixer.core.model.marshalling.request.DebugParams +import com.twitter.product_mixer.core.model.marshalling.request.Product +import com.twitter.product_mixer.core.model.marshalling.request.ProductContext +import com.twitter.product_mixer.core.model.marshalling.request.Request + +case class HomeMixerRequest( + override val clientContext: ClientContext, + override val product: Product, + // Product-specific parameters should be placed in the Product Context + override val productContext: Option[ProductContext], + override val serializedRequestCursor: Option[String], + override val maxResults: Option[Int], + override val debugParams: Option[DebugParams], + // Parameters that apply to all products can be promoted to the request-level + homeRequestParam: Boolean) + extends Request diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/AdvertiserBrandSafetySettingsStoreModule.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/AdvertiserBrandSafetySettingsStoreModule.scala new file mode 100644 index 0000000000..d095d20548 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/AdvertiserBrandSafetySettingsStoreModule.scala @@ -0,0 +1,56 @@ +package com.twitter.home_mixer.module + +import com.google.inject.Provides + +import com.twitter.adserver.{thriftscala => ads} +import com.twitter.conversions.DurationOps._ +import com.twitter.finagle.mtls.authentication.ServiceIdentifier +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.inject.TwitterModule +import com.twitter.storage.client.manhattan.kv.Guarantee +import com.twitter.storehaus.ReadableStore +import com.twitter.storehaus_internal.manhattan.ManhattanCluster +import com.twitter.storehaus_internal.manhattan.ManhattanClusters +import com.twitter.timelines.clients.ads.AdvertiserBrandSafetySettingsStore +import com.twitter.timelines.clients.manhattan.mhv3.ManhattanClientBuilder +import com.twitter.timelines.clients.manhattan.mhv3.ManhattanClientConfigWithDataset +import com.twitter.util.Duration + +import javax.inject.Singleton + +object AdvertiserBrandSafetySettingsStoreModule extends TwitterModule { + + @Provides + @Singleton + def providesAdvertiserBrandSafetySettingsStore( + injectedServiceIdentifier: ServiceIdentifier, + statsReceiver: StatsReceiver + ): ReadableStore[Long, ads.AdvertiserBrandSafetySettings] = { + val advertiserBrandSafetySettingsManhattanClientConfig = new ManhattanClientConfigWithDataset { + override val cluster: ManhattanCluster = ManhattanClusters.apollo + override val appId: String = "brand_safety_apollo" + override val dataset = "advertiser_brand_safety_settings" + override val statsScope: String = "AdvertiserBrandSafetySettingsManhattanClient" + override val defaultGuarantee = Guarantee.Weak + override val defaultMaxTimeout: Duration = 100.milliseconds + override val maxRetryCount: Int = 1 + override val isReadOnly: Boolean = true + override val serviceIdentifier: ServiceIdentifier = injectedServiceIdentifier + } + + val advertiserBrandSafetySettingsManhattanEndpoint = ManhattanClientBuilder + .buildManhattanEndpoint(advertiserBrandSafetySettingsManhattanClientConfig, statsReceiver) + + val advertiserBrandSafetySettingsStore: ReadableStore[Long, ads.AdvertiserBrandSafetySettings] = + AdvertiserBrandSafetySettingsStore + .cached( + advertiserBrandSafetySettingsManhattanEndpoint, + advertiserBrandSafetySettingsManhattanClientConfig.dataset, + ttl = 60.minutes, + maxKeys = 100000, + windowSize = 10L + )(statsReceiver) + + advertiserBrandSafetySettingsStore + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/BUILD.bazel b/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/BUILD.bazel new file mode 100644 index 0000000000..651d2f1d8f --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/BUILD.bazel @@ -0,0 +1,87 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + strict_deps = True, + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/com/twitter/bijection:scrooge", + "3rdparty/jvm/com/twitter/bijection:thrift", + "3rdparty/jvm/com/twitter/src/java/com/twitter/logpipeline/client:logpipeline-event-publisher-thin", + "3rdparty/jvm/com/twitter/storehaus:core", + "3rdparty/jvm/io/netty:netty4-tcnative-boringssl-static", + "eventbus/client/src/main/scala/com/twitter/eventbus/client", + "finagle-internal/finagle-grpc/src/main/scala", + "finagle-internal/mtls/src/main/scala/com/twitter/finagle/mtls/authentication", + "finagle-internal/mtls/src/main/scala/com/twitter/finagle/mtls/client", + "finagle/finagle-core/src/main", + "finagle/finagle-memcached/src/main/scala", + "finagle/finagle-mux/src/main/scala", + "finagle/finagle-thriftmux/src/main/scala", + "finatra/inject/inject-core/src/main/scala", + "hermit/hermit-core/src/main/scala/com/twitter/hermit/store/common", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/param", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/store", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/util", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/util/earlybird", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/util/tweetypie", + "home-mixer/thrift/src/main/thrift:thrift-scala", + "interests-service/thrift/src/main/thrift:thrift-scala", + "people-discovery/api/thrift/src/main/thrift:thrift-scala", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/module", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline", + "product-mixer/shared-library/src/main/scala/com/twitter/product_mixer/shared_library/manhattan_client", + "product-mixer/shared-library/src/main/scala/com/twitter/product_mixer/shared_library/memcached_client", + "product-mixer/shared-library/src/main/scala/com/twitter/product_mixer/shared_library/thrift_client", + "servo/client/src/main/scala/com/twitter/servo/client", + "servo/manhattan", + "servo/util", + "socialgraph/server/src/main/scala/com/twitter/socialgraph/util", + "src/scala/com/twitter/ml/featurestore/lib", + "src/scala/com/twitter/scalding_internal/multiformat/format", + "src/scala/com/twitter/storehaus_internal", + "src/scala/com/twitter/summingbird_internal/bijection:bijection-implicits", + "src/scala/com/twitter/timelines/util", + "src/thrift/com/twitter/ads/adserver:adserver_rpc-scala", + "src/thrift/com/twitter/clientapp/gen:clientapp-scala", + "src/thrift/com/twitter/hermit/candidate:hermit-candidate-scala", + "src/thrift/com/twitter/manhattan:v1-scala", + "src/thrift/com/twitter/manhattan:v2-scala", + "src/thrift/com/twitter/onboarding/relevance/features:features-java", + "src/thrift/com/twitter/search:earlybird-scala", + "src/thrift/com/twitter/socialgraph:thrift-scala", + "src/thrift/com/twitter/timelines/author_features:thrift-java", + "src/thrift/com/twitter/timelines/impression_bloom_filter:thrift-scala", + "src/thrift/com/twitter/timelines/impression_store:thrift-scala", + "src/thrift/com/twitter/timelines/real_graph:real_graph-scala", + "src/thrift/com/twitter/timelines/suggests/common:poly_data_record-java", + "src/thrift/com/twitter/timelines/timeline_logging:thrift-scala", + "src/thrift/com/twitter/timelinescorer/common/scoredtweetcandidate:thrift-scala", + "src/thrift/com/twitter/topic_recos:topic_recos-thrift-java", + "src/thrift/com/twitter/user_session_store:thrift-java", + "src/thrift/com/twitter/wtf/candidate:wtf-candidate-scala", + "stitch/stitch-socialgraph", + "stitch/stitch-tweetypie", + "strato/src/main/scala/com/twitter/strato/client", + "timelinemixer/common/src/main/scala/com/twitter/timelinemixer/clients/feedback", + "timelinemixer/common/src/main/scala/com/twitter/timelinemixer/clients/manhattan", + "timelinemixer/server/src/main/scala/com/twitter/timelinemixer/injection/store/persistence", + "timelines:decider", + "timelines/src/main/scala/com/twitter/timelines/clients/ads", + "timelines/src/main/scala/com/twitter/timelines/clients/manhattan", + "timelines/src/main/scala/com/twitter/timelines/clients/manhattan/store", + "timelines/src/main/scala/com/twitter/timelines/clients/predictionservice", + "timelines/src/main/scala/com/twitter/timelines/clients/strato", + "timelines/src/main/scala/com/twitter/timelines/clients/strato/twistly", + "timelines/src/main/scala/com/twitter/timelines/config", + "timelines/src/main/scala/com/twitter/timelines/impressionstore/impressionbloomfilter", + "timelines/src/main/scala/com/twitter/timelines/impressionstore/store", + "timelines/src/main/scala/com/twitter/timelines/util/stats", + "timelineservice/common/src/main/scala/com/twitter/timelineservice/model", + "tweetconvosvc/client/src/main/scala/com/twitter/tweetconvosvc/client/builder", + "twitter-config/yaml", + ], + exports = [ + "timelines/src/main/scala/com/twitter/timelines/clients/predictionservice", + ], +) diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/ClientSentImpressionsPublisherModule.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/ClientSentImpressionsPublisherModule.scala new file mode 100644 index 0000000000..f7c5e9bc76 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/ClientSentImpressionsPublisherModule.scala @@ -0,0 +1,48 @@ +package com.twitter.home_mixer.module + +import com.google.inject.Provides +import com.twitter.conversions.DurationOps._ +import com.twitter.eventbus.client.EventBusPublisher +import com.twitter.eventbus.client.EventBusPublisherBuilder +import com.twitter.finagle.mtls.authentication.ServiceIdentifier +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.inject.TwitterModule +import com.twitter.timelines.config.ConfigUtils +import com.twitter.timelines.config.Env +import com.twitter.timelines.impressionstore.thriftscala.PublishedImpressionList +import javax.inject.Singleton + +object ClientSentImpressionsPublisherModule extends TwitterModule with ConfigUtils { + private val serviceName = "home-mixer" + + @Singleton + @Provides + def providesClientSentImpressionsPublisher( + serviceIdentifier: ServiceIdentifier, + statsReceiver: StatsReceiver + ): EventBusPublisher[PublishedImpressionList] = { + val env = serviceIdentifier.environment.toLowerCase match { + case "prod" => Env.prod + case "staging" => Env.staging + case "local" => Env.local + case _ => Env.devel + } + + val streamName = env match { + case Env.prod => "timelinemixer_client_sent_impressions_prod" + case _ => "timelinemixer_client_sent_impressions_devel" + } + + EventBusPublisherBuilder() + .clientId(clientIdWithScopeOpt(serviceName, env)) + .serviceIdentifier(serviceIdentifier) + .streamName(streamName) + .statsReceiver(statsReceiver.scope("eventbus")) + .thriftStruct(PublishedImpressionList) + .tcpConnectTimeout(20.milliseconds) + .connectTimeout(100.milliseconds) + .requestTimeout(1.second) + .publishTimeout(1.second) + .build() + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/ConversationServiceModule.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/ConversationServiceModule.scala new file mode 100644 index 0000000000..9a7d8771c6 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/ConversationServiceModule.scala @@ -0,0 +1,37 @@ +package com.twitter.home_mixer.module + +import com.twitter.conversions.DurationOps._ +import com.twitter.finagle.ThriftMux +import com.twitter.finagle.thriftmux.MethodBuilder +import com.twitter.finatra.mtls.thriftmux.modules.MtlsClient +import com.twitter.inject.Injector +import com.twitter.inject.thrift.modules.ThriftMethodBuilderClientModule +import com.twitter.tweetconvosvc.thriftscala.ConversationService +import com.twitter.util.Duration +import org.apache.thrift.protocol.TCompactProtocol + +object ConversationServiceModule + extends ThriftMethodBuilderClientModule[ + ConversationService.ServicePerEndpoint, + ConversationService.MethodPerEndpoint + ] + with MtlsClient { + + override val label: String = "tweetconvosvc" + override val dest: String = "/s/tweetconvosvc/tweetconvosvc" + + override protected def configureMethodBuilder( + injector: Injector, + methodBuilder: MethodBuilder + ): MethodBuilder = methodBuilder.withTimeoutPerRequest(100.milliseconds) + + override def configureThriftMuxClient( + injector: Injector, + client: ThriftMux.Client + ): ThriftMux.Client = + super + .configureThriftMuxClient(injector, client) + .withProtocolFactory(new TCompactProtocol.Factory()) + + override protected def sessionAcquisitionTimeout: Duration = 500.milliseconds +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/FeedbackHistoryClientModule.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/FeedbackHistoryClientModule.scala new file mode 100644 index 0000000000..83093de453 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/FeedbackHistoryClientModule.scala @@ -0,0 +1,39 @@ +package com.twitter.home_mixer.module + +import com.google.inject.Provides +import com.twitter.finagle.mtls.authentication.ServiceIdentifier +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.inject.TwitterModule +import com.twitter.timelinemixer.clients.feedback.FeedbackHistoryManhattanClient +import com.twitter.timelinemixer.clients.feedback.FeedbackHistoryManhattanClientConfig +import com.twitter.timelines.clients.manhattan.mhv3.ManhattanClientBuilder +import javax.inject.Singleton + +object FeedbackHistoryClientModule extends TwitterModule { + private val ProdDataset = "feedback_history" + private val StagingDataset = "feedback_history_nonprod" + + @Provides + @Singleton + def providesFeedbackHistoryClient( + serviceId: ServiceIdentifier, + statsReceiver: StatsReceiver + ) = { + val manhattanDataset = serviceId.environment.toLowerCase match { + case "prod" => ProdDataset + case _ => StagingDataset + } + + val config = new FeedbackHistoryManhattanClientConfig { + val dataset = manhattanDataset + val isReadOnly = true + val serviceIdentifier = serviceId + } + + new FeedbackHistoryManhattanClient( + ManhattanClientBuilder.buildManhattanEndpoint(config, statsReceiver), + manhattanDataset, + statsReceiver + ) + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/HomeAdsCandidateSourceModule.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/HomeAdsCandidateSourceModule.scala new file mode 100644 index 0000000000..73e1500a0e --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/HomeAdsCandidateSourceModule.scala @@ -0,0 +1,32 @@ +package com.twitter.home_mixer.module + +import com.twitter.adserver.thriftscala.NewAdServer +import com.twitter.conversions.DurationOps._ +import com.twitter.finagle.thriftmux.MethodBuilder +import com.twitter.finatra.mtls.thriftmux.modules.MtlsClient +import com.twitter.inject.Injector +import com.twitter.inject.thrift.modules.ThriftMethodBuilderClientModule +import com.twitter.util.Duration + +object HomeAdsCandidateSourceModule + extends ThriftMethodBuilderClientModule[ + NewAdServer.ServicePerEndpoint, + NewAdServer.MethodPerEndpoint + ] + with MtlsClient { + + override val label = "adserver" + override val dest = "/s/ads/adserver" + + override protected def configureMethodBuilder( + injector: Injector, + methodBuilder: MethodBuilder + ): MethodBuilder = { + methodBuilder + .withTimeoutPerRequest(1200.milliseconds) + .withTimeoutTotal(1200.milliseconds) + .withMaxRetries(2) + } + + override protected def sessionAcquisitionTimeout: Duration = 150.milliseconds +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/HomeMixerFlagsModule.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/HomeMixerFlagsModule.scala new file mode 100644 index 0000000000..4031596a71 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/HomeMixerFlagsModule.scala @@ -0,0 +1,53 @@ +package com.twitter.home_mixer.module + +import com.twitter.conversions.DurationOps.RichDuration +import com.twitter.home_mixer.param.HomeMixerFlagName +import com.twitter.inject.TwitterModule +import com.twitter.util.Duration + +object HomeMixerFlagsModule extends TwitterModule { + + import HomeMixerFlagName._ + + flag[Boolean]( + name = ScribeClientEventsFlag, + default = false, + help = "Toggles logging client events to Scribe" + ) + + flag[Boolean]( + name = ScribeServedEntriesFlag, + default = false, + help = "Toggles logging served entries to Scribe" + ) + + flag[Boolean]( + name = ScribeServedCommonFeaturesAndCandidateFeaturesFlag, + default = false, + help = "Toggles logging served common features and candidates features to Scribe" + ) + + flag[String]( + name = DataRecordMetadataStoreConfigsYmlFlag, + default = "", + help = "The YML file that contains the necessary info for creating metadata store MySQL client." + ) + + flag[String]( + name = DarkTrafficFilterDeciderKey, + default = "dark_traffic_filter", + help = "Dark traffic filter decider key" + ) + + flag[Duration]( + TargetFetchLatency, + 300.millis, + "Target fetch latency from candidate sources for Quality Factor" + ) + + flag[Duration]( + TargetScoringLatency, + 700.millis, + "Target scoring latency for Quality Factor" + ) +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/HomeMixerResourcesModule.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/HomeMixerResourcesModule.scala new file mode 100644 index 0000000000..d4bfab2212 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/HomeMixerResourcesModule.scala @@ -0,0 +1,18 @@ +package com.twitter.home_mixer.module + +import com.google.inject.Provides +import com.twitter.config.yaml.YamlMap +import com.twitter.home_mixer.param.HomeMixerInjectionNames.DDGStatsAuthors +import com.twitter.inject.TwitterModule +import javax.inject.Named +import javax.inject.Singleton + +object HomeMixerResourcesModule extends TwitterModule { + + private val AuthorsFile = "/config/authors.yml" + + @Provides + @Singleton + @Named(DDGStatsAuthors) + def providesDDGStatsAuthors(): YamlMap = YamlMap.load(AuthorsFile) +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/HomeNaviModelClientModule.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/HomeNaviModelClientModule.scala new file mode 100644 index 0000000000..df70592c27 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/HomeNaviModelClientModule.scala @@ -0,0 +1,52 @@ +package com.twitter.home_mixer.module + +import com.google.inject.Provides +import com.twitter.conversions.DurationOps._ +import com.twitter.finagle.Http +import com.twitter.finagle.grpc.FinagleChannelBuilder +import com.twitter.finagle.mtls.authentication.ServiceIdentifier +import com.twitter.finagle.mtls.client.MtlsStackClient.MtlsStackClientSyntax +import com.twitter.inject.TwitterModule +import com.twitter.timelines.clients.predictionservice.PredictionGRPCService +import com.twitter.util.Duration +import io.grpc.ManagedChannel + +import javax.inject.Singleton + +object HomeNaviModelClientModule extends TwitterModule { + + @Singleton + @Provides + def providesPredictionGRPCService( + serviceIdentifier: ServiceIdentifier, + ): PredictionGRPCService = { + // Wily path to the ML Model service (e.g. /s/ml-serving/navi-explore-ranker). + val modelPath = "/s/ml-serving/navi_home_recap_onnx" + + // timeout for prediction service requests. + val MaxPredictionTimeoutMs: Duration = 300.millis + val ConnectTimeoutMs: Duration = 200.millis + val AcquisitionTimeoutMs: Duration = 20000.millis + val MaxRetryAttempts: Int = 2 + + val client = Http.client + .withLabel(modelPath) + .withMutualTls(serviceIdentifier) + .withRequestTimeout(MaxPredictionTimeoutMs) + .withTransport.connectTimeout(ConnectTimeoutMs) + .withSession.acquisitionTimeout(AcquisitionTimeoutMs) + .withHttpStats + + val channel: ManagedChannel = FinagleChannelBuilder + .forTarget(modelPath) + .overrideAuthority("rustserving") + .maxRetryAttempts(MaxRetryAttempts) + .enableRetryForStatus(io.grpc.Status.RESOURCE_EXHAUSTED) + .enableRetryForStatus(io.grpc.Status.UNKNOWN) + .enableUnsafeFullyBufferingMode() + .httpClient(client) + .build() + + new PredictionGRPCService(channel) + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/ImpressionBloomFilterModule.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/ImpressionBloomFilterModule.scala new file mode 100644 index 0000000000..cb339a1d58 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/ImpressionBloomFilterModule.scala @@ -0,0 +1,54 @@ +package com.twitter.home_mixer.module + +import com.google.inject.Provides +import com.twitter.conversions.DurationOps._ +import com.twitter.finagle.mtls.authentication.ServiceIdentifier +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.inject.TwitterModule +import com.twitter.storage.client.manhattan.kv.Guarantee +import com.twitter.storehaus_internal.manhattan.ManhattanClusters +import com.twitter.timelines.clients.manhattan.store._ +import com.twitter.timelines.impressionstore.impressionbloomfilter.ImpressionBloomFilter +import com.twitter.timelines.impressionstore.impressionbloomfilter.ImpressionBloomFilterManhattanKeyValueDescriptor +import javax.inject.Singleton + +object ImpressionBloomFilterModule extends TwitterModule { + + private val ProdAppId = "impression_bloom_filter_store" + private val ProdDataset = "impression_bloom_filter" + private val StagingAppId = "impression_bloom_filter_store_staging" + private val StagingDataset = "impression_bloom_filter_staging" + private val ClientStatsScope = "tweetBloomFilterImpressionManhattanClient" + private val DefaultTTL = 7.days + + @Provides + @Singleton + def providesImpressionBloomFilter( + serviceIdentifier: ServiceIdentifier, + statsReceiver: StatsReceiver + ): ImpressionBloomFilter = { + val (appId, dataset) = serviceIdentifier.environment.toLowerCase match { + case "prod" => (ProdAppId, ProdDataset) + case _ => (StagingAppId, StagingDataset) + } + + implicit val manhattanKeyValueDescriptor = ImpressionBloomFilterManhattanKeyValueDescriptor( + dataset = dataset, + ttl = DefaultTTL + ) + + val manhattanClient = ManhattanStoreClientBuilder.buildManhattanClient( + serviceIdentifier = serviceIdentifier, + cluster = ManhattanClusters.nash, + appId = appId, + defaultMaxTimeout = 100.milliseconds, + maxRetryCount = 2, + defaultGuarantee = Some(Guarantee.SoftDcReadMyWrites), + isReadOnly = false, + statsScope = ClientStatsScope, + statsReceiver = statsReceiver + ) + + ImpressionBloomFilter(manhattanClient) + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/InjectionHistoryClientModule.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/InjectionHistoryClientModule.scala new file mode 100644 index 0000000000..fe274ff1d0 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/InjectionHistoryClientModule.scala @@ -0,0 +1,88 @@ +package com.twitter.home_mixer.module + +import com.google.inject.Provides +import com.twitter.conversions.DurationOps._ +import com.twitter.finagle.ThriftMux +import com.twitter.finagle.builder.ClientBuilder +import com.twitter.finagle.mtls.authentication.ServiceIdentifier +import com.twitter.finagle.mtls.client.MtlsStackClient._ +import com.twitter.finagle.service.RetryPolicy +import com.twitter.finagle.ssl.OpportunisticTls +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.inject.TwitterModule +import com.twitter.manhattan.v2.thriftscala.{ManhattanCoordinator => ManhattanV2} +import com.twitter.timelinemixer.clients.manhattan.InjectionHistoryClient +import com.twitter.timelinemixer.clients.manhattan.ManhattanDatasetConfig +import com.twitter.timelines.clients.manhattan.Dataset +import com.twitter.timelines.clients.manhattan.ManhattanClient +import com.twitter.timelines.util.stats.RequestScope +import javax.inject.Singleton +import org.apache.thrift.protocol.TBinaryProtocol +import com.twitter.timelines.config.TimelinesUnderlyingClientConfiguration.ConnectTimeout +import com.twitter.timelines.config.TimelinesUnderlyingClientConfiguration.TCPConnectTimeout + +object InjectionHistoryClientModule extends TwitterModule { + private val ProdDataset = "suggestion_history" + private val StagingDataset = "suggestion_history_nonprod" + private val AppId = "twitter_suggests" + private val ServiceName = "manhattan.omega" + private val OmegaManhattanDest = "/s/manhattan/omega.native-thrift" + private val InjectionRequestScope = RequestScope("injectionHistoryClient") + private val RequestTimeout = 75.millis + private val Timeout = 150.millis + + val retryPolicy = RetryPolicy.tries( + 2, + RetryPolicy.TimeoutAndWriteExceptionsOnly + .orElse(RetryPolicy.ChannelClosedExceptionsOnly)) + + @Provides + @Singleton + def providesInjectionHistoryClient( + serviceIdentifier: ServiceIdentifier, + statsReceiver: StatsReceiver + ) = { + val dataset = serviceIdentifier.environment.toLowerCase match { + case "prod" => ProdDataset + case _ => StagingDataset + } + + val thriftMuxClient = ClientBuilder() + .name(ServiceName) + .daemon(daemonize = true) + .failFast(enabled = true) + .retryPolicy(retryPolicy) + .tcpConnectTimeout(TCPConnectTimeout) + .connectTimeout(ConnectTimeout) + .dest(OmegaManhattanDest) + .requestTimeout(RequestTimeout) + .timeout(Timeout) + .stack(ThriftMux.client + .withMutualTls(serviceIdentifier) + .withOpportunisticTls(OpportunisticTls.Required)) + .build() + + val manhattanOmegaClient = new ManhattanV2.FinagledClient( + service = thriftMuxClient, + protocolFactory = new TBinaryProtocol.Factory(), + serviceName = ServiceName, + ) + + val readOnlyMhClient = new ManhattanClient( + appId = AppId, + manhattan = manhattanOmegaClient, + requestScope = InjectionRequestScope, + serviceName = ServiceName, + statsReceiver = statsReceiver + ).readOnly + + val mhDatasetConfig = new ManhattanDatasetConfig { + override val SuggestionHistoryDataset = Dataset(dataset) + } + + new InjectionHistoryClient( + readOnlyMhClient, + mhDatasetConfig + ) + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/ManhattanClientsModule.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/ManhattanClientsModule.scala new file mode 100644 index 0000000000..5bdbf9e974 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/ManhattanClientsModule.scala @@ -0,0 +1,56 @@ +package com.twitter.home_mixer.module + +import com.google.inject.Provides +import com.twitter.finagle.mtls.authentication.ServiceIdentifier +import com.twitter.home_mixer.param.HomeMixerInjectionNames.RealGraphManhattanEndpoint +import com.twitter.home_mixer.param.HomeMixerInjectionNames.UserMetadataManhattanEndpoint +import com.twitter.inject.TwitterModule +import com.twitter.storage.client.manhattan.kv._ +import com.twitter.timelines.config.ConfigUtils +import com.twitter.util.Duration +import javax.inject.Named +import javax.inject.Singleton + +object ManhattanClientsModule extends TwitterModule with ConfigUtils { + + private val starbuckDest: String = "/s/manhattan/starbuck.native-thrift" + private val apolloDest: String = "/s/manhattan/apollo.native-thrift" + + @Provides + @Singleton + @Named(RealGraphManhattanEndpoint) + def providesRealGraphManhattanEndpoint( + serviceIdentifier: ServiceIdentifier + ): ManhattanKVEndpoint = { + lazy val client = ManhattanKVClient( + appId = "real_graph", + dest = apolloDest, + mtlsParams = ManhattanKVClientMtlsParams(serviceIdentifier = serviceIdentifier), + label = "real-graph-data" + ) + + ManhattanKVEndpointBuilder(client) + .maxRetryCount(2) + .defaultMaxTimeout(Duration.fromMilliseconds(100)) + .build() + } + + @Provides + @Singleton + @Named(UserMetadataManhattanEndpoint) + def providesUserMetadataManhattanEndpoint( + serviceIdentifier: ServiceIdentifier + ): ManhattanKVEndpoint = { + val client = ManhattanKVClient( + appId = "user_metadata", + dest = starbuckDest, + mtlsParams = ManhattanKVClientMtlsParams(serviceIdentifier = serviceIdentifier), + label = "user-metadata" + ) + + ManhattanKVEndpointBuilder(client) + .maxRetryCount(1) + .defaultMaxTimeout(Duration.fromMilliseconds(70)) + .build() + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/ManhattanFeatureRepositoryModule.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/ManhattanFeatureRepositoryModule.scala new file mode 100644 index 0000000000..f9afc1dcb3 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/ManhattanFeatureRepositoryModule.scala @@ -0,0 +1,453 @@ +package com.twitter.home_mixer.module + +import com.google.inject.Provides +import com.twitter.bijection.Injection +import com.twitter.bijection.scrooge.BinaryScalaCodec +import com.twitter.bijection.scrooge.CompactScalaCodec +import com.twitter.bijection.thrift.ThriftCodec +import com.twitter.conversions.DurationOps._ +import com.twitter.finagle.mtls.authentication.ServiceIdentifier +import com.twitter.home_mixer.param.HomeMixerInjectionNames._ +import com.twitter.home_mixer.util.InjectionTransformerImplicits._ +import com.twitter.home_mixer.util.TensorFlowUtil +import com.twitter.inject.TwitterModule +import com.twitter.manhattan.v1.thriftscala.ManhattanCoordinator +import com.twitter.manhattan.v1.{thriftscala => mh} +import com.twitter.ml.api.thriftscala.FloatTensor +import com.twitter.ml.api.{thriftscala => ml} +import com.twitter.ml.featurestore.lib.UserId +import com.twitter.ml.featurestore.thriftscala.EntityId +import com.twitter.onboarding.relevance.features.{thriftjava => rf} +import com.twitter.product_mixer.shared_library.manhattan_client.ManhattanClientBuilder +import com.twitter.scalding_internal.multiformat.format.keyval.KeyValInjection.ScalaBinaryThrift +import com.twitter.servo.cache._ +import com.twitter.servo.manhattan.ManhattanKeyValueRepository +import com.twitter.servo.repository.CachingKeyValueRepository +import com.twitter.servo.repository.ChunkingStrategy +import com.twitter.servo.repository.KeyValueRepository +import com.twitter.servo.repository.Repository +import com.twitter.servo.repository.keysAsQuery +import com.twitter.servo.util.Transformer +import com.twitter.storehaus_internal.manhattan.ManhattanClusters +import com.twitter.timelines.author_features.v1.{thriftjava => af} +import com.twitter.timelines.suggests.common.dense_data_record.thriftscala.DenseFeatureMetadata +import com.twitter.user_session_store.thriftscala.UserSession +import com.twitter.user_session_store.{thriftjava => uss} +import com.twitter.util.Duration +import com.twitter.util.Try +import java.nio.ByteBuffer +import javax.inject.Named +import javax.inject.Singleton +import org.apache.thrift.protocol.TCompactProtocol +import org.apache.thrift.transport.TMemoryInputTransport +import org.apache.thrift.transport.TTransport + +object ManhattanFeatureRepositoryModule extends TwitterModule { + + private val DEFAULT_RPC_CHUNK_SIZE = 50 + + private val ThriftEntityIdInjection = ScalaBinaryThrift(EntityId) + + val UserIdKeyTransformer = new Transformer[Long, ByteBuffer] { + override def to(userId: Long): Try[ByteBuffer] = { + Try(ByteBuffer.wrap(ThriftEntityIdInjection.apply(UserId(userId).toThrift))) + } + override def from(b: ByteBuffer): Try[Long] = ??? + } + + val FloatTensorTransformer = new Transformer[ByteBuffer, FloatTensor] { + override def to(input: ByteBuffer): Try[FloatTensor] = { + val floatTensor = TensorFlowUtil.embeddingByteBufferToFloatTensor(input) + Try(floatTensor) + } + + override def from(b: FloatTensor): Try[ByteBuffer] = ??? + } + + // manhattan clients + + @Provides + @Singleton + @Named(ManhattanApolloClient) + def providesManhattanApolloClient( + serviceIdentifier: ServiceIdentifier + ): mh.ManhattanCoordinator.MethodPerEndpoint = { + ManhattanClientBuilder + .buildManhattanV1FinagleClient( + ManhattanClusters.apollo, + serviceIdentifier + ) + } + + @Provides + @Singleton + @Named(ManhattanAthenaClient) + def providesManhattanAthenaClient( + serviceIdentifier: ServiceIdentifier + ): mh.ManhattanCoordinator.MethodPerEndpoint = { + ManhattanClientBuilder + .buildManhattanV1FinagleClient( + ManhattanClusters.athena, + serviceIdentifier + ) + } + + @Provides + @Singleton + @Named(ManhattanOmegaClient) + def providesManhattanOmegaClient( + serviceIdentifier: ServiceIdentifier + ): mh.ManhattanCoordinator.MethodPerEndpoint = { + ManhattanClientBuilder + .buildManhattanV1FinagleClient( + ManhattanClusters.omega, + serviceIdentifier + ) + } + + @Provides + @Singleton + @Named(ManhattanStarbuckClient) + def providesManhattanStarbuckClient( + serviceIdentifier: ServiceIdentifier + ): mh.ManhattanCoordinator.MethodPerEndpoint = { + ManhattanClientBuilder + .buildManhattanV1FinagleClient( + ManhattanClusters.starbuck, + serviceIdentifier + ) + } + + // non-cached manhattan repositories + + @Provides + @Singleton + @Named(MetricCenterUserCountingFeatureRepository) + def providesMetricCenterUserCountingFeatureRepository( + @Named(ManhattanStarbuckClient) client: mh.ManhattanCoordinator.MethodPerEndpoint + ): KeyValueRepository[Seq[Long], Long, rf.MCUserCountingFeatures] = { + + val keyTransformer = Injection + .connect[Long, Array[Byte]] + .toByteBufferTransformer() + + val valueTransformer = ThriftCodec + .toCompact[rf.MCUserCountingFeatures] + .toByteBufferTransformer() + .flip + + batchedManhattanKeyValueRepository[Long, rf.MCUserCountingFeatures]( + client = client, + keyTransformer = keyTransformer, + valueTransformer = valueTransformer, + appId = "wtf_ml", + dataset = "mc_user_counting_features_v0_starbuck", + timeoutInMillis = 100 + ) + } + + /** + * A repository of the offline aggregate feature metadata necessary to decode + * DenseCompactDataRecords. + * + * This repository is expected to virtually always pick up the metadata form the local cache with + * nearly 0 latency. + */ + @Provides + @Singleton + @Named(TimelineAggregateMetadataRepository) + def providesTimelineAggregateMetadataRepository( + @Named(ManhattanAthenaClient) client: mh.ManhattanCoordinator.MethodPerEndpoint + ): Repository[Int, Option[DenseFeatureMetadata]] = { + + val keyTransformer = Injection + .connect[Int, Array[Byte]] + .toByteBufferTransformer() + + val valueTransformer = new Transformer[ByteBuffer, DenseFeatureMetadata] { + private val compactProtocolFactory = new TCompactProtocol.Factory + + def to(buffer: ByteBuffer): Try[DenseFeatureMetadata] = Try { + val transport = transportFromByteBuffer(buffer) + DenseFeatureMetadata.decode(compactProtocolFactory.getProtocol(transport)) + } + + // Encoding intentionally not implemented as it is never used + def from(metadata: DenseFeatureMetadata): Try[ByteBuffer] = ??? + } + + val inProcessCache: Cache[Int, Cached[DenseFeatureMetadata]] = InProcessLruCacheFactory( + ttl = Duration.fromMinutes(20), + lruSize = 30 + ).apply(serializer = Transformer(_ => ???, _ => ???)) // Serialization is not necessary here. + + val keyValueRepository = new ManhattanKeyValueRepository( + client = client, + keyTransformer = keyTransformer, + valueTransformer = valueTransformer, + appId = "timelines_dense_aggregates_encoding_metadata", // Expected QPS is negligible. + dataset = "user_session_dense_feature_metadata", + timeoutInMillis = 100 + ) + + KeyValueRepository + .singular( + new CachingKeyValueRepository[Seq[Int], Int, DenseFeatureMetadata]( + keyValueRepository, + new NonLockingCache(inProcessCache), + keysAsQuery[Int] + ) + ) + } + + @Provides + @Singleton + @Named(RealGraphFeatureRepository) + def providesRealGraphFeatureRepository( + @Named(ManhattanApolloClient) client: mh.ManhattanCoordinator.MethodPerEndpoint + ): Repository[Long, Option[UserSession]] = { + val valueTransformer = CompactScalaCodec(UserSession).toByteBufferTransformer().flip + + KeyValueRepository.singular( + new ManhattanKeyValueRepository( + client = client, + keyTransformer = UserIdKeyTransformer, + valueTransformer = valueTransformer, + appId = "real_graph", + dataset = "real_graph_user_features", + timeoutInMillis = 100, + ) + ) + } + + // cached manhattan repositories + + @Provides + @Singleton + @Named(AuthorFeatureRepository) + def providesAuthorFeatureRepository( + @Named(ManhattanAthenaClient) client: mh.ManhattanCoordinator.MethodPerEndpoint, + @Named(HomeAuthorFeaturesCacheClient) cacheClient: Memcache + ): KeyValueRepository[Seq[Long], Long, af.AuthorFeatures] = { + + val keyTransformer = Injection + .connect[Long, Array[Byte]] + .toByteBufferTransformer() + + val valueInjection = ThriftCodec + .toCompact[af.AuthorFeatures] + + val keyValueRepository = batchedManhattanKeyValueRepository( + client = client, + keyTransformer = keyTransformer, + valueTransformer = valueInjection.toByteBufferTransformer().flip, + appId = "timelines_author_feature_store_athena", + dataset = "timelines_author_features", + timeoutInMillis = 100 + ) + + val remoteCacheRepo = buildMemCachedRepository( + keyValueRepository = keyValueRepository, + cacheClient = cacheClient, + cachePrefix = "AuthorFeatureHydrator", + ttl = 12.hours, + valueInjection = valueInjection) + + buildInProcessCachedRepository( + keyValueRepository = remoteCacheRepo, + ttl = 15.minutes, + size = 8000, + valueInjection = valueInjection + ) + } + + @Provides + @Singleton + @Named(TwhinAuthorFollow20200101FeatureRepository) + def providesTwhinAuthorFollow20200101FeatureRepository( + @Named(ManhattanApolloClient) client: mh.ManhattanCoordinator.MethodPerEndpoint, + @Named(TwhinAuthorFollow20200101FeatureCacheClient) cacheClient: Memcache + ): KeyValueRepository[Seq[Long], Long, ml.Embedding] = { + + val keyTransformer = Injection + .connect[Long, Array[Byte]] + .toByteBufferTransformer() + + val valueInjection: Injection[ml.Embedding, Array[Byte]] = + BinaryScalaCodec(ml.Embedding) + + val keyValueRepository = batchedManhattanKeyValueRepository( + client = client, + keyTransformer = keyTransformer, + valueTransformer = valueInjection.toByteBufferTransformer().flip, + appId = "twhin", + dataset = "twhinauthor_follow_0101", + timeoutInMillis = 100 + ) + + buildMemCachedRepository( + keyValueRepository = keyValueRepository, + cacheClient = cacheClient, + cachePrefix = "TwhinAuthorFollow20200101FeatureHydrator", + ttl = 48.hours, + valueInjection = valueInjection + ) + } + + @Provides + @Singleton + @Named(TwhinUserFollowFeatureRepository) + def providesTwhinUserFollowFeatureRepository( + @Named(ManhattanApolloClient) client: mh.ManhattanCoordinator.MethodPerEndpoint + ): KeyValueRepository[Seq[Long], Long, FloatTensor] = { + + batchedManhattanKeyValueRepository( + client = client, + keyTransformer = UserIdKeyTransformer, + valueTransformer = FloatTensorTransformer, + appId = "ml_features_apollo", + dataset = "twhin_user_follow_embedding_fsv1__v1_thrift__embedding", + timeoutInMillis = 100 + ) + } + + @Provides + @Singleton + @Named(TimelineAggregatePartARepository) + def providesTimelineAggregatePartARepository( + @Named(ManhattanApolloClient) client: mh.ManhattanCoordinator.MethodPerEndpoint, + ): Repository[Long, Option[uss.UserSession]] = + timelineAggregateRepository( + mhClient = client, + mhDataset = "timelines_aggregates_v2_features_by_user_part_a_apollo", + mhAppId = "timelines_aggregates_v2_features_by_user_part_a_apollo" + ) + + @Provides + @Singleton + @Named(TimelineAggregatePartBRepository) + def providesTimelineAggregatePartBRepository( + @Named(ManhattanApolloClient) client: mh.ManhattanCoordinator.MethodPerEndpoint, + ): Repository[Long, Option[uss.UserSession]] = + timelineAggregateRepository( + mhClient = client, + mhDataset = "timelines_aggregates_v2_features_by_user_part_b_apollo", + mhAppId = "timelines_aggregates_v2_features_by_user_part_b_apollo" + ) + + @Provides + @Singleton + @Named(TwhinUserEngagementFeatureRepository) + def providesTwhinUserEngagementFeatureRepository( + @Named(ManhattanApolloClient) client: mh.ManhattanCoordinator.MethodPerEndpoint + ): KeyValueRepository[Seq[Long], Long, FloatTensor] = { + + batchedManhattanKeyValueRepository( + client = client, + keyTransformer = UserIdKeyTransformer, + valueTransformer = FloatTensorTransformer, + appId = "ml_features_apollo", + dataset = "twhin_user_engagement_embedding_fsv1__v1_thrift__embedding", + timeoutInMillis = 100 + ) + } + + private def buildMemCachedRepository[K, V]( + keyValueRepository: KeyValueRepository[Seq[K], K, V], + cacheClient: Memcache, + cachePrefix: String, + ttl: Duration, + valueInjection: Injection[V, Array[Byte]] + ): CachingKeyValueRepository[Seq[K], K, V] = { + val cachedSerializer = CachedSerializer.binary( + valueInjection.toByteArrayTransformer() + ) + + val cache = MemcacheCacheFactory( + cacheClient, + ttl, + PrefixKeyTransformerFactory(cachePrefix) + )[K, Cached[V]](cachedSerializer) + + new CachingKeyValueRepository( + keyValueRepository, + new NonLockingCache(cache), + keysAsQuery[K] + ) + } + + private def buildInProcessCachedRepository[K, V]( + keyValueRepository: KeyValueRepository[Seq[K], K, V], + ttl: Duration, + size: Int, + valueInjection: Injection[V, Array[Byte]] + ): CachingKeyValueRepository[Seq[K], K, V] = { + val cachedSerializer = CachedSerializer.binary( + valueInjection.toByteArrayTransformer() + ) + + val cache = InProcessLruCacheFactory( + ttl = ttl, + lruSize = size + )[K, Cached[V]](cachedSerializer) + + new CachingKeyValueRepository( + keyValueRepository, + new NonLockingCache(cache), + keysAsQuery[K] + ) + } + + private def batchedManhattanKeyValueRepository[K, V]( + client: ManhattanCoordinator.MethodPerEndpoint, + keyTransformer: Transformer[K, ByteBuffer], + valueTransformer: Transformer[ByteBuffer, V], + appId: String, + dataset: String, + timeoutInMillis: Int, + chunkSize: Int = DEFAULT_RPC_CHUNK_SIZE + ): KeyValueRepository[Seq[K], K, V] = + KeyValueRepository.chunked( + new ManhattanKeyValueRepository( + client = client, + keyTransformer = keyTransformer, + valueTransformer = valueTransformer, + appId = appId, + dataset = dataset, + timeoutInMillis = timeoutInMillis + ), + chunker = ChunkingStrategy.equalSize(chunkSize) + ) + + private def transportFromByteBuffer(buffer: ByteBuffer): TTransport = + new TMemoryInputTransport( + buffer.array(), + buffer.arrayOffset() + buffer.position(), + buffer.remaining()) + + private def timelineAggregateRepository( + mhClient: mh.ManhattanCoordinator.MethodPerEndpoint, + mhDataset: String, + mhAppId: String + ): Repository[Long, Option[uss.UserSession]] = { + val keyTransformer = Injection + .connect[Long, Array[Byte]] + .toByteBufferTransformer() + + val valueInjection = ThriftCodec + .toCompact[uss.UserSession] + + KeyValueRepository.singular( + new ManhattanKeyValueRepository( + client = mhClient, + keyTransformer = keyTransformer, + valueTransformer = valueInjection.toByteBufferTransformer().flip, + appId = mhAppId, + dataset = mhDataset, + timeoutInMillis = 100 + ) + ) + + } + +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/ManhattanTweetImpressionStoreModule.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/ManhattanTweetImpressionStoreModule.scala new file mode 100644 index 0000000000..1f6ae824a1 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/ManhattanTweetImpressionStoreModule.scala @@ -0,0 +1,52 @@ +package com.twitter.home_mixer.module + +import com.google.inject.Provides +import com.twitter.conversions.DurationOps._ +import com.twitter.finagle.mtls.authentication.ServiceIdentifier +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.inject.TwitterModule +import com.twitter.storage.client.manhattan.kv.Guarantee +import com.twitter.storehaus_internal.manhattan.ManhattanClusters +import com.twitter.timelines.clients.manhattan.mhv3.ManhattanClientBuilder +import com.twitter.timelines.impressionstore.store.ManhattanTweetImpressionStoreClientConfig +import com.twitter.timelines.impressionstore.store.ManhattanTweetImpressionStoreClient +import javax.inject.Singleton + +object ManhattanTweetImpressionStoreModule extends TwitterModule { + + private val ProdAppId = "timelines_tweet_impression_store_v2" + private val ProdDataset = "timelines_tweet_impressions_v2" + private val StagingAppId = "timelines_tweet_impression_store_staging" + private val StagingDataset = "timelines_tweet_impressions_staging" + private val StatsScope = "manhattanTweetImpressionStoreClient" + private val DefaultTTL = 2.days + + @Provides + @Singleton + def providesManhattanTweetImpressionStoreClient( + serviceIdentifier: ServiceIdentifier, + statsReceiver: StatsReceiver + ): ManhattanTweetImpressionStoreClient = { + + val (appId, dataset) = serviceIdentifier.environment.toLowerCase match { + case "prod" => (ProdAppId, ProdDataset) + case _ => (StagingAppId, StagingDataset) + } + + val config = ManhattanTweetImpressionStoreClientConfig( + cluster = ManhattanClusters.nash, + appId = appId, + dataset = dataset, + statsScope = StatsScope, + defaultGuarantee = Guarantee.SoftDcReadMyWrites, + defaultMaxTimeout = 100.milliseconds, + maxRetryCount = 2, + isReadOnly = false, + serviceIdentifier = serviceIdentifier, + ttl = DefaultTTL + ) + + val manhattanEndpoint = ManhattanClientBuilder.buildManhattanEndpoint(config, statsReceiver) + ManhattanTweetImpressionStoreClient(config, manhattanEndpoint, statsReceiver) + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/MemcachedFeatureRepositoryModule.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/MemcachedFeatureRepositoryModule.scala new file mode 100644 index 0000000000..2f5ce1f5d7 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/MemcachedFeatureRepositoryModule.scala @@ -0,0 +1,113 @@ +package com.twitter.home_mixer.module + +import com.google.inject.Provides +import com.twitter.conversions.DurationOps._ +import com.twitter.finagle.Memcached +import com.twitter.finagle.mtls.authentication.ServiceIdentifier +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.home_mixer.param.HomeMixerInjectionNames.HomeAuthorFeaturesCacheClient +import com.twitter.home_mixer.param.HomeMixerInjectionNames.RealTimeInteractionGraphUserVertexClient +import com.twitter.home_mixer.param.HomeMixerInjectionNames.TimelinesRealTimeAggregateClient +import com.twitter.home_mixer.param.HomeMixerInjectionNames.TwhinAuthorFollow20200101FeatureCacheClient +import com.twitter.inject.TwitterModule +import com.twitter.product_mixer.shared_library.memcached_client.MemcachedClientBuilder +import com.twitter.servo.cache.FinagleMemcacheFactory +import com.twitter.servo.cache.Memcache +import javax.inject.Named +import javax.inject.Singleton + +object MemcachedFeatureRepositoryModule extends TwitterModule { + + // This must match the respective parameter on the write path. Note that servo sets a different + // hasher by default. See [[com.twitter.hashing.KeyHasher]] for the list of other available + // hashers. + private val memcacheKeyHasher = "ketama" + + @Provides + @Singleton + @Named(TimelinesRealTimeAggregateClient) + def providesTimelinesRealTimeAggregateClient( + serviceIdentifier: ServiceIdentifier, + statsReceiver: StatsReceiver + ): Memcache = { + val rawClient = MemcachedClientBuilder.buildRawMemcachedClient( + numTries = 1, + requestTimeout = 150.milliseconds, + globalTimeout = 150.milliseconds, + connectTimeout = 200.milliseconds, + acquisitionTimeout = 200.milliseconds, + serviceIdentifier = serviceIdentifier, + statsReceiver = statsReceiver + ) + + buildMemcacheClient(rawClient, "/s/cache/timelines_real_time_aggregates:twemcaches") + } + + @Provides + @Singleton + @Named(HomeAuthorFeaturesCacheClient) + def providesHomeAuthorFeaturesCacheClient( + serviceIdentifier: ServiceIdentifier, + statsReceiver: StatsReceiver + ): Memcache = { + val cacheClient = MemcachedClientBuilder.buildRawMemcachedClient( + numTries = 1, + requestTimeout = 50.milliseconds, + globalTimeout = 50.milliseconds, + connectTimeout = 200.milliseconds, + acquisitionTimeout = 200.milliseconds, + serviceIdentifier = serviceIdentifier, + statsReceiver = statsReceiver + ) + + buildMemcacheClient(cacheClient, "/s/cache/timelines_author_features:twemcaches") + } + + @Provides + @Singleton + @Named(TwhinAuthorFollow20200101FeatureCacheClient) + def providesTwhinAuthorFollow20200101FeatureCacheClient( + serviceIdentifier: ServiceIdentifier, + statsReceiver: StatsReceiver + ): Memcache = { + val cacheClient = MemcachedClientBuilder.buildRawMemcachedClient( + numTries = 1, + requestTimeout = 50.milliseconds, + globalTimeout = 50.milliseconds, + connectTimeout = 200.milliseconds, + acquisitionTimeout = 200.milliseconds, + serviceIdentifier = serviceIdentifier, + statsReceiver = statsReceiver + ) + + buildMemcacheClient(cacheClient, "/s/cache/home_twhin_author_features:twemcaches") + } + + @Provides + @Singleton + @Named(RealTimeInteractionGraphUserVertexClient) + def providesRealTimeInteractionGraphUserVertexClient( + serviceIdentifier: ServiceIdentifier, + statsReceiver: StatsReceiver + ): Memcache = { + val cacheClient = MemcachedClientBuilder.buildRawMemcachedClient( + numTries = 1, + requestTimeout = 100.milliseconds, + globalTimeout = 100.milliseconds, + connectTimeout = 200.milliseconds, + acquisitionTimeout = 200.milliseconds, + serviceIdentifier = serviceIdentifier, + statsReceiver = statsReceiver + ) + + buildMemcacheClient(cacheClient, "/s/cache/realtime_interactive_graph_prod_v2:twemcaches") + } + + private def buildMemcacheClient(cacheClient: Memcached.Client, dest: String): Memcache = + FinagleMemcacheFactory( + client = cacheClient, + dest = dest, + hashName = memcacheKeyHasher + )() + +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/OptimizedStratoClientModule.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/OptimizedStratoClientModule.scala new file mode 100644 index 0000000000..7575685f1f --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/OptimizedStratoClientModule.scala @@ -0,0 +1,46 @@ +package com.twitter.home_mixer.module + +import com.google.inject.Provides +import com.twitter.conversions.DurationOps._ +import com.twitter.finagle.mtls.authentication.ServiceIdentifier +import com.twitter.finagle.service.Retries +import com.twitter.finagle.service.RetryPolicy +import com.twitter.finagle.ssl.OpportunisticTls +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.home_mixer.param.HomeMixerInjectionNames.BatchedStratoClientWithModerateTimeout +import com.twitter.inject.TwitterModule +import com.twitter.strato.client.Client +import com.twitter.strato.client.Strato +import com.twitter.util.Try +import javax.inject.Named +import javax.inject.Singleton + +object OptimizedStratoClientModule extends TwitterModule { + + private val ModerateStratoServerClientRequestTimeout = 150.millis + + private val DefaultRetryPartialFunction: PartialFunction[Try[Nothing], Boolean] = + RetryPolicy.TimeoutAndWriteExceptionsOnly + .orElse(RetryPolicy.ChannelClosedExceptionsOnly) + + protected def mkRetryPolicy(tries: Int): RetryPolicy[Try[Nothing]] = + RetryPolicy.tries(tries, DefaultRetryPartialFunction) + + @Singleton + @Provides + @Named(BatchedStratoClientWithModerateTimeout) + def providesStratoClient( + serviceIdentifier: ServiceIdentifier, + statsReceiver: StatsReceiver + ): Client = { + Strato.client + .withMutualTls(serviceIdentifier, opportunisticLevel = OpportunisticTls.Required) + .withSession.acquisitionTimeout(150.milliseconds) + .withRequestTimeout(ModerateStratoServerClientRequestTimeout) + .withPerRequestTimeout(ModerateStratoServerClientRequestTimeout) + .withRpcBatchSize(5) + .configured(Retries.Policy(mkRetryPolicy(1))) + .withStatsReceiver(statsReceiver.scope("strato_client")) + .build() + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/PeopleDiscoveryServiceModule.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/PeopleDiscoveryServiceModule.scala new file mode 100644 index 0000000000..47353afa71 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/PeopleDiscoveryServiceModule.scala @@ -0,0 +1,35 @@ +package com.twitter.home_mixer.module + +import com.twitter.conversions.DurationOps._ +import com.twitter.finagle.thriftmux.MethodBuilder +import com.twitter.finatra.mtls.thriftmux.modules.MtlsClient +import com.twitter.inject.Injector +import com.twitter.inject.thrift.modules.ThriftMethodBuilderClientModule +import com.twitter.peoplediscovery.api.thriftscala.ThriftPeopleDiscoveryService +import com.twitter.util.Duration + +/** + * Copy of com.twitter.product_mixer.component_library.module.PeopleDiscoveryServiceModule + */ +object PeopleDiscoveryServiceModule + extends ThriftMethodBuilderClientModule[ + ThriftPeopleDiscoveryService.ServicePerEndpoint, + ThriftPeopleDiscoveryService.MethodPerEndpoint + ] + with MtlsClient { + + override val label: String = "people-discovery-api" + + override val dest: String = "/s/people-discovery-api/people-discovery-api:thrift" + + override protected def configureMethodBuilder( + injector: Injector, + methodBuilder: MethodBuilder + ): MethodBuilder = { + methodBuilder + .withTimeoutPerRequest(350.millis) + .withTimeoutTotal(350.millis) + } + + override protected def sessionAcquisitionTimeout: Duration = 500.milliseconds +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/PipelineFailureExceptionMapper.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/PipelineFailureExceptionMapper.scala new file mode 100644 index 0000000000..65f0ac7bd4 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/PipelineFailureExceptionMapper.scala @@ -0,0 +1,29 @@ +package com.twitter.home_mixer.module + +import com.twitter.finatra.thrift.exceptions.ExceptionMapper +import com.twitter.home_mixer.{thriftscala => t} +import com.twitter.inject.Logging +import com.twitter.product_mixer.core.pipeline.pipeline_failure.PipelineFailure +import com.twitter.product_mixer.core.pipeline.pipeline_failure.ProductDisabled +import com.twitter.scrooge.ThriftException +import com.twitter.util.Future +import javax.inject.Singleton + +@Singleton +class PipelineFailureExceptionMapper + extends ExceptionMapper[PipelineFailure, ThriftException] + with Logging { + + def handleException(throwable: PipelineFailure): Future[ThriftException] = { + throwable match { + // SliceService (unlike UrtService) throws an exception when the requested product is disabled + case PipelineFailure(ProductDisabled, reason, _, _) => + Future.exception( + t.ValidationExceptionList(errors = + Seq(t.ValidationException(t.ValidationErrorCode.ProductDisabled, reason)))) + case _ => + error("Unhandled PipelineFailure", throwable) + Future.exception(throwable) + } + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/RealGraphInNetworkScoresModule.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/RealGraphInNetworkScoresModule.scala new file mode 100644 index 0000000000..7dc6a072d6 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/RealGraphInNetworkScoresModule.scala @@ -0,0 +1,26 @@ +package com.twitter.home_mixer.module + +import com.google.inject.Provides +import com.google.inject.name.Named +import com.twitter.home_mixer.param.HomeMixerInjectionNames.RealGraphInNetworkScores +import com.twitter.home_mixer.param.HomeMixerInjectionNames.RealGraphManhattanEndpoint +import com.twitter.home_mixer.store.RealGraphInNetworkScoresStore +import com.twitter.inject.TwitterModule +import com.twitter.storage.client.manhattan.kv.ManhattanKVEndpoint +import com.twitter.storehaus.ReadableStore +import com.twitter.timelines.util.CommonTypes.ViewerId +import com.twitter.wtf.candidate.thriftscala.Candidate + +import javax.inject.Singleton + +object RealGraphInNetworkScoresModule extends TwitterModule { + + @Provides + @Singleton + @Named(RealGraphInNetworkScores) + def providesRealGraphInNetworkScoresFeaturesStore( + @Named(RealGraphManhattanEndpoint) realGraphInNetworkScoresManhattanKVEndpoint: ManhattanKVEndpoint + ): ReadableStore[ViewerId, Seq[Candidate]] = { + new RealGraphInNetworkScoresStore(realGraphInNetworkScoresManhattanKVEndpoint) + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/RealtimeAggregateFeatureRepositoryModule.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/RealtimeAggregateFeatureRepositoryModule.scala new file mode 100644 index 0000000000..19512e6639 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/RealtimeAggregateFeatureRepositoryModule.scala @@ -0,0 +1,253 @@ +package com.twitter.home_mixer.module + +import com.google.inject.Provides +import com.google.inject.name.Named +import com.twitter.bijection.Injection +import com.twitter.bijection.scrooge.BinaryScalaCodec +import com.twitter.bijection.thrift.ThriftCodec +import com.twitter.home_mixer.param.HomeMixerInjectionNames.EngagementsReceivedByAuthorCache +import com.twitter.home_mixer.param.HomeMixerInjectionNames.RealTimeInteractionGraphUserVertexCache +import com.twitter.home_mixer.param.HomeMixerInjectionNames.RealTimeInteractionGraphUserVertexClient +import com.twitter.home_mixer.param.HomeMixerInjectionNames.TimelinesRealTimeAggregateClient +import com.twitter.home_mixer.param.HomeMixerInjectionNames.TopicCountryEngagementCache +import com.twitter.home_mixer.param.HomeMixerInjectionNames.TopicEngagementCache +import com.twitter.home_mixer.param.HomeMixerInjectionNames.TweetCountryEngagementCache +import com.twitter.home_mixer.param.HomeMixerInjectionNames.TweetEngagementCache +import com.twitter.home_mixer.param.HomeMixerInjectionNames.TwitterListEngagementCache +import com.twitter.home_mixer.param.HomeMixerInjectionNames.UserAuthorEngagementCache +import com.twitter.home_mixer.param.HomeMixerInjectionNames.UserEngagementCache +import com.twitter.home_mixer.param.HomeMixerInjectionNames.UserTopicEngagementForNewUserCache +import com.twitter.home_mixer.util.InjectionTransformerImplicits._ +import com.twitter.inject.TwitterModule +import com.twitter.ml.api.DataRecord +import com.twitter.ml.api.Feature +import com.twitter.ml.{api => ml} +import com.twitter.servo.cache.KeyValueTransformingReadCache +import com.twitter.servo.cache.Memcache +import com.twitter.servo.cache.ReadCache +import com.twitter.servo.util.Transformer +import com.twitter.storehaus_internal.memcache.MemcacheHelper +import com.twitter.summingbird.batch.Batcher +import com.twitter.summingbird_internal.bijection.BatchPairImplicits +import com.twitter.timelines.data_processing.ml_util.aggregation_framework.AggregationKey +import com.twitter.timelines.data_processing.ml_util.aggregation_framework.AggregationKeyInjection +import com.twitter.wtf.real_time_interaction_graph.{thriftscala => ig} + +import javax.inject.Singleton + +object RealtimeAggregateFeatureRepositoryModule + extends TwitterModule + with RealtimeAggregateHelpers { + + private val authorIdFeature = new Feature.Discrete("entities.source_author_id") + private val countryCodeFeature = new Feature.Text("geo.user_location.country_code") + private val listIdFeature = new Feature.Discrete("list.id") + private val userIdFeature = new Feature.Discrete("meta.user_id") + private val topicIdFeature = new Feature.Discrete("entities.topic_id") + private val tweetIdFeature = new Feature.Discrete("entities.source_tweet_id") + + @Provides + @Singleton + @Named(UserTopicEngagementForNewUserCache) + def providesUserTopicEngagementForNewUserCache( + @Named(TimelinesRealTimeAggregateClient) client: Memcache + ): ReadCache[(Long, Long), ml.DataRecord] = { + new KeyValueTransformingReadCache( + client, + dataRecordValueTransformer, + keyTransformD2(userIdFeature, topicIdFeature) + ) + } + + @Provides + @Singleton + @Named(TwitterListEngagementCache) + def providesTwitterListEngagementCache( + @Named(TimelinesRealTimeAggregateClient) client: Memcache + ): ReadCache[Long, ml.DataRecord] = { + new KeyValueTransformingReadCache( + client, + dataRecordValueTransformer, + keyTransformD1(listIdFeature) + ) + } + + @Provides + @Singleton + @Named(TopicEngagementCache) + def providesTopicEngagementCache( + @Named(TimelinesRealTimeAggregateClient) client: Memcache + ): ReadCache[Long, ml.DataRecord] = { + new KeyValueTransformingReadCache( + client, + dataRecordValueTransformer, + keyTransformD1(topicIdFeature) + ) + } + + @Provides + @Singleton + @Named(UserAuthorEngagementCache) + def providesUserAuthorEngagementCache( + @Named(TimelinesRealTimeAggregateClient) client: Memcache + ): ReadCache[(Long, Long), ml.DataRecord] = { + new KeyValueTransformingReadCache( + client, + dataRecordValueTransformer, + keyTransformD2(userIdFeature, authorIdFeature) + ) + } + + @Provides + @Singleton + @Named(UserEngagementCache) + def providesUserEngagementCache( + @Named(TimelinesRealTimeAggregateClient) client: Memcache + ): ReadCache[Long, ml.DataRecord] = { + new KeyValueTransformingReadCache( + client, + dataRecordValueTransformer, + keyTransformD1(userIdFeature) + ) + } + + @Provides + @Singleton + @Named(TweetCountryEngagementCache) + def providesTweetCountryEngagementCache( + @Named(TimelinesRealTimeAggregateClient) client: Memcache + ): ReadCache[(Long, String), ml.DataRecord] = { + + new KeyValueTransformingReadCache( + client, + dataRecordValueTransformer, + keyTransformD1T1(tweetIdFeature, countryCodeFeature) + ) + } + + @Provides + @Singleton + @Named(TweetEngagementCache) + def providesTweetEngagementCache( + @Named(TimelinesRealTimeAggregateClient) client: Memcache + ): ReadCache[Long, ml.DataRecord] = { + new KeyValueTransformingReadCache( + client, + dataRecordValueTransformer, + keyTransformD1(tweetIdFeature) + ) + } + + @Provides + @Singleton + @Named(EngagementsReceivedByAuthorCache) + def providesEngagementsReceivedByAuthorCache( + @Named(TimelinesRealTimeAggregateClient) client: Memcache + ): ReadCache[Long, ml.DataRecord] = { + new KeyValueTransformingReadCache( + client, + dataRecordValueTransformer, + keyTransformD1(authorIdFeature) + ) + } + + @Provides + @Singleton + @Named(TopicCountryEngagementCache) + def providesTopicCountryEngagementCache( + @Named(TimelinesRealTimeAggregateClient) client: Memcache + ): ReadCache[(Long, String), ml.DataRecord] = { + new KeyValueTransformingReadCache( + client, + dataRecordValueTransformer, + keyTransformD1T1(topicIdFeature, countryCodeFeature) + ) + } + + @Provides + @Singleton + @Named(RealTimeInteractionGraphUserVertexCache) + def providesRealTimeInteractionGraphUserVertexCache( + @Named(RealTimeInteractionGraphUserVertexClient) client: Memcache + ): ReadCache[Long, ig.UserVertex] = { + + val valueTransformer = BinaryScalaCodec(ig.UserVertex).toByteArrayTransformer() + + val underlyingKey: Long => String = { + val cacheKeyPrefix = "user_vertex" + val defaultBatchID = Batcher.unit.currentBatch + val batchPairInjection = BatchPairImplicits.keyInjection(Injection.connect[Long, Array[Byte]]) + MemcacheHelper + .keyEncoder(cacheKeyPrefix)(batchPairInjection) + .compose((k: Long) => (k, defaultBatchID)) + } + + new KeyValueTransformingReadCache( + client, + valueTransformer, + underlyingKey + ) + } +} + +trait RealtimeAggregateHelpers { + + private def customKeyBuilder[K](prefix: String, f: K => Array[Byte]): K => String = { + // intentionally not implementing injection inverse because it is never used + def g(arr: Array[Byte]) = ??? + + MemcacheHelper.keyEncoder(prefix)(Injection.build(f)(g)) + } + + private val keyEncoder: AggregationKey => String = { + val cacheKeyPrefix = "" + val defaultBatchID = Batcher.unit.currentBatch + + val batchPairInjection = BatchPairImplicits.keyInjection(AggregationKeyInjection) + customKeyBuilder(cacheKeyPrefix, batchPairInjection) + .compose((k: AggregationKey) => (k, defaultBatchID)) + } + + protected def keyTransformD1(f1: Feature.Discrete)(key: Long): String = { + val aggregationKey = AggregationKey( + Map(f1.getFeatureId -> key), + Map.empty + ) + + keyEncoder(aggregationKey) + } + + protected def keyTransformD2( + f1: Feature.Discrete, + f2: Feature.Discrete + )( + keys: (Long, Long) + ): String = { + val (k1, k2) = keys + val aggregationKey = AggregationKey( + Map(f1.getFeatureId -> k1, f2.getFeatureId -> k2), + Map.empty + ) + + keyEncoder(aggregationKey) + } + + protected def keyTransformD1T1( + f1: Feature.Discrete, + f2: Feature.Text + )( + keys: (Long, String) + ): String = { + val (k1, k2) = keys + val aggregationKey = AggregationKey( + Map(f1.getFeatureId -> k1), + Map(f2.getFeatureId -> k2) + ) + + keyEncoder(aggregationKey) + } + + protected val dataRecordValueTransformer: Transformer[DataRecord, Array[Byte]] = ThriftCodec + .toCompact[ml.DataRecord] + .toByteArrayTransformer() +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/ScoredTweetsMemcacheModule.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/ScoredTweetsMemcacheModule.scala new file mode 100644 index 0000000000..b8ac940b1c --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/ScoredTweetsMemcacheModule.scala @@ -0,0 +1,65 @@ +package com.twitter.home_mixer.module + +import com.google.inject.Provides +import com.twitter.conversions.DurationOps._ +import com.twitter.finagle.mtls.authentication.ServiceIdentifier +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.home_mixer.{thriftscala => t} +import com.twitter.inject.TwitterModule +import com.twitter.product_mixer.shared_library.memcached_client.MemcachedClientBuilder +import com.twitter.servo.cache.FinagleMemcache +import com.twitter.servo.cache.KeyTransformer +import com.twitter.servo.cache.KeyValueTransformingTtlCache +import com.twitter.servo.cache.ObservableTtlCache +import com.twitter.servo.cache.Serializer +import com.twitter.servo.cache.ThriftSerializer +import com.twitter.servo.cache.TtlCache +import com.twitter.timelines.model.UserId +import org.apache.thrift.protocol.TCompactProtocol + +import javax.inject.Singleton + +object ScoredTweetsMemcacheModule extends TwitterModule { + + private val ScopeName = "ScoredTweetsCache" + private val ProdDestName = "/srv#/prod/local/cache/home_scored_tweets:twemcaches" + private val StagingDestName = "/srv#/test/local/cache/twemcache_home_scored_tweets:twemcaches" + private val cachedScoredTweetsSerializer: Serializer[t.CachedScoredTweets] = + new ThriftSerializer[t.CachedScoredTweets](t.CachedScoredTweets, new TCompactProtocol.Factory()) + private val userIdKeyTransformer: KeyTransformer[UserId] = (userId: UserId) => userId.toString + + @Singleton + @Provides + def providesScoredTweetsCache( + serviceIdentifier: ServiceIdentifier, + statsReceiver: StatsReceiver + ): TtlCache[UserId, t.CachedScoredTweets] = { + val destName = serviceIdentifier.environment.toLowerCase match { + case "prod" => ProdDestName + case _ => StagingDestName + } + val client = MemcachedClientBuilder.buildMemcachedClient( + destName = destName, + numTries = 2, + requestTimeout = 200.milliseconds, + globalTimeout = 400.milliseconds, + connectTimeout = 100.milliseconds, + acquisitionTimeout = 100.milliseconds, + serviceIdentifier = serviceIdentifier, + statsReceiver = statsReceiver.scope(ScopeName) + ) + val underlyingCache = new FinagleMemcache(client) + val baseCache: KeyValueTransformingTtlCache[UserId, String, t.CachedScoredTweets, Array[Byte]] = + new KeyValueTransformingTtlCache( + underlyingCache = underlyingCache, + transformer = cachedScoredTweetsSerializer, + underlyingKey = userIdKeyTransformer + ) + ObservableTtlCache( + underlyingCache = baseCache, + statsReceiver = statsReceiver, + windowSize = 1000L, + name = ScopeName + ) + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/ScribeEventPublisherModule.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/ScribeEventPublisherModule.scala new file mode 100644 index 0000000000..9c74fb1c3e --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/ScribeEventPublisherModule.scala @@ -0,0 +1,125 @@ +package com.twitter.home_mixer.module + +import com.google.inject.Provides +import com.twitter.clientapp.{thriftscala => ca} +import com.twitter.home_mixer.param.HomeMixerFlagName.ScribeClientEventsFlag +import com.twitter.home_mixer.param.HomeMixerFlagName.ScribeServedCommonFeaturesAndCandidateFeaturesFlag +import com.twitter.home_mixer.param.HomeMixerFlagName.ScribeServedEntriesFlag +import com.twitter.home_mixer.param.HomeMixerInjectionNames.CandidateFeaturesScribeEventPublisher +import com.twitter.home_mixer.param.HomeMixerInjectionNames.CommonFeaturesScribeEventPublisher +import com.twitter.home_mixer.param.HomeMixerInjectionNames.MinimumFeaturesScribeEventPublisher +import com.twitter.inject.TwitterModule +import com.twitter.inject.annotations.Flag +import com.twitter.logpipeline.client.EventPublisherManager +import com.twitter.logpipeline.client.common.EventPublisher +import com.twitter.logpipeline.client.serializers.EventLogMsgTBinarySerializer +import com.twitter.logpipeline.client.serializers.EventLogMsgThriftStructSerializer +import com.twitter.timelines.suggests.common.poly_data_record.{thriftjava => pldr} +import com.twitter.timelines.timeline_logging.{thriftscala => tl} + +import javax.inject.Named +import javax.inject.Singleton + +object ScribeEventPublisherModule extends TwitterModule { + + val InMemoryBufferSize = 10000 + val ClientEventLogCategory = "client_event" + val ServedEntriesLogCategory = "home_timeline_served_entries" + val ServedCommonFeaturesLogCategory = "tq_served_common_features_offline" + val ServedCandidateFeaturesLogCategory = "tq_served_candidate_features_offline" + val ServedMinimumFeaturesLogCategory = "tq_served_minimum_features_offline" + + @Provides + @Singleton + def providesClientEventsScribeEventPublisher( + @Flag(ScribeClientEventsFlag) sendToScribe: Boolean + ): EventPublisher[ca.LogEvent] = { + val serializer = EventLogMsgThriftStructSerializer.getNewSerializer[ca.LogEvent]() + + if (sendToScribe) + EventPublisherManager.buildScribeLogPipelinePublisher(ClientEventLogCategory, serializer) + else + EventPublisherManager.buildInMemoryPublisher( + ClientEventLogCategory, + serializer, + InMemoryBufferSize + ) + } + + @Provides + @Singleton + @Named(CommonFeaturesScribeEventPublisher) + def providesCommonFeaturesScribeEventPublisher( + @Flag(ScribeServedCommonFeaturesAndCandidateFeaturesFlag) sendToScribe: Boolean + ): EventPublisher[pldr.PolyDataRecord] = { + val serializer = EventLogMsgTBinarySerializer.getNewSerializer + + if (sendToScribe) + EventPublisherManager.buildScribeLogPipelinePublisher( + ServedCommonFeaturesLogCategory, + serializer) + else + EventPublisherManager.buildInMemoryPublisher( + ServedCommonFeaturesLogCategory, + serializer, + InMemoryBufferSize + ) + } + + @Provides + @Singleton + @Named(CandidateFeaturesScribeEventPublisher) + def providesCandidateFeaturesScribeEventPublisher( + @Flag(ScribeServedCommonFeaturesAndCandidateFeaturesFlag) sendToScribe: Boolean + ): EventPublisher[pldr.PolyDataRecord] = { + val serializer = EventLogMsgTBinarySerializer.getNewSerializer + + if (sendToScribe) + EventPublisherManager.buildScribeLogPipelinePublisher( + ServedCandidateFeaturesLogCategory, + serializer) + else + EventPublisherManager.buildInMemoryPublisher( + ServedCandidateFeaturesLogCategory, + serializer, + InMemoryBufferSize + ) + } + + @Provides + @Singleton + @Named(MinimumFeaturesScribeEventPublisher) + def providesMinimumFeaturesScribeEventPublisher( + @Flag(ScribeServedCommonFeaturesAndCandidateFeaturesFlag) sendToScribe: Boolean + ): EventPublisher[pldr.PolyDataRecord] = { + val serializer = EventLogMsgTBinarySerializer.getNewSerializer + + if (sendToScribe) + EventPublisherManager.buildScribeLogPipelinePublisher( + ServedMinimumFeaturesLogCategory, + serializer) + else + EventPublisherManager.buildInMemoryPublisher( + ServedMinimumFeaturesLogCategory, + serializer, + InMemoryBufferSize + ) + } + + @Provides + @Singleton + def providesServedEntriesScribeEventPublisher( + @Flag(ScribeServedEntriesFlag) sendToScribe: Boolean + ): EventPublisher[tl.Timeline] = { + val serializer = EventLogMsgThriftStructSerializer.getNewSerializer[tl.Timeline]() + + if (sendToScribe) + EventPublisherManager.buildScribeLogPipelinePublisher(ServedEntriesLogCategory, serializer) + else + EventPublisherManager.buildInMemoryPublisher( + ServedEntriesLogCategory, + serializer, + InMemoryBufferSize + ) + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/SimClustersRecentEngagementsClientModule.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/SimClustersRecentEngagementsClientModule.scala new file mode 100644 index 0000000000..7e819d51e0 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/SimClustersRecentEngagementsClientModule.scala @@ -0,0 +1,23 @@ +package com.twitter.home_mixer.module + +import com.google.inject.Provides +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.home_mixer.param.HomeMixerInjectionNames.BatchedStratoClientWithModerateTimeout +import com.twitter.inject.TwitterModule +import com.twitter.strato.client.Client +import com.twitter.timelines.clients.strato.twistly.SimClustersRecentEngagementSimilarityClient +import com.twitter.timelines.clients.strato.twistly.SimClustersRecentEngagementSimilarityClientImpl +import javax.inject.Named +import javax.inject.Singleton + +object SimClustersRecentEngagementsClientModule extends TwitterModule { + @Singleton + @Provides + def providesSimilarityClient( + @Named(BatchedStratoClientWithModerateTimeout) + stratoClient: Client, + statsReceiver: StatsReceiver + ): SimClustersRecentEngagementSimilarityClient = { + new SimClustersRecentEngagementSimilarityClientImpl(stratoClient, statsReceiver) + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/StaleTweetsCacheModule.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/StaleTweetsCacheModule.scala new file mode 100644 index 0000000000..070de320a3 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/StaleTweetsCacheModule.scala @@ -0,0 +1,37 @@ +package com.twitter.home_mixer.module + +import com.google.inject.Provides +import com.google.inject.name.Named +import com.twitter.conversions.DurationOps._ +import com.twitter.finagle.memcached.{Client => MemcachedClient} +import com.twitter.finagle.mtls.authentication.ServiceIdentifier +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.hashing.KeyHasher +import com.twitter.home_mixer.param.HomeMixerInjectionNames.StaleTweetsCache +import com.twitter.inject.TwitterModule +import com.twitter.product_mixer.shared_library.memcached_client.MemcachedClientBuilder +import javax.inject.Singleton + +object StaleTweetsCacheModule extends TwitterModule { + + @Singleton + @Provides + @Named(StaleTweetsCache) + def providesCache( + serviceIdentifier: ServiceIdentifier, + statsReceiver: StatsReceiver + ): MemcachedClient = { + MemcachedClientBuilder.buildMemcachedClient( + destName = "/srv#/prod/local/cache/staletweetscache:twemcaches", + numTries = 3, + requestTimeout = 200.milliseconds, + globalTimeout = 500.milliseconds, + connectTimeout = 200.milliseconds, + acquisitionTimeout = 200.milliseconds, + serviceIdentifier = serviceIdentifier, + statsReceiver = statsReceiver, + failureAccrualPolicy = None, + keyHasher = Some(KeyHasher.FNV1_32) + ) + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/ThriftFeatureRepositoryModule.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/ThriftFeatureRepositoryModule.scala new file mode 100644 index 0000000000..5a2a8701a3 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/ThriftFeatureRepositoryModule.scala @@ -0,0 +1,375 @@ +package com.twitter.home_mixer.module + +import com.google.inject.Provides +import com.twitter.conversions.DurationOps._ +import com.twitter.conversions.PercentOps._ +import com.twitter.finagle.mtls.authentication.ServiceIdentifier +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.finagle.thrift.ClientId +import com.twitter.graph_feature_service.{thriftscala => gfs} +import com.twitter.home_mixer.param.HomeMixerInjectionNames.EarlybirdRepository +import com.twitter.home_mixer.param.HomeMixerInjectionNames.GraphTwoHopRepository +import com.twitter.home_mixer.param.HomeMixerInjectionNames.InterestsThriftServiceClient +import com.twitter.home_mixer.param.HomeMixerInjectionNames.TweetypieContentRepository +import com.twitter.home_mixer.param.HomeMixerInjectionNames.UserFollowedTopicIdsRepository +import com.twitter.home_mixer.param.HomeMixerInjectionNames.UtegSocialProofRepository +import com.twitter.home_mixer.util.earlybird.EarlybirdRequestUtil +import com.twitter.home_mixer.util.tweetypie.RequestFields +import com.twitter.inject.TwitterModule +import com.twitter.interests.{thriftscala => int} +import com.twitter.product_mixer.shared_library.memcached_client.MemcachedClientBuilder +import com.twitter.product_mixer.shared_library.thrift_client.FinagleThriftClientBuilder +import com.twitter.product_mixer.shared_library.thrift_client.Idempotent +import com.twitter.recos.recos_common.{thriftscala => rc} +import com.twitter.recos.user_tweet_entity_graph.{thriftscala => uteg} +import com.twitter.search.earlybird.{thriftscala => eb} +import com.twitter.servo.cache.Cached +import com.twitter.servo.cache.CachedSerializer +import com.twitter.servo.cache.FinagleMemcacheFactory +import com.twitter.servo.cache.MemcacheCacheFactory +import com.twitter.servo.cache.NonLockingCache +import com.twitter.servo.cache.ThriftSerializer +import com.twitter.servo.keyvalue.KeyValueResultBuilder +import com.twitter.servo.repository.CachingKeyValueRepository +import com.twitter.servo.repository.ChunkingStrategy +import com.twitter.servo.repository.KeyValueRepository +import com.twitter.servo.repository.KeyValueResult +import com.twitter.servo.repository.keysAsQuery +import com.twitter.spam.rtf.{thriftscala => sp} +import com.twitter.tweetypie.{thriftscala => tp} +import com.twitter.util.Future +import com.twitter.util.Return +import javax.inject.Named +import javax.inject.Singleton +import org.apache.thrift.protocol.TCompactProtocol + +object ThriftFeatureRepositoryModule extends TwitterModule { + + private val DefaultRPCChunkSize = 50 + private val GFSInteractionIdsLimit = 10 + + type EarlybirdQuery = (Seq[Long], Long) + type UtegQuery = (Seq[Long], (Long, Map[Long, Double])) + + @Provides + @Singleton + @Named(InterestsThriftServiceClient) + def providesInterestsThriftServiceClient( + clientId: ClientId, + serviceIdentifier: ServiceIdentifier, + statsReceiver: StatsReceiver + ): int.InterestsThriftService.MethodPerEndpoint = { + FinagleThriftClientBuilder + .buildFinagleMethodPerEndpoint[ + int.InterestsThriftService.ServicePerEndpoint, + int.InterestsThriftService.MethodPerEndpoint]( + serviceIdentifier = serviceIdentifier, + clientId = clientId, + dest = "/s/interests-thrift-service/interests-thrift-service", + label = "interests", + statsReceiver = statsReceiver, + idempotency = Idempotent(1.percent), + timeoutPerRequest = 100.milliseconds, + timeoutTotal = 100.milliseconds + ) + } + + @Provides + @Singleton + @Named(UserFollowedTopicIdsRepository) + def providesUserFollowedTopicIdsRepository( + @Named(InterestsThriftServiceClient) client: int.InterestsThriftService.MethodPerEndpoint + ): KeyValueRepository[Seq[Long], Long, Seq[Long]] = { + + val lookupContext = Some( + int.ExplicitInterestLookupContext(Some(Seq(int.InterestRelationType.Followed))) + ) + + def lookup(userId: Long): Future[Seq[Long]] = { + client.getUserExplicitInterests(userId, lookupContext).map { interests => + interests.flatMap { + _.interestId match { + case int.InterestId.SemanticCore(semanticCoreInterest) => Some(semanticCoreInterest.id) + case _ => None + } + } + } + } + + val keyValueRepository = toRepository(lookup) + + keyValueRepository + } + + @Provides + @Singleton + @Named(UtegSocialProofRepository) + def providesUtegSocialProofRepository( + clientId: ClientId, + serviceIdentifier: ServiceIdentifier, + statsReceiver: StatsReceiver + ): KeyValueRepository[UtegQuery, Long, uteg.TweetRecommendation] = { + val client = FinagleThriftClientBuilder.buildFinagleMethodPerEndpoint[ + uteg.UserTweetEntityGraph.ServicePerEndpoint, + uteg.UserTweetEntityGraph.MethodPerEndpoint]( + serviceIdentifier = serviceIdentifier, + clientId = clientId, + dest = "/s/cassowary/user_tweet_entity_graph", + label = "uteg-social-proof-repo", + statsReceiver = statsReceiver, + idempotency = Idempotent(1.percent), + timeoutPerRequest = 150.milliseconds, + timeoutTotal = 250.milliseconds + ) + + val utegSocialProofTypes = Seq( + rc.SocialProofType.Favorite, + rc.SocialProofType.Retweet, + rc.SocialProofType.Reply + ) + + def lookup( + tweetIds: Seq[Long], + view: (Long, Map[Long, Double]) + ): Future[Seq[Option[uteg.TweetRecommendation]]] = { + val (userId, seedsWithWeights) = view + val socialProofRequest = uteg.SocialProofRequest( + requesterId = Some(userId), + seedsWithWeights = seedsWithWeights, + inputTweets = tweetIds, + socialProofTypes = Some(utegSocialProofTypes) + ) + client.findTweetSocialProofs(socialProofRequest).map { result => + val resultMap = result.socialProofResults.map(t => t.tweetId -> t).toMap + tweetIds.map(resultMap.get) + } + } + + toRepositoryBatchWithView(lookup, chunkSize = 200) + } + + @Provides + @Singleton + @Named(TweetypieContentRepository) + def providesTweetypieContentRepository( + clientId: ClientId, + serviceIdentifier: ServiceIdentifier, + statsReceiver: StatsReceiver + ): KeyValueRepository[Seq[Long], Long, tp.Tweet] = { + val client = FinagleThriftClientBuilder + .buildFinagleMethodPerEndpoint[ + tp.TweetService.ServicePerEndpoint, + tp.TweetService.MethodPerEndpoint]( + serviceIdentifier = serviceIdentifier, + clientId = clientId, + dest = "/s/tweetypie/tweetypie", + label = "tweetypie-content-repo", + statsReceiver = statsReceiver, + idempotency = Idempotent(1.percent), + timeoutPerRequest = 150.milliseconds, + timeoutTotal = 250.milliseconds + ) + + def lookup(tweetIds: Seq[Long]): Future[Seq[Option[tp.Tweet]]] = { + val getTweetFieldsOptions = tp.GetTweetFieldsOptions( + tweetIncludes = RequestFields.ContentFields, + includeRetweetedTweet = false, + includeQuotedTweet = false, + forUserId = None, + // Service needs to be whitelisted + // We rely on the VF at the end of serving. No need to filter now. + safetyLevel = Some(sp.SafetyLevel.FilterNone), + visibilityPolicy = tp.TweetVisibilityPolicy.NoFiltering + ) + val request = tp.GetTweetFieldsRequest( + tweetIds = tweetIds, + options = getTweetFieldsOptions + ) + client.getTweetFields(request).map { results => + results.map { + case tp.GetTweetFieldsResult(_, tp.TweetFieldsResultState.Found(found), _, _) => + Some(found.tweet) + case _ => None + } + } + } + + val keyValueRepository = toRepositoryBatch(lookup, chunkSize = 20) + + // Cache + val cacheClient = MemcachedClientBuilder.buildRawMemcachedClient( + numTries = 1, + requestTimeout = 100.milliseconds, + globalTimeout = 100.milliseconds, + connectTimeout = 200.milliseconds, + acquisitionTimeout = 200.milliseconds, + serviceIdentifier = serviceIdentifier, + statsReceiver = statsReceiver + ) + val finagleMemcacheFactory = + FinagleMemcacheFactory(cacheClient, "/s/cache/home_content_features:twemcaches") + val cacheValueTransformer = + new ThriftSerializer[tp.Tweet](tp.Tweet, new TCompactProtocol.Factory()) + val cachedSerializer = CachedSerializer.binary(cacheValueTransformer) + + val cache = MemcacheCacheFactory( + memcache = finagleMemcacheFactory(), + ttl = 48.hours + )[Long, Cached[tp.Tweet]](cachedSerializer) + + val lockingCache = new NonLockingCache(cache) + val cachedKeyValueRepository = new CachingKeyValueRepository( + keyValueRepository, + lockingCache, + keysAsQuery[Long] + ) + cachedKeyValueRepository + } + + @Provides + @Singleton + @Named(GraphTwoHopRepository) + def providesGraphTwoHopRepository( + clientId: ClientId, + serviceIdentifier: ServiceIdentifier, + statsReceiver: StatsReceiver + ): KeyValueRepository[(Seq[Long], Long), Long, Seq[gfs.IntersectionValue]] = { + val client = FinagleThriftClientBuilder + .buildFinagleMethodPerEndpoint[gfs.Server.ServicePerEndpoint, gfs.Server.MethodPerEndpoint]( + serviceIdentifier = serviceIdentifier, + clientId = clientId, + dest = "/s/cassowary/graph_feature_service-server", + label = "gfs-repo", + statsReceiver = statsReceiver, + idempotency = Idempotent(1.percent), + timeoutPerRequest = 350.milliseconds, + timeoutTotal = 500.milliseconds + ) + + def lookup( + userIds: Seq[Long], + viewerId: Long + ): Future[Seq[Option[Seq[gfs.IntersectionValue]]]] = { + val gfsIntersectionRequest = gfs.GfsPresetIntersectionRequest( + userId = viewerId, + candidateUserIds = userIds, + presetFeatureTypes = gfs.PresetFeatureTypes.HtlTwoHop, + intersectionIdLimit = Some(GFSInteractionIdsLimit) + ) + + client + .getPresetIntersection(gfsIntersectionRequest) + .map { graphFeatureServiceResponse => + val resultMap = graphFeatureServiceResponse.results + .map(result => result.candidateUserId -> result.intersectionValues).toMap + userIds.map(resultMap.get(_)) + } + } + + toRepositoryBatchWithView(lookup, chunkSize = 200) + } + + @Provides + @Singleton + @Named(EarlybirdRepository) + def providesEarlybirdSearchRepository( + client: eb.EarlybirdService.MethodPerEndpoint, + clientId: ClientId + ): KeyValueRepository[EarlybirdQuery, Long, eb.ThriftSearchResult] = { + + def lookup( + tweetIds: Seq[Long], + viewerId: Long + ): Future[Seq[Option[eb.ThriftSearchResult]]] = { + val request = EarlybirdRequestUtil.getTweetsEBFeaturesRequest( + userId = Some(viewerId), + tweetIds = Some(tweetIds), + clientId = Some(clientId.name) + ) + + client + .search(request).map { response => + val resultMap = response.searchResults + .map(_.results.map { result => result.id -> result }.toMap).getOrElse(Map.empty) + tweetIds.map(resultMap.get) + } + } + toRepositoryBatchWithView(lookup) + } + + protected def toRepository[K, V]( + hydrate: K => Future[V] + ): KeyValueRepository[Seq[K], K, V] = { + def asRepository(keys: Seq[K]): Future[KeyValueResult[K, V]] = { + Future.collect(keys.map(hydrate(_).liftToTry)).map { results => + keys + .zip(results) + .foldLeft(new KeyValueResultBuilder[K, V]()) { + case (bldr, (k, result)) => + result match { + case Return(v) => bldr.addFound(k, v) + case _ => bldr.addNotFound(k) + } + }.result + } + } + + asRepository + } + + protected def toRepositoryBatch[K, V]( + hydrate: Seq[K] => Future[Seq[Option[V]]], + chunkSize: Int = DefaultRPCChunkSize + ): KeyValueRepository[Seq[K], K, V] = { + def repository(keys: Seq[K]): Future[KeyValueResult[K, V]] = + batchRepositoryProcess(keys, hydrate(keys)) + + KeyValueRepository.chunked(repository, ChunkingStrategy.equalSize(chunkSize)) + } + + protected def toRepositoryBatchWithView[K, T, V]( + hydrate: (Seq[K], T) => Future[Seq[Option[V]]], + chunkSize: Int = DefaultRPCChunkSize + ): KeyValueRepository[(Seq[K], T), K, V] = { + def repository(input: (Seq[K], T)): Future[KeyValueResult[K, V]] = { + val (keys, view) = input + batchRepositoryProcess(keys, hydrate(keys, view)) + } + + KeyValueRepository.chunked(repository, CustomChunkingStrategy.equalSizeWithView(chunkSize)) + } + + private def batchRepositoryProcess[K, V]( + keys: Seq[K], + f: Future[Seq[Option[V]]] + ): Future[KeyValueResult[K, V]] = { + f.liftToTry + .map { + case Return(values) => + keys + .zip(values) + .foldLeft(new KeyValueResultBuilder[K, V]()) { + case (bldr, (k, value)) => + value match { + case Some(v) => bldr.addFound(k, v) + case _ => bldr.addNotFound(k) + } + }.result + case _ => + keys + .foldLeft(new KeyValueResultBuilder[K, V]()) { + case (bldr, k) => bldr.addNotFound(k) + }.result + } + } + + // Use only for cases not already covered by Servo's [[ChunkingStrategy]] + object CustomChunkingStrategy { + def equalSizeWithView[K, T](maxSize: Int): ((Seq[K], T)) => Seq[(Seq[K], T)] = { + case (keys, view) => + ChunkingStrategy + .equalSize[K](maxSize)(keys) + .map { chunk: Seq[K] => (chunk, view) } + } + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/TimelinesPersistenceStoreClientModule.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/TimelinesPersistenceStoreClientModule.scala new file mode 100644 index 0000000000..4292b36bec --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/TimelinesPersistenceStoreClientModule.scala @@ -0,0 +1,43 @@ +package com.twitter.home_mixer.module + +import com.google.inject.Provides +import com.twitter.conversions.DurationOps._ +import com.twitter.finagle.mtls.authentication.ServiceIdentifier +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.inject.TwitterModule +import com.twitter.timelinemixer.clients.persistence.TimelinePersistenceManhattanClientBuilder +import com.twitter.timelinemixer.clients.persistence.TimelinePersistenceManhattanClientConfig +import com.twitter.timelinemixer.clients.persistence.TimelineResponseBatchesClient +import com.twitter.timelinemixer.clients.persistence.TimelineResponseV3 +import javax.inject.Singleton + +object TimelinesPersistenceStoreClientModule extends TwitterModule { + private val StagingDataset = "timeline_response_batches_v5_nonprod" + private val ProdDataset = "timeline_response_batches_v5" + + @Provides + @Singleton + def providesTimelinesPersistenceStoreClient( + injectedServiceIdentifier: ServiceIdentifier, + statsReceiver: StatsReceiver + ): TimelineResponseBatchesClient[TimelineResponseV3] = { + val (timelineResponseBatchesDataset, manhattanReadOnly) = + injectedServiceIdentifier.environment.toLowerCase match { + case "prod" => (ProdDataset, false) + case _ => (StagingDataset, true) + } + + val timelineResponseBatchesConfig = new TimelinePersistenceManhattanClientConfig { + val dataset = timelineResponseBatchesDataset + val isReadOnly = manhattanReadOnly + val serviceIdentifier = injectedServiceIdentifier + override val defaultMaxTimeout = 300.milliseconds + override val maxRetryCount = 1 + } + + TimelinePersistenceManhattanClientBuilder.buildTimelineResponseV3BatchesClient( + timelineResponseBatchesConfig, + statsReceiver + ) + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/TweetyPieClientModule.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/TweetyPieClientModule.scala new file mode 100644 index 0000000000..84b634cc3e --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/TweetyPieClientModule.scala @@ -0,0 +1,51 @@ +package com.twitter.home_mixer.module + +import com.google.inject.Provides +import com.twitter.conversions.DurationOps._ +import com.twitter.finagle.mtls.authentication.ServiceIdentifier +import com.twitter.finagle.thrift.ClientId +import com.twitter.finagle.thriftmux.MethodBuilder +import com.twitter.finatra.mtls.thriftmux.modules.MtlsClient +import com.twitter.inject.Injector +import com.twitter.inject.thrift.modules.ThriftMethodBuilderClientModule +import com.twitter.stitch.tweetypie.TweetyPie +import com.twitter.tweetypie.thriftscala.TweetService +import com.twitter.util.Duration +import javax.inject.Singleton + +/** + * Idempotent TweetyPie Thrift and Stitch client. + */ +object TweetyPieClientModule + extends ThriftMethodBuilderClientModule[ + TweetService.ServicePerEndpoint, + TweetService.MethodPerEndpoint + ] + with MtlsClient { + override val label: String = "tweetypie" + override val dest: String = "/s/tweetypie/tweetypie" + + @Singleton + @Provides + def providesTweetypieStitchClient(tweetService: TweetService.MethodPerEndpoint): TweetyPie = + new TweetyPie(tweetService) + + /** + * TweetyPie client id must be in the form of {service.env} or it will not be treated as an + * unauthorized client + */ + override protected def clientId(injector: Injector): ClientId = { + val serviceIdentifier = injector.instance[ServiceIdentifier] + ClientId(s"${serviceIdentifier.service}.${serviceIdentifier.environment}") + } + + override protected def configureMethodBuilder( + injector: Injector, + methodBuilder: MethodBuilder + ): MethodBuilder = + methodBuilder + .withTimeoutPerRequest(500.milliseconds) + .withTimeoutTotal(500.milliseconds) + + override protected def sessionAcquisitionTimeout: Duration = 250.milliseconds +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/TweetypieStaticEntitiesCacheClientModule.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/TweetypieStaticEntitiesCacheClientModule.scala new file mode 100644 index 0000000000..2d0d5e08b6 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/TweetypieStaticEntitiesCacheClientModule.scala @@ -0,0 +1,69 @@ +package com.twitter.home_mixer.module + +import com.google.inject.name.Named +import com.google.inject.Provides +import com.twitter.conversions.DurationOps.RichDuration +import com.twitter.finagle.mtls.authentication.ServiceIdentifier +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.home_mixer.param.HomeMixerInjectionNames.TweetypieStaticEntitiesCache +import com.twitter.inject.TwitterModule +import com.twitter.product_mixer.shared_library.memcached_client.MemcachedClientBuilder +import com.twitter.servo.cache.FinagleMemcache +import com.twitter.servo.cache.KeyTransformer +import com.twitter.servo.cache.KeyValueTransformingTtlCache +import com.twitter.servo.cache.ObservableTtlCache +import com.twitter.servo.cache.Serializer +import com.twitter.servo.cache.ThriftSerializer +import com.twitter.servo.cache.TtlCache +import com.twitter.tweetypie.{thriftscala => tp} +import javax.inject.Singleton +import org.apache.thrift.protocol.TCompactProtocol + +object TweetypieStaticEntitiesCacheClientModule extends TwitterModule { + + private val ScopeName = "TweetypieStaticEntitiesMemcache" + private val ProdDest = "/srv#/prod/local/cache/timelinescorer_tweet_core_data:twemcaches" + + private val tweetsSerializer: Serializer[tp.Tweet] = { + new ThriftSerializer[tp.Tweet](tp.Tweet, new TCompactProtocol.Factory()) + } + private val keyTransformer: KeyTransformer[Long] = { tweetId => tweetId.toString } + + @Provides + @Singleton + @Named(TweetypieStaticEntitiesCache) + def providesTweetypieStaticEntitiesCache( + statsReceiver: StatsReceiver, + serviceIdentifier: ServiceIdentifier + ): TtlCache[Long, tp.Tweet] = { + val memCacheClient = MemcachedClientBuilder.buildMemcachedClient( + destName = ProdDest, + numTries = 1, + requestTimeout = 50.milliseconds, + globalTimeout = 100.milliseconds, + connectTimeout = 100.milliseconds, + acquisitionTimeout = 100.milliseconds, + serviceIdentifier = serviceIdentifier, + statsReceiver = statsReceiver + ) + mkCache(new FinagleMemcache(memCacheClient), statsReceiver) + } + + private def mkCache( + finagleMemcache: FinagleMemcache, + statsReceiver: StatsReceiver + ): TtlCache[Long, tp.Tweet] = { + val baseCache: KeyValueTransformingTtlCache[Long, String, tp.Tweet, Array[Byte]] = + new KeyValueTransformingTtlCache( + underlyingCache = finagleMemcache, + transformer = tweetsSerializer, + underlyingKey = keyTransformer + ) + ObservableTtlCache( + underlyingCache = baseCache, + statsReceiver = statsReceiver.scope(ScopeName), + windowSize = 1000, + name = ScopeName + ) + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/UserMetadataStoreModule.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/UserMetadataStoreModule.scala new file mode 100644 index 0000000000..7a450ac81b --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/module/UserMetadataStoreModule.scala @@ -0,0 +1,26 @@ +package com.twitter.home_mixer.module + +import com.google.inject.Provides +import com.google.inject.name.Named +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.home_mixer.param.HomeMixerInjectionNames.UserMetadataManhattanEndpoint +import com.twitter.home_mixer.param.HomeMixerInjectionNames.UserLanguagesStore +import com.twitter.home_mixer.store.UserLanguagesStore +import com.twitter.inject.TwitterModule +import com.twitter.search.common.constants.{thriftscala => scc} +import com.twitter.storage.client.manhattan.kv.ManhattanKVEndpoint +import com.twitter.storehaus.ReadableStore +import javax.inject.Singleton + +object UserMetadataStoreModule extends TwitterModule { + + @Provides + @Singleton + @Named(UserLanguagesStore) + def providesUserLanguagesFeaturesStore( + @Named(UserMetadataManhattanEndpoint) UserMetadataManhattanKVEndpoint: ManhattanKVEndpoint, + statsReceiver: StatsReceiver + ): ReadableStore[Long, Seq[scc.ThriftLanguage]] = { + new UserLanguagesStore(UserMetadataManhattanKVEndpoint, statsReceiver) + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/param/BUILD.bazel b/home-mixer/server/src/main/scala/com/twitter/home_mixer/param/BUILD.bazel new file mode 100644 index 0000000000..25e9a2e312 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/param/BUILD.bazel @@ -0,0 +1,9 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + strict_deps = True, + tags = ["bazel-compatible"], + dependencies = [ + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/product", + ], +) diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/param/GlobalParamConfigModule.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/param/GlobalParamConfigModule.scala new file mode 100644 index 0000000000..304e7bdc8b --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/param/GlobalParamConfigModule.scala @@ -0,0 +1,10 @@ +package com.twitter.home_mixer.param + +import com.twitter.inject.TwitterModule +import com.twitter.product_mixer.core.functional_component.configapi.registry.GlobalParamConfig + +object GlobalParamConfigModule extends TwitterModule { + override def configure(): Unit = { + bind[GlobalParamConfig].to[HomeGlobalParamConfig] + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/param/HomeGlobalParamConfig.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/param/HomeGlobalParamConfig.scala new file mode 100644 index 0000000000..a998aa9ca3 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/param/HomeGlobalParamConfig.scala @@ -0,0 +1,40 @@ +package com.twitter.home_mixer.param + +import com.twitter.home_mixer.param.HomeGlobalParams._ +import com.twitter.product_mixer.core.functional_component.configapi.registry.GlobalParamConfig +import javax.inject.Inject +import javax.inject.Singleton + +/** + * Register Params that do not relate to a specific product. See GlobalParamConfig -> ParamConfig + * for hooks to register Params based on type. + */ +@Singleton +class HomeGlobalParamConfig @Inject() () extends GlobalParamConfig { + + override val booleanFSOverrides = Seq( + AdsDisableInjectionBasedOnUserRoleParam, + EnableSendScoresToClient, + EnableNahFeedbackInfoParam, + EnableNewTweetsPillAvatarsParam, + EnableServedCandidateKafkaPublishingParam, + EnableSocialContextParam, + EnableGizmoduckAuthorSafetyFeatureHydratorParam, + EnableAdvertiserBrandSafetySettingsFeatureHydratorParam, + EnableFeedbackFatigueParam + ) + + override val boundedIntFSOverrides = Seq( + MaxNumberReplaceInstructionsParam, + TimelinesPersistenceStoreMaxEntriesPerClient, + ) + + override val boundedDoubleFSOverrides = Seq( + BlueVerifiedAuthorInNetworkMultiplierParam, + BlueVerifiedAuthorOutOfNetworkMultiplierParam + ) + + override val longSetFSOverrides = Seq( + AuthorListForStatsParam + ) +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/param/HomeGlobalParams.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/param/HomeGlobalParams.scala new file mode 100644 index 0000000000..0bd5dd1f06 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/param/HomeGlobalParams.scala @@ -0,0 +1,110 @@ +package com.twitter.home_mixer.param + +import com.twitter.timelines.configapi.FSBoundedParam +import com.twitter.timelines.configapi.FSParam + +/** + * Instantiate Params that do not relate to a specific product. + * + * @see [[com.twitter.product_mixer.core.product.ProductParamConfig.supportedClientFSName]] + */ +object HomeGlobalParams { + + /** + * This param is used to disable ads injection for timelines served by home-mixer. + * It is currently used to maintain user-role based no-ads lists for automation accounts, + * and should NOT be used for other purposes. + */ + object AdsDisableInjectionBasedOnUserRoleParam + extends FSParam("home_mixer_ads_disable_injection_based_on_user_role", false) + + object EnableSendScoresToClient + extends FSParam[Boolean]( + name = "home_mixer_enable_send_scores_to_client", + default = false + ) + + object EnableNahFeedbackInfoParam + extends FSParam[Boolean]( + name = "home_mixer_enable_nah_feedback_info", + default = false + ) + + object MaxNumberReplaceInstructionsParam + extends FSBoundedParam[Int]( + name = "home_mixer_max_number_replace_instructions", + default = 100, + min = 0, + max = 200 + ) + + object TimelinesPersistenceStoreMaxEntriesPerClient + extends FSBoundedParam[Int]( + name = "home_mixer_timelines_persistence_store_max_entries_per_client", + default = 1800, + min = 500, + max = 5000 + ) + + object EnableNewTweetsPillAvatarsParam + extends FSParam[Boolean]( + name = "home_mixer_enable_new_tweets_pill_avatars", + default = true + ) + + object EnableServedCandidateKafkaPublishingParam + extends FSParam[Boolean]( + name = "home_mixer_enable_served_candidate_kafka_publishing", + default = true + ) + + /** + * This author ID list is used purely for realtime metrics collection around how often we + * are serving Tweets from these authors and which candidate sources they are coming from. + */ + object AuthorListForStatsParam + extends FSParam[Set[Long]]( + name = "home_mixer_author_list_for_stats", + default = Set.empty + ) + + object EnableSocialContextParam + extends FSParam[Boolean]( + name = "home_mixer_enable_social_context", + default = false + ) + + object EnableGizmoduckAuthorSafetyFeatureHydratorParam + extends FSParam[Boolean]( + name = "home_mixer_enable_gizmoduck_author_safety_feature_hydrator", + default = true + ) + + object EnableFeedbackFatigueParam + extends FSParam[Boolean]( + name = "home_mixer_enable_feedback_fatigue", + default = true + ) + + object BlueVerifiedAuthorInNetworkMultiplierParam + extends FSBoundedParam[Double]( + name = "home_mixer_blue_verified_author_in_network_multiplier", + default = 4.0, + min = 0.0, + max = 100.0 + ) + + object BlueVerifiedAuthorOutOfNetworkMultiplierParam + extends FSBoundedParam[Double]( + name = "home_mixer_blue_verified_author_out_of_network_multiplier", + default = 2.0, + min = 0.0, + max = 100.0 + ) + + object EnableAdvertiserBrandSafetySettingsFeatureHydratorParam + extends FSParam[Boolean]( + name = "home_mixer_enable_advertiser_brand_safety_settings_feature_hydrator", + default = true + ) +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/param/HomeMixerFlagName.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/param/HomeMixerFlagName.scala new file mode 100644 index 0000000000..afe23c35a9 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/param/HomeMixerFlagName.scala @@ -0,0 +1,12 @@ +package com.twitter.home_mixer.param + +object HomeMixerFlagName { + final val ScribeClientEventsFlag = "scribe.client_events" + final val ScribeServedEntriesFlag = "scribe.served_entries" + final val ScribeServedCommonFeaturesAndCandidateFeaturesFlag = + "scribe.served_common_features_and_candidate_features" + final val DataRecordMetadataStoreConfigsYmlFlag = "data.record.metadata.store.configs.yml" + final val DarkTrafficFilterDeciderKey = "thrift.dark.traffic.filter.decider_key" + final val TargetFetchLatency = "target.fetch.latency" + final val TargetScoringLatency = "target.scoring.latency" +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/param/HomeMixerInjectionNames.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/param/HomeMixerInjectionNames.scala new file mode 100644 index 0000000000..ab2f45b4fd --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/param/HomeMixerInjectionNames.scala @@ -0,0 +1,50 @@ +package com.twitter.home_mixer.param + +object HomeMixerInjectionNames { + final val AuthorFeatureRepository = "AuthorFeatureRepository" + final val CandidateFeaturesScribeEventPublisher = "CandidateFeaturesScribeEventPublisher" + final val CommonFeaturesScribeEventPublisher = "CommonFeaturesScribeEventPublisher" + final val DDGStatsAuthors = "DDGStatsAuthors" + final val EarlybirdRepository = "EarlybirdRepository" + final val EngagementsReceivedByAuthorCache = "EngagementsReceivedByAuthorCache" + final val GraphTwoHopRepository = "GraphTwoHopRepository" + final val HomeAuthorFeaturesCacheClient = "HomeAuthorFeaturesCacheClient" + final val InterestsThriftServiceClient = "InterestsThriftServiceClient" + final val BatchedStratoClientWithModerateTimeout = "BatchedStratoClientWithModerateTimeout" + final val ManhattanApolloClient = "ManhattanApolloClient" + final val ManhattanAthenaClient = "ManhattanAthenaClient" + final val ManhattanOmegaClient = "ManhattanOmegaClient" + final val ManhattanStarbuckClient = "ManhattanStarbuckClient" + final val MetricCenterUserCountingFeatureRepository = "MetricCenterUserCountingFeatureRepository" + final val MinimumFeaturesScribeEventPublisher = "MinimumFeaturesScribeEventPublisher" + final val RealGraphInNetworkScores = "RealGraphInNetworkScores" + final val RealGraphManhattanEndpoint = "RealGraphFeaturesManhattanEndpoint" + final val RealGraphFeatureRepository = "RealGraphFeatureRepository" + final val RealTimeInteractionGraphUserVertexCache = "RealTimeInteractionGraphUserVertexCache" + final val RealTimeInteractionGraphUserVertexClient = "RealTimeInteractionGraphUserVertexClient" + final val StaleTweetsCache = "StaleTweetsCache" + final val TimelineAggregateMetadataRepository = "TimelineAggregateMetadataRepository" + final val TimelineAggregatePartARepository = "TimelineAggregatePartARepository" + final val TimelineAggregatePartBRepository = "TimelineAggregatePartBRepository" + final val TimelinesRealTimeAggregateClient = "TimelinesRealTimeAggregateClient" + final val TopicCountryEngagementCache = "TopicCountryEngagementCache" + final val TopicEngagementCache = "TopicEngagementCache" + final val TweetCountryEngagementCache = "TweetCountryEngagementCache" + final val TweetEngagementCache = "TweetEngagementCache" + final val TweetypieContentRepository = "TweetypieContentRepository" + final val TweetypieStaticEntitiesCache = "TweetypieStaticEntitiesCache" + final val TwhinAuthorFollow20200101FeatureCacheClient = + "TwhinAuthorFollow20200101FeatureCacheClient" + final val TwhinAuthorFollow20200101FeatureRepository = + "TwhinAuthorFollow20200101FeatureRepository" + final val TwhinUserEngagementFeatureRepository = "TwhinUserEngagementFeatureRepository" + final val TwhinUserFollowFeatureRepository = "TwhinUserFollowFeatureRepository" + final val TwitterListEngagementCache = "TwitterListEngagementCache" + final val UserAuthorEngagementCache = "UserAuthorEngagementCache" + final val UserEngagementCache = "UserEngagementCache" + final val UserFollowedTopicIdsRepository = "UserFollowedTopicIdsRepository" + final val UserLanguagesStore = "UserLanguagesStore" + final val UserMetadataManhattanEndpoint = "UserMetadataManhattanEndpoint" + final val UserTopicEngagementForNewUserCache = "UserTopicEngagementForNewUserCache" + final val UtegSocialProofRepository = "UtegSocialProofRepository" +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/param/decider/BUILD.bazel b/home-mixer/server/src/main/scala/com/twitter/home_mixer/param/decider/BUILD.bazel new file mode 100644 index 0000000000..5974c186f1 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/param/decider/BUILD.bazel @@ -0,0 +1,10 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + platform = "java8", + strict_deps = True, + tags = ["bazel-compatible"], + dependencies = [ + "servo/decider", + ], +) diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/param/decider/DeciderKey.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/param/decider/DeciderKey.scala new file mode 100644 index 0000000000..b81ea760a6 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/param/decider/DeciderKey.scala @@ -0,0 +1,40 @@ +package com.twitter.home_mixer.param.decider + +import com.twitter.servo.decider.DeciderKeyEnum + +/** + * These values must correspond to the deciders configured in the + * home-mixer/server/src/main/resources/config/decider.yml file + * + * @see [[com.twitter.product_mixer.core.product.ProductParamConfig.enabledDeciderKey]] + */ +object DeciderKey extends DeciderKeyEnum { + // Products + val EnableForYouProduct = Value("enable_for_you_product") + + val EnableFollowingProduct = Value("enable_following_product") + + val EnableScoredTweetsProduct = Value("enable_scored_tweets_product") + + val EnableListTweetsProduct = Value("enable_list_tweets_product") + + val EnableListRecommendedUsersProduct = Value("enable_list_recommended_users_product") + + // Candidate Pipelines + val EnableForYouScoredTweetsCandidatePipeline = Value( + "enable_for_you_scored_tweets_candidate_pipeline") + + val EnableScoredTweetsCrMixerCandidatePipeline = Value( + "enable_scored_tweets_cr_mixer_candidate_pipeline") + + val EnableScoredTweetsFrsCandidatePipeline = Value("enable_scored_tweets_frs_candidate_pipeline") + + val EnableScoredTweetsInNetworkCandidatePipeline = Value( + "enable_scored_tweets_in_network_candidate_pipeline") + + val EnableScoredTweetsUtegCandidatePipeline = Value( + "enable_scored_tweets_uteg_candidate_pipeline") + + val EnableSimClustersSimilarityFeatureHydration = Value( + "enable_simclusters_similarity_feature_hydration") +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/BUILD.bazel b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/BUILD.bazel new file mode 100644 index 0000000000..b59b7d9a3e --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/BUILD.bazel @@ -0,0 +1,26 @@ +scala_library( + compiler_option_sets = ["fatal_warnings"], + strict_deps = True, + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/javax/inject:javax.inject", + "finatra/inject/inject-core/src/main/scala", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/model", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/model/request", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/product/following", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/product/following/model", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/product/for_you", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/product/for_you/model", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_recommended_users", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_recommended_users/model", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_tweets", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_tweets/model", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/model", + "home-mixer/thrift/src/main/thrift:thrift-scala", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/query/ads", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/product", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/product/guice", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/product/registry", + ], +) diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/HomeMixerProductModule.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/HomeMixerProductModule.scala new file mode 100644 index 0000000000..cff4da4fb4 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/HomeMixerProductModule.scala @@ -0,0 +1,11 @@ +package com.twitter.home_mixer.product + +import com.twitter.inject.TwitterModule +import com.twitter.product_mixer.core.product.registry.ProductPipelineRegistryConfig + +object HomeMixerProductModule extends TwitterModule { + + override def configure(): Unit = { + bind[ProductPipelineRegistryConfig].to[HomeProductPipelineRegistryConfig] + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/HomeProductPipelineRegistryConfig.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/HomeProductPipelineRegistryConfig.scala new file mode 100644 index 0000000000..2131433511 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/HomeProductPipelineRegistryConfig.scala @@ -0,0 +1,54 @@ +package com.twitter.home_mixer.product + +import com.twitter.home_mixer.model.request.FollowingProduct +import com.twitter.home_mixer.model.request.ForYouProduct +import com.twitter.home_mixer.model.request.ListRecommendedUsersProduct +import com.twitter.home_mixer.model.request.ListTweetsProduct +import com.twitter.home_mixer.model.request.ScoredTweetsProduct +import com.twitter.home_mixer.product.following.FollowingProductPipelineConfig +import com.twitter.home_mixer.product.for_you.ForYouProductPipelineConfig +import com.twitter.home_mixer.product.list_recommended_users.ListRecommendedUsersProductPipelineConfig +import com.twitter.home_mixer.product.scored_tweets.ScoredTweetsProductPipelineConfig +import com.twitter.home_mixer.product.list_tweets.ListTweetsProductPipelineConfig +import com.twitter.inject.Injector +import com.twitter.product_mixer.core.product.guice.ProductScope +import com.twitter.product_mixer.core.product.registry.ProductPipelineRegistryConfig + +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class HomeProductPipelineRegistryConfig @Inject() ( + injector: Injector, + productScope: ProductScope) + extends ProductPipelineRegistryConfig { + + private val followingProductPipelineConfig = productScope.let(FollowingProduct) { + injector.instance[FollowingProductPipelineConfig] + } + + private val forYouProductPipelineConfig = productScope.let(ForYouProduct) { + injector.instance[ForYouProductPipelineConfig] + } + + private val scoredTweetsProductPipelineConfig = productScope.let(ScoredTweetsProduct) { + injector.instance[ScoredTweetsProductPipelineConfig] + } + + private val listTweetsProductPipelineConfig = productScope.let(ListTweetsProduct) { + injector.instance[ListTweetsProductPipelineConfig] + } + + private val listRecommendedUsersProductPipelineConfig = + productScope.let(ListRecommendedUsersProduct) { + injector.instance[ListRecommendedUsersProductPipelineConfig] + } + + override val productPipelineConfigs = Seq( + followingProductPipelineConfig, + forYouProductPipelineConfig, + scoredTweetsProductPipelineConfig, + listTweetsProductPipelineConfig, + listRecommendedUsersProductPipelineConfig + ) +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/following/BUILD.bazel b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/following/BUILD.bazel new file mode 100644 index 0000000000..3d3273daff --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/following/BUILD.bazel @@ -0,0 +1,95 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + strict_deps = True, + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/javax/inject:javax.inject", + "ads-injection/lib/src/main/scala/com/twitter/goldfinch/api", + "finagle/finagle-memcached/src/main/scala", + "finatra/inject/inject-core/src/main/scala", + "finatra/inject/inject-core/src/main/scala/com/twitter/inject", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/candidate_pipeline", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/candidate_source", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/builder", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/urt/builder", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/filter", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/gate", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/selector", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/side_effect", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/marshaller/timelines", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/model", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/model/request", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/param", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/product/following/model", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/product/following/param", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/service", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/util", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/candidate_source/account_recommendations_mixer", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/candidate_source/ads", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/candidate_source/hermit", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/candidate_source/people_discovery", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/candidate_source/tweetconvosvc", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/feature_hydrator/candidate/ads", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/feature_hydrator/candidate/param_gated", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/feature_hydrator/query/async", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/filter", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/gate", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/candidate", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/candidate/ads", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/presentation/urt", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/pipeline/candidate/ads", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/pipeline/candidate/flexible_injection_pipeline", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/pipeline/candidate/who_to_follow_module", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/premarshaller/urt", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/selector", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/selector/ads", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/side_effect", + "product-mixer/core/src/main/java/com/twitter/product_mixer/core/product/guice/scope", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/configapi", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/decorator", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/decorator/urt/builder", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/common/presentation", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/request", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/item", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/candidate", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/mixer", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/product", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/product", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/product/guice", + "src/java/com/twitter/search/common/schema/base", + "src/java/com/twitter/search/common/schema/earlybird", + "src/java/com/twitter/search/common/util/lang", + "src/java/com/twitter/search/queryparser/query:core-query-nodes", + "src/java/com/twitter/search/queryparser/query/search:search-query-nodes", + "src/scala/com/twitter/suggests/controller_data", + "src/thrift/com/twitter/search:earlybird-scala", + "src/thrift/com/twitter/search/common:constants-java", + "src/thrift/com/twitter/suggests/controller_data:controller_data-scala", + "src/thrift/com/twitter/timelinemixer:thrift-scala", + "src/thrift/com/twitter/timelines/render:thrift-scala", + "src/thrift/com/twitter/timelinescorer:thrift-scala", + "src/thrift/com/twitter/timelinescorer/common/scoredtweetcandidate:thrift-scala", + "src/thrift/com/twitter/timelinescorer/server/internal:thrift-scala", + "src/thrift/com/twitter/tweetypie:service-scala", + "stitch/stitch-gizmoduck", + "stitch/stitch-tweetypie", + "stringcenter/client", + "stringcenter/client/src/main/java", + "timelinemixer/common/src/main/scala/com/twitter/timelinemixer/clients/manhattan", + "timelinemixer/server/src/main/scala/com/twitter/timelinemixer/injection/model/candidate", + "timelinemixer/server/src/main/scala/com/twitter/timelinemixer/injection/store/persistence", + "timelines/src/main/scala/com/twitter/timelines/clients/relevance_search", + "timelines/src/main/scala/com/twitter/timelines/injection/scribe", + ], + exports = [ + "home-mixer/server/src/main/scala/com/twitter/home_mixer/product/following/param", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt", + "src/thrift/com/twitter/timelines/render:thrift-scala", + ], +) diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/following/FollowingAdsCandidatePipelineBuilder.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/following/FollowingAdsCandidatePipelineBuilder.scala new file mode 100644 index 0000000000..77477652fc --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/following/FollowingAdsCandidatePipelineBuilder.scala @@ -0,0 +1,102 @@ +package com.twitter.home_mixer.product.following + +import com.twitter.adserver.{thriftscala => ads} +import com.twitter.home_mixer.functional_component.decorator.HomeAdsClientEventDetailsBuilder +import com.twitter.home_mixer.functional_component.gate.ExcludeSoftUserGate +import com.twitter.home_mixer.param.HomeGlobalParams +import com.twitter.home_mixer.param.HomeGlobalParams.EnableAdvertiserBrandSafetySettingsFeatureHydratorParam +import com.twitter.home_mixer.product.following.model.FollowingQuery +import com.twitter.home_mixer.product.following.param.FollowingParam.EnableAdsCandidatePipelineParam +import com.twitter.home_mixer.product.following.param.FollowingParam.EnableFastAds +import com.twitter.home_mixer.service.HomeMixerAlertConfig +import com.twitter.product_mixer.component_library.candidate_source.ads.AdsProdThriftCandidateSource +import com.twitter.product_mixer.component_library.decorator.urt.UrtItemCandidateDecorator +import com.twitter.product_mixer.component_library.decorator.urt.builder.contextual_ref.ContextualTweetRefBuilder +import com.twitter.product_mixer.component_library.decorator.urt.builder.item.ad.AdsCandidateUrtItemBuilder +import com.twitter.product_mixer.component_library.decorator.urt.builder.metadata.ClientEventInfoBuilder +import com.twitter.product_mixer.component_library.feature_hydrator.candidate.ads.AdvertiserBrandSafetySettingsFeatureHydrator +import com.twitter.product_mixer.component_library.feature_hydrator.candidate.param_gated.ParamGatedCandidateFeatureHydrator +import com.twitter.product_mixer.component_library.gate.NonEmptyCandidatesGate +import com.twitter.product_mixer.component_library.model.candidate.ads.AdsCandidate +import com.twitter.product_mixer.component_library.pipeline.candidate.ads.AdsDependentCandidatePipelineConfig +import com.twitter.product_mixer.component_library.pipeline.candidate.ads.AdsDependentCandidatePipelineConfigBuilder +import com.twitter.product_mixer.component_library.pipeline.candidate.ads.CountCandidatesFromPipelines +import com.twitter.product_mixer.component_library.pipeline.candidate.ads.PipelineScopedOrganicItemIds +import com.twitter.product_mixer.component_library.pipeline.candidate.ads.ValidAdImpressionIdFilter +import com.twitter.product_mixer.core.functional_component.common.CandidateScope +import com.twitter.product_mixer.core.gate.ParamNotGate +import com.twitter.product_mixer.core.model.common.identifier.CandidatePipelineIdentifier +import com.twitter.product_mixer.core.model.marshalling.response.rtf.safety_level.TimelineHomePromotedHydrationSafetyLevel +import com.twitter.product_mixer.core.model.marshalling.response.urt.contextual_ref.TweetHydrationContext +import com.twitter.timelines.injection.scribe.InjectionScribeUtil +import com.twitter.timelineservice.suggests.{thriftscala => st} +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class FollowingAdsCandidatePipelineBuilder @Inject() ( + adsCandidatePipelineConfigBuilder: AdsDependentCandidatePipelineConfigBuilder, + adsCandidateSource: AdsProdThriftCandidateSource, + advertiserBrandSafetySettingsFeatureHydrator: AdvertiserBrandSafetySettingsFeatureHydrator[ + FollowingQuery, + AdsCandidate + ]) { + + private val identifier: CandidatePipelineIdentifier = CandidatePipelineIdentifier("FollowingAds") + + private val suggestType = st.SuggestType.Promoted + + private val clientEventInfoBuilder = ClientEventInfoBuilder( + component = InjectionScribeUtil.scribeComponent(suggestType).get, + detailsBuilder = Some(HomeAdsClientEventDetailsBuilder(Some(suggestType.name))) + ) + + private val contextualTweetRefBuilder = ContextualTweetRefBuilder( + TweetHydrationContext( + safetyLevelOverride = Some(TimelineHomePromotedHydrationSafetyLevel), + outerTweetContext = None + )) + + private val decorator = UrtItemCandidateDecorator( + AdsCandidateUrtItemBuilder( + tweetClientEventInfoBuilder = Some(clientEventInfoBuilder), + contextualTweetRefBuilder = Some(contextualTweetRefBuilder) + )) + + private val alerts = Seq( + HomeMixerAlertConfig.BusinessHours.defaultSuccessRateAlert(), + HomeMixerAlertConfig.BusinessHours.defaultEmptyResponseRateAlert() + ) + + def build( + organicCandidatePipelines: CandidateScope + ): AdsDependentCandidatePipelineConfig[FollowingQuery] = + adsCandidatePipelineConfigBuilder.build[FollowingQuery]( + adsCandidateSource = adsCandidateSource, + identifier = identifier, + adsDisplayLocationBuilder = query => + if (query.params.getBoolean(EnableFastAds)) ads.DisplayLocation.TimelineHomeReverseChron + else ads.DisplayLocation.TimelineHome, + getOrganicItemIds = PipelineScopedOrganicItemIds(organicCandidatePipelines), + countNumOrganicItems = CountCandidatesFromPipelines(organicCandidatePipelines), + supportedClientParam = Some(EnableAdsCandidatePipelineParam), + gates = Seq( + ParamNotGate( + name = "AdsDisableInjectionBasedOnUserRole", + param = HomeGlobalParams.AdsDisableInjectionBasedOnUserRoleParam + ), + ExcludeSoftUserGate, + NonEmptyCandidatesGate(organicCandidatePipelines) + ), + filters = Seq(ValidAdImpressionIdFilter), + postFilterFeatureHydration = Seq( + ParamGatedCandidateFeatureHydrator( + EnableAdvertiserBrandSafetySettingsFeatureHydratorParam, + advertiserBrandSafetySettingsFeatureHydrator + ) + ), + decorator = Some(decorator), + alerts = alerts, + urtRequest = Some(true), + ) +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/following/FollowingEarlybirdCandidatePipelineConfig.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/following/FollowingEarlybirdCandidatePipelineConfig.scala new file mode 100644 index 0000000000..ea26ca19e7 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/following/FollowingEarlybirdCandidatePipelineConfig.scala @@ -0,0 +1,54 @@ +package com.twitter.home_mixer.product.following + +import com.twitter.home_mixer.candidate_pipeline.FollowingEarlybirdResponseFeatureTransformer +import com.twitter.home_mixer.functional_component.candidate_source.EarlybirdCandidateSource +import com.twitter.home_mixer.functional_component.feature_hydrator.SGSFollowedUsersFeature +import com.twitter.home_mixer.functional_component.gate.NonEmptySeqFeatureGate +import com.twitter.home_mixer.product.following.model.FollowingQuery +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.functional_component.candidate_source.BaseCandidateSource +import com.twitter.product_mixer.core.functional_component.gate.Gate +import com.twitter.product_mixer.core.functional_component.transformer.CandidateFeatureTransformer +import com.twitter.product_mixer.core.functional_component.transformer.CandidatePipelineQueryTransformer +import com.twitter.product_mixer.core.functional_component.transformer.CandidatePipelineResultsTransformer +import com.twitter.product_mixer.core.model.common.identifier.CandidatePipelineIdentifier +import com.twitter.product_mixer.core.pipeline.candidate.CandidatePipelineConfig +import com.twitter.search.earlybird.{thriftscala => t} +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class FollowingEarlybirdCandidatePipelineConfig @Inject() ( + earlybirdCandidateSource: EarlybirdCandidateSource, + followingEarlybirdQueryTransformer: FollowingEarlybirdQueryTransformer) + extends CandidatePipelineConfig[ + FollowingQuery, + t.EarlybirdRequest, + t.ThriftSearchResult, + TweetCandidate + ] { + + override val identifier: CandidatePipelineIdentifier = + CandidatePipelineIdentifier("FollowingEarlybird") + + override val candidateSource: BaseCandidateSource[t.EarlybirdRequest, t.ThriftSearchResult] = + earlybirdCandidateSource + + override val gates: Seq[Gate[FollowingQuery]] = Seq( + NonEmptySeqFeatureGate(SGSFollowedUsersFeature) + ) + + override val queryTransformer: CandidatePipelineQueryTransformer[ + FollowingQuery, + t.EarlybirdRequest + ] = followingEarlybirdQueryTransformer + + override val featuresFromCandidateSourceTransformers: Seq[ + CandidateFeatureTransformer[t.ThriftSearchResult] + ] = Seq(FollowingEarlybirdResponseFeatureTransformer) + + override val resultTransformer: CandidatePipelineResultsTransformer[ + t.ThriftSearchResult, + TweetCandidate + ] = { sourceResult => TweetCandidate(id = sourceResult.id) } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/following/FollowingEarlybirdQueryTransformer.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/following/FollowingEarlybirdQueryTransformer.scala new file mode 100644 index 0000000000..9f7dd306eb --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/following/FollowingEarlybirdQueryTransformer.scala @@ -0,0 +1,84 @@ +package com.twitter.home_mixer.product.following + +import com.twitter.finagle.thrift.ClientId +import com.twitter.finagle.tracing.Trace +import com.twitter.home_mixer.functional_component.feature_hydrator.SGSFollowedUsersFeature +import com.twitter.home_mixer.model.HomeFeatures.RealGraphInNetworkScoresFeature +import com.twitter.home_mixer.product.following.model.FollowingQuery +import com.twitter.home_mixer.product.following.param.FollowingParam.ServerMaxResultsParam +import com.twitter.product_mixer.core.functional_component.transformer.CandidatePipelineQueryTransformer +import com.twitter.product_mixer.core.model.marshalling.response.urt.operation.BottomCursor +import com.twitter.product_mixer.core.model.marshalling.response.urt.operation.GapCursor +import com.twitter.product_mixer.core.model.marshalling.response.urt.operation.TopCursor +import com.twitter.product_mixer.core.pipeline.pipeline_failure.MalformedCursor +import com.twitter.product_mixer.core.pipeline.pipeline_failure.PipelineFailure +import com.twitter.search.common.schema.earlybird.EarlybirdFieldConstants.EarlybirdFieldConstant +import com.twitter.search.earlybird.{thriftscala => t} +import com.twitter.search.queryparser.query.Conjunction +import com.twitter.search.queryparser.query.search.SearchOperator +import javax.inject.Inject +import javax.inject.Singleton +import scala.jdk.CollectionConverters.asJavaIterableConverter + +@Singleton +case class FollowingEarlybirdQueryTransformer @Inject() (clientId: ClientId) + extends CandidatePipelineQueryTransformer[FollowingQuery, t.EarlybirdRequest] { + + override def transform(query: FollowingQuery): t.EarlybirdRequest = { + val followedUserIds = + query.features.map(_.get(SGSFollowedUsersFeature)).getOrElse(Seq.empty).toSet + val realGraphInNetworkFollowedUserIds = + query.features.map(_.get(RealGraphInNetworkScoresFeature)).getOrElse(Map.empty).keySet + val userId = query.getRequiredUserId + val combinedUserIds = userId +: (followedUserIds ++ realGraphInNetworkFollowedUserIds).toSeq + + val baseFollowedUsersSearchOperator = new SearchOperator.Builder() + .setType(SearchOperator.Type.FEATURE_VALUE_IN_ACCEPT_LIST_OR_UNSET) + .addOperand(EarlybirdFieldConstant.DIRECTED_AT_USER_ID_CSF.getFieldName) + + val followedUsersQuery = + baseFollowedUsersSearchOperator.addOperands(combinedUserIds.map(_.toString).asJava).build() + + val searchQuery = query.pipelineCursor + .map { cursor => + val sinceIdQuery = + (id: Long) => new SearchOperator(SearchOperator.Type.SINCE_ID, id.toString) + val maxIdQuery = // max ID is inclusive, so subtract 1 + (id: Long) => new SearchOperator(SearchOperator.Type.MAX_ID, (id - 1).toString) + + (cursor.cursorType, cursor.id, cursor.gapBoundaryId) match { + case (Some(TopCursor), Some(sinceId), _) => + new Conjunction(sinceIdQuery(sinceId), followedUsersQuery) + case (Some(BottomCursor), Some(maxId), _) => + new Conjunction(maxIdQuery(maxId), followedUsersQuery) + case (Some(GapCursor), Some(maxId), Some(sinceId)) => + new Conjunction(sinceIdQuery(sinceId), maxIdQuery(maxId), followedUsersQuery) + case (Some(GapCursor), _, _) => + throw PipelineFailure(MalformedCursor, "Invalid cursor " + cursor.toString) + case _ => followedUsersQuery + } + }.getOrElse(followedUsersQuery) + + val metadataOptions = t.ThriftSearchResultMetadataOptions( + getInReplyToStatusId = true, + getReferencedTweetAuthorId = true, + getFromUserId = true + ) + + t.EarlybirdRequest( + searchQuery = t.ThriftSearchQuery( + serializedQuery = Some(searchQuery.serialize), + fromUserIDFilter64 = Some(combinedUserIds), + numResults = query.requestedMaxResults.getOrElse(query.params(ServerMaxResultsParam)), + rankingMode = t.ThriftSearchRankingMode.Recency, + resultMetadataOptions = Some(metadataOptions), + searcherId = query.getOptionalUserId, + ), + getOlderResults = Some(true), // needed for archive access to older tweets + clientRequestID = Some(s"${Trace.id.traceId}"), + followedUserIds = Some(combinedUserIds), + numResultsToReturnAtRoot = Some(query.params(ServerMaxResultsParam)), + clientId = Some(clientId.name), + ) + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/following/FollowingEarlybirdResponseFeatureTransformer.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/following/FollowingEarlybirdResponseFeatureTransformer.scala new file mode 100644 index 0000000000..0169e6dd74 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/following/FollowingEarlybirdResponseFeatureTransformer.scala @@ -0,0 +1,38 @@ +package com.twitter.home_mixer.candidate_pipeline + +import com.twitter.home_mixer.model.HomeFeatures.AuthorIdFeature +import com.twitter.home_mixer.model.HomeFeatures.InReplyToTweetIdFeature +import com.twitter.home_mixer.model.HomeFeatures.IsRetweetFeature +import com.twitter.home_mixer.model.HomeFeatures.SourceTweetIdFeature +import com.twitter.home_mixer.model.HomeFeatures.SourceUserIdFeature +import com.twitter.product_mixer.core.feature.Feature +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.feature.featuremap.FeatureMapBuilder +import com.twitter.product_mixer.core.functional_component.transformer.CandidateFeatureTransformer +import com.twitter.product_mixer.core.model.common.identifier.TransformerIdentifier +import com.twitter.search.earlybird.{thriftscala => t} + +object FollowingEarlybirdResponseFeatureTransformer + extends CandidateFeatureTransformer[t.ThriftSearchResult] { + + override val identifier: TransformerIdentifier = + TransformerIdentifier("FollowingEarlybirdResponse") + + override val features: Set[Feature[_, _]] = Set( + AuthorIdFeature, + InReplyToTweetIdFeature, + IsRetweetFeature, + SourceTweetIdFeature, + SourceUserIdFeature, + ) + + override def transform(candidate: t.ThriftSearchResult): FeatureMap = FeatureMapBuilder() + .add(AuthorIdFeature, candidate.tweetypieTweet.flatMap(_.coreData.map(_.userId))) + .add( + InReplyToTweetIdFeature, + candidate.tweetypieTweet.flatMap(_.coreData.flatMap(_.reply.flatMap(_.inReplyToStatusId)))) + .add(IsRetweetFeature, candidate.metadata.exists(_.isRetweet.contains(true))) + .add(SourceTweetIdFeature, candidate.sourceTweetypieTweet.map(_.id)) + .add(SourceUserIdFeature, candidate.sourceTweetypieTweet.flatMap(_.coreData.map(_.userId))) + .build() +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/following/FollowingMixerPipelineConfig.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/following/FollowingMixerPipelineConfig.scala new file mode 100644 index 0000000000..23b4164a5a --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/following/FollowingMixerPipelineConfig.scala @@ -0,0 +1,278 @@ +package com.twitter.home_mixer.product.following + +import com.twitter.clientapp.{thriftscala => ca} +import com.twitter.goldfinch.api.AdsInjectionSurfaceAreas +import com.twitter.home_mixer.candidate_pipeline.ConversationServiceCandidatePipelineConfigBuilder +import com.twitter.home_mixer.candidate_pipeline.EditedTweetsCandidatePipelineConfig +import com.twitter.home_mixer.candidate_pipeline.NewTweetsPillCandidatePipelineConfig +import com.twitter.home_mixer.functional_component.decorator.HomeConversationServiceCandidateDecorator +import com.twitter.home_mixer.functional_component.decorator.HomeFeedbackActionInfoBuilder +import com.twitter.home_mixer.functional_component.decorator.urt.builder.AddEntriesWithReplaceAndShowAlertAndCoverInstructionBuilder +import com.twitter.home_mixer.functional_component.feature_hydrator._ +import com.twitter.home_mixer.functional_component.selector.UpdateHomeClientEventDetails +import com.twitter.home_mixer.functional_component.selector.UpdateNewTweetsPillDecoration +import com.twitter.home_mixer.functional_component.side_effect._ +import com.twitter.home_mixer.model.GapIncludeInstruction +import com.twitter.home_mixer.param.HomeGlobalParams.MaxNumberReplaceInstructionsParam +import com.twitter.home_mixer.product.following.model.FollowingQuery +import com.twitter.home_mixer.product.following.model.HomeMixerExternalStrings +import com.twitter.home_mixer.product.following.param.FollowingParam.EnableFlipInjectionModuleCandidatePipelineParam +import com.twitter.home_mixer.product.following.param.FollowingParam.FlipInlineInjectionModulePosition +import com.twitter.home_mixer.product.following.param.FollowingParam.ServerMaxResultsParam +import com.twitter.home_mixer.product.following.param.FollowingParam.WhoToFollowPositionParam +import com.twitter.home_mixer.util.CandidatesUtil +import com.twitter.logpipeline.client.common.EventPublisher +import com.twitter.product_mixer.component_library.feature_hydrator.query.async.AsyncQueryFeatureHydrator +import com.twitter.product_mixer.component_library.gate.NonEmptyCandidatesGate +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.component_library.pipeline.candidate.flexible_injection_pipeline.FlipPromptDependentCandidatePipelineConfigBuilder +import com.twitter.product_mixer.component_library.pipeline.candidate.who_to_follow_module.WhoToFollowArmCandidatePipelineConfig +import com.twitter.product_mixer.component_library.premarshaller.urt.UrtDomainMarshaller +import com.twitter.product_mixer.component_library.premarshaller.urt.builder.OrderedBottomCursorBuilder +import com.twitter.product_mixer.component_library.premarshaller.urt.builder.OrderedGapCursorBuilder +import com.twitter.product_mixer.component_library.premarshaller.urt.builder.OrderedTopCursorBuilder +import com.twitter.product_mixer.component_library.premarshaller.urt.builder.ReplaceAllEntries +import com.twitter.product_mixer.component_library.premarshaller.urt.builder.ReplaceEntryInstructionBuilder +import com.twitter.product_mixer.component_library.premarshaller.urt.builder.ShowAlertInstructionBuilder +import com.twitter.product_mixer.component_library.premarshaller.urt.builder.ShowCoverInstructionBuilder +import com.twitter.product_mixer.component_library.premarshaller.urt.builder.StaticTimelineScribeConfigBuilder +import com.twitter.product_mixer.component_library.premarshaller.urt.builder.UrtMetadataBuilder +import com.twitter.product_mixer.component_library.selector.DropMaxCandidates +import com.twitter.product_mixer.component_library.selector.DropMaxModuleItemCandidates +import com.twitter.product_mixer.component_library.selector.DropModuleTooFewModuleItemResults +import com.twitter.product_mixer.component_library.selector.InsertAppendResults +import com.twitter.product_mixer.component_library.selector.InsertFixedPositionResults +import com.twitter.product_mixer.component_library.selector.SelectConditionally +import com.twitter.product_mixer.component_library.selector.UpdateSortCandidates +import com.twitter.product_mixer.component_library.selector.ads.AdsInjector +import com.twitter.product_mixer.component_library.selector.ads.InsertAdResults +import com.twitter.product_mixer.core.functional_component.common.SpecificPipeline +import com.twitter.product_mixer.core.functional_component.common.SpecificPipelines +import com.twitter.product_mixer.core.functional_component.configapi.StaticParam +import com.twitter.product_mixer.core.functional_component.feature_hydrator.QueryFeatureHydrator +import com.twitter.product_mixer.core.functional_component.marshaller.TransportMarshaller +import com.twitter.product_mixer.core.functional_component.marshaller.response.urt.UrtTransportMarshaller +import com.twitter.product_mixer.core.functional_component.premarshaller.DomainMarshaller +import com.twitter.product_mixer.core.functional_component.selector.Selector +import com.twitter.product_mixer.core.functional_component.side_effect.PipelineResultSideEffect +import com.twitter.product_mixer.core.model.common.UniversalNoun +import com.twitter.product_mixer.core.model.common.identifier.CandidatePipelineIdentifier +import com.twitter.product_mixer.core.model.common.identifier.MixerPipelineIdentifier +import com.twitter.product_mixer.core.model.marshalling.response.urt.Timeline +import com.twitter.product_mixer.core.model.marshalling.response.urt.TimelineModule +import com.twitter.product_mixer.core.model.marshalling.response.urt.TimelineScribeConfig +import com.twitter.product_mixer.core.model.marshalling.response.urt.item.tweet.TweetItem +import com.twitter.product_mixer.core.pipeline.FailOpenPolicy +import com.twitter.product_mixer.core.pipeline.candidate.CandidatePipelineConfig +import com.twitter.product_mixer.core.pipeline.candidate.DependentCandidatePipelineConfig +import com.twitter.product_mixer.core.pipeline.mixer.MixerPipelineConfig +import com.twitter.product_mixer.core.product.guice.scope.ProductScoped +import com.twitter.stringcenter.client.StringCenter +import com.twitter.timelines.render.{thriftscala => urt} +import javax.inject.Inject +import javax.inject.Provider +import javax.inject.Singleton + +@Singleton +class FollowingMixerPipelineConfig @Inject() ( + followingEarlybirdCandidatePipelineConfig: FollowingEarlybirdCandidatePipelineConfig, + conversationServiceCandidatePipelineConfigBuilder: ConversationServiceCandidatePipelineConfigBuilder[ + FollowingQuery + ], + homeFeedbackActionInfoBuilder: HomeFeedbackActionInfoBuilder, + followingAdsCandidatePipelineBuilder: FollowingAdsCandidatePipelineBuilder, + followingWhoToFollowArmCandidatePipelineConfigBuilder: FollowingWhoToFollowArmCandidatePipelineConfigBuilder, + flipPromptDependentCandidatePipelineConfigBuilder: FlipPromptDependentCandidatePipelineConfigBuilder, + editedTweetsCandidatePipelineConfig: EditedTweetsCandidatePipelineConfig, + newTweetsPillCandidatePipelineConfig: NewTweetsPillCandidatePipelineConfig[FollowingQuery], + dismissInfoQueryFeatureHydrator: DismissInfoQueryFeatureHydrator, + gizmoduckUserQueryFeatureHydrator: GizmoduckUserQueryFeatureHydrator, + persistenceStoreQueryFeatureHydrator: PersistenceStoreQueryFeatureHydrator, + realGraphInNetworkSourceQueryHydrator: RealGraphInNetworkScoresQueryFeatureHydrator, + requestQueryFeatureHydrator: RequestQueryFeatureHydrator[FollowingQuery], + sgsFollowedUsersQueryFeatureHydrator: SGSFollowedUsersQueryFeatureHydrator, + tweetImpressionsQueryFeatureHydrator: TweetImpressionsQueryFeatureHydrator[FollowingQuery], + lastNonPollingTimeQueryFeatureHydrator: LastNonPollingTimeQueryFeatureHydrator, + adsInjector: AdsInjector, + updateLastNonPollingTimeSideEffect: UpdateLastNonPollingTimeSideEffect[FollowingQuery, Timeline], + publishClientSentImpressionsEventBusSideEffect: PublishClientSentImpressionsEventBusSideEffect, + publishClientSentImpressionsManhattanSideEffect: PublishClientSentImpressionsManhattanSideEffect, + updateTimelinesPersistenceStoreSideEffect: UpdateTimelinesPersistenceStoreSideEffect, + truncateTimelinesPersistenceStoreSideEffect: TruncateTimelinesPersistenceStoreSideEffect, + homeTimelineServedEntriesSideEffect: HomeScribeServedEntriesSideEffect, + clientEventsScribeEventPublisher: EventPublisher[ca.LogEvent], + externalStrings: HomeMixerExternalStrings, + @ProductScoped stringCenterProvider: Provider[StringCenter], + urtTransportMarshaller: UrtTransportMarshaller) + extends MixerPipelineConfig[FollowingQuery, Timeline, urt.TimelineResponse] { + + override val identifier: MixerPipelineIdentifier = MixerPipelineIdentifier("Following") + + private val dependentCandidatesStep = MixerPipelineConfig.dependentCandidatePipelinesStep + private val resultSelectorsStep = MixerPipelineConfig.resultSelectorsStep + + override val fetchQueryFeatures: Seq[QueryFeatureHydrator[FollowingQuery]] = Seq( + requestQueryFeatureHydrator, + sgsFollowedUsersQueryFeatureHydrator, + realGraphInNetworkSourceQueryHydrator, + AsyncQueryFeatureHydrator(dependentCandidatesStep, dismissInfoQueryFeatureHydrator), + AsyncQueryFeatureHydrator(dependentCandidatesStep, gizmoduckUserQueryFeatureHydrator), + AsyncQueryFeatureHydrator(dependentCandidatesStep, persistenceStoreQueryFeatureHydrator), + AsyncQueryFeatureHydrator(dependentCandidatesStep, lastNonPollingTimeQueryFeatureHydrator), + AsyncQueryFeatureHydrator(resultSelectorsStep, tweetImpressionsQueryFeatureHydrator), + ) + + private val earlybirdCandidatePipelineScope = + SpecificPipeline(followingEarlybirdCandidatePipelineConfig.identifier) + + private val conversationServiceCandidatePipelineConfig = + conversationServiceCandidatePipelineConfigBuilder.build( + Seq(NonEmptyCandidatesGate(earlybirdCandidatePipelineScope)), + HomeConversationServiceCandidateDecorator(homeFeedbackActionInfoBuilder) + ) + + private val followingAdsCandidatePipelineConfig = + followingAdsCandidatePipelineBuilder.build(earlybirdCandidatePipelineScope) + + private val followingWhoToFollowArmCandidatePipelineConfig = + followingWhoToFollowArmCandidatePipelineConfigBuilder.build(earlybirdCandidatePipelineScope) + + private val flipPromptCandidatePipelineConfig = + flipPromptDependentCandidatePipelineConfigBuilder.build[FollowingQuery]( + supportedClientParam = Some(EnableFlipInjectionModuleCandidatePipelineParam) + ) + + override val candidatePipelines: Seq[CandidatePipelineConfig[FollowingQuery, _, _, _]] = + Seq(followingEarlybirdCandidatePipelineConfig) + + override val dependentCandidatePipelines: Seq[ + DependentCandidatePipelineConfig[FollowingQuery, _, _, _] + ] = Seq( + conversationServiceCandidatePipelineConfig, + followingAdsCandidatePipelineConfig, + followingWhoToFollowArmCandidatePipelineConfig, + flipPromptCandidatePipelineConfig, + editedTweetsCandidatePipelineConfig, + newTweetsPillCandidatePipelineConfig + ) + + override val failOpenPolicies: Map[CandidatePipelineIdentifier, FailOpenPolicy] = Map( + followingAdsCandidatePipelineConfig.identifier -> FailOpenPolicy.Always, + followingWhoToFollowArmCandidatePipelineConfig.identifier -> FailOpenPolicy.Always, + flipPromptCandidatePipelineConfig.identifier -> FailOpenPolicy.Always, + editedTweetsCandidatePipelineConfig.identifier -> FailOpenPolicy.Always, + newTweetsPillCandidatePipelineConfig.identifier -> FailOpenPolicy.Always, + ) + + override val resultSelectors: Seq[Selector[FollowingQuery]] = Seq( + UpdateSortCandidates( + ordering = CandidatesUtil.reverseChronTweetsOrdering, + candidatePipeline = conversationServiceCandidatePipelineConfig.identifier + ), + DropMaxCandidates( + candidatePipeline = editedTweetsCandidatePipelineConfig.identifier, + maxSelectionsParam = MaxNumberReplaceInstructionsParam + ), + DropMaxCandidates( + candidatePipeline = conversationServiceCandidatePipelineConfig.identifier, + maxSelectionsParam = ServerMaxResultsParam + ), + DropModuleTooFewModuleItemResults( + candidatePipeline = followingWhoToFollowArmCandidatePipelineConfig.identifier, + minModuleItemsParam = StaticParam(WhoToFollowArmCandidatePipelineConfig.MinCandidatesSize) + ), + DropMaxModuleItemCandidates( + candidatePipeline = followingWhoToFollowArmCandidatePipelineConfig.identifier, + maxModuleItemsParam = StaticParam(WhoToFollowArmCandidatePipelineConfig.MaxCandidatesSize) + ), + InsertAppendResults(candidatePipeline = conversationServiceCandidatePipelineConfig.identifier), + InsertFixedPositionResults( + candidatePipeline = followingWhoToFollowArmCandidatePipelineConfig.identifier, + positionParam = WhoToFollowPositionParam + ), + InsertFixedPositionResults( + candidatePipeline = flipPromptCandidatePipelineConfig.identifier, + positionParam = FlipInlineInjectionModulePosition + ), + InsertAdResults( + surfaceAreaName = AdsInjectionSurfaceAreas.HomeTimeline, + adsInjector = adsInjector.forSurfaceArea(AdsInjectionSurfaceAreas.HomeTimeline), + adsCandidatePipeline = followingAdsCandidatePipelineConfig.identifier + ), + // This selector must come after the tweets are inserted into the results + UpdateNewTweetsPillDecoration( + pipelineScope = SpecificPipelines( + conversationServiceCandidatePipelineConfig.identifier, + newTweetsPillCandidatePipelineConfig.identifier + ), + stringCenter = stringCenterProvider.get(), + seeNewTweetsString = externalStrings.seeNewTweetsString, + tweetedString = externalStrings.tweetedString + ), + InsertAppendResults(candidatePipeline = editedTweetsCandidatePipelineConfig.identifier), + SelectConditionally( + selector = + InsertAppendResults(candidatePipeline = newTweetsPillCandidatePipelineConfig.identifier), + includeSelector = (_, _, results) => CandidatesUtil.containsType[TweetCandidate](results) + ), + UpdateHomeClientEventDetails( + candidatePipelines = Set(conversationServiceCandidatePipelineConfig.identifier) + ), + ) + + private val homeScribeClientEventSideEffect = HomeScribeClientEventSideEffect( + logPipelinePublisher = clientEventsScribeEventPublisher, + injectedTweetsCandidatePipelineIdentifiers = + Seq(conversationServiceCandidatePipelineConfig.identifier), + adsCandidatePipelineIdentifier = followingAdsCandidatePipelineConfig.identifier, + whoToFollowCandidatePipelineIdentifier = + Some(followingWhoToFollowArmCandidatePipelineConfig.identifier), + ) + + override val resultSideEffects: Seq[PipelineResultSideEffect[FollowingQuery, Timeline]] = Seq( + updateLastNonPollingTimeSideEffect, + publishClientSentImpressionsEventBusSideEffect, + publishClientSentImpressionsManhattanSideEffect, + homeScribeClientEventSideEffect, + updateTimelinesPersistenceStoreSideEffect, + truncateTimelinesPersistenceStoreSideEffect, + homeTimelineServedEntriesSideEffect + ) + + override val domainMarshaller: DomainMarshaller[FollowingQuery, Timeline] = { + val instructionBuilders = Seq( + ReplaceEntryInstructionBuilder(ReplaceAllEntries), + AddEntriesWithReplaceAndShowAlertAndCoverInstructionBuilder(), + ShowAlertInstructionBuilder(), + ShowCoverInstructionBuilder(), + ) + + val idSelector: PartialFunction[UniversalNoun[_], Long] = { + // exclude ads while determining tweet cursor values + case item: TweetItem if item.promotedMetadata.isEmpty => item.id + case module: TimelineModule + if module.items.headOption.exists(_.item.isInstanceOf[TweetItem]) => + module.items.last.item match { case item: TweetItem => item.id } + } + val topCursorBuilder = OrderedTopCursorBuilder(idSelector) + val bottomCursorBuilder = + OrderedBottomCursorBuilder(idSelector, GapIncludeInstruction.inverse()) + val gapCursorBuilder = OrderedGapCursorBuilder(idSelector, GapIncludeInstruction) + + val metadataBuilder = UrtMetadataBuilder( + title = None, + scribeConfigBuilder = Some( + StaticTimelineScribeConfigBuilder( + TimelineScribeConfig(page = Some("following"), section = None, entityToken = None))) + ) + + UrtDomainMarshaller( + instructionBuilders = instructionBuilders, + metadataBuilder = Some(metadataBuilder), + cursorBuilders = Seq(topCursorBuilder, bottomCursorBuilder, gapCursorBuilder) + ) + } + + override val transportMarshaller: TransportMarshaller[Timeline, urt.TimelineResponse] = + urtTransportMarshaller +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/following/FollowingProductPipelineConfig.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/following/FollowingProductPipelineConfig.scala new file mode 100644 index 0000000000..28c7cd6a01 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/following/FollowingProductPipelineConfig.scala @@ -0,0 +1,131 @@ +package com.twitter.home_mixer.product.following + +import com.twitter.conversions.DurationOps._ +import com.twitter.home_mixer.marshaller.timelines.ChronologicalCursorUnmarshaller +import com.twitter.home_mixer.model.request.FollowingProduct +import com.twitter.home_mixer.model.request.FollowingProductContext +import com.twitter.home_mixer.model.request.HomeMixerRequest +import com.twitter.home_mixer.product.following.model.FollowingQuery +import com.twitter.home_mixer.product.following.param.FollowingParam.ServerMaxResultsParam +import com.twitter.home_mixer.product.following.param.FollowingParamConfig +import com.twitter.home_mixer.service.HomeMixerAccessPolicy.DefaultHomeMixerAccessPolicy +import com.twitter.home_mixer.service.HomeMixerAlertConfig.DefaultNotificationGroup +import com.twitter.product_mixer.component_library.model.cursor.UrtOrderedCursor +import com.twitter.product_mixer.component_library.premarshaller.cursor.UrtCursorSerializer +import com.twitter.product_mixer.core.functional_component.common.access_policy.AccessPolicy +import com.twitter.product_mixer.core.functional_component.common.alert.Alert +import com.twitter.product_mixer.core.functional_component.common.alert.EmptyResponseRateAlert +import com.twitter.product_mixer.core.functional_component.common.alert.LatencyAlert +import com.twitter.product_mixer.core.functional_component.common.alert.P99 +import com.twitter.product_mixer.core.functional_component.common.alert.SuccessRateAlert +import com.twitter.product_mixer.core.functional_component.common.alert.ThroughputAlert +import com.twitter.product_mixer.core.functional_component.common.alert.predicate.TriggerIfAbove +import com.twitter.product_mixer.core.functional_component.common.alert.predicate.TriggerIfBelow +import com.twitter.product_mixer.core.functional_component.common.alert.predicate.TriggerIfLatencyAbove +import com.twitter.product_mixer.core.model.common.identifier.ComponentIdentifier +import com.twitter.product_mixer.core.model.common.identifier.ProductPipelineIdentifier +import com.twitter.product_mixer.core.model.marshalling.request.Product +import com.twitter.product_mixer.core.model.marshalling.response.urt.operation.GapCursor +import com.twitter.product_mixer.core.model.marshalling.response.urt.operation.TopCursor +import com.twitter.product_mixer.core.pipeline.PipelineConfig +import com.twitter.product_mixer.core.pipeline.pipeline_failure.BadRequest +import com.twitter.product_mixer.core.pipeline.pipeline_failure.MalformedCursor +import com.twitter.product_mixer.core.pipeline.pipeline_failure.PipelineFailure +import com.twitter.product_mixer.core.pipeline.product.ProductPipelineConfig +import com.twitter.product_mixer.core.product.ProductParamConfig +import com.twitter.product_mixer.core.util.SortIndexBuilder +import com.twitter.timelines.configapi.Params +import com.twitter.timelines.render.{thriftscala => urt} +import com.twitter.timelines.util.RequestCursorSerializer +import com.twitter.util.Time +import com.twitter.util.Try +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class FollowingProductPipelineConfig @Inject() ( + followingMixerPipelineConfig: FollowingMixerPipelineConfig, + followingParamConfig: FollowingParamConfig) + extends ProductPipelineConfig[HomeMixerRequest, FollowingQuery, urt.TimelineResponse] { + + override val identifier: ProductPipelineIdentifier = ProductPipelineIdentifier("Following") + + override val product: Product = FollowingProduct + override val paramConfig: ProductParamConfig = followingParamConfig + + override def pipelineQueryTransformer( + request: HomeMixerRequest, + params: Params + ): FollowingQuery = { + val context = request.productContext match { + case Some(context: FollowingProductContext) => context + case _ => throw PipelineFailure(BadRequest, "FollowingProductContext not found") + } + + val debugOptions = request.debugParams.flatMap(_.debugOptions) + + /** + * Unlike other clients, newly created tweets on Android have the sort index set to the current + * time instead of the top sort index + 1, so these tweets get stuck at the top of the timeline + * if subsequent timeline responses use the sort index from the previous response instead of + * the current time. + */ + val pipelineCursor = request.serializedRequestCursor.flatMap { cursor => + Try(UrtCursorSerializer.deserializeOrderedCursor(cursor)) + .getOrElse(ChronologicalCursorUnmarshaller(RequestCursorSerializer.deserialize(cursor))) + .map { + case UrtOrderedCursor(_, id, Some(GapCursor), gapBoundaryId) + if id.isEmpty || gapBoundaryId.isEmpty => + throw PipelineFailure(MalformedCursor, "Gap Cursor bounds not defined") + case topCursor @ UrtOrderedCursor(_, _, Some(TopCursor), _) => + val queryTime = debugOptions.flatMap(_.requestTimeOverride).getOrElse(Time.now) + topCursor.copy(initialSortIndex = SortIndexBuilder.timeToId(queryTime)) + case cursor => cursor + } + } + + FollowingQuery( + params = params, + clientContext = request.clientContext, + features = None, + pipelineCursor = pipelineCursor, + requestedMaxResults = Some(params(ServerMaxResultsParam)), + debugOptions = debugOptions, + deviceContext = context.deviceContext, + seenTweetIds = context.seenTweetIds, + dspClientContext = context.dspClientContext + ) + } + + override val pipelines: Seq[PipelineConfig] = Seq(followingMixerPipelineConfig) + + override def pipelineSelector( + query: FollowingQuery + ): ComponentIdentifier = followingMixerPipelineConfig.identifier + + override val alerts: Seq[Alert] = Seq( + SuccessRateAlert( + notificationGroup = DefaultNotificationGroup, + warnPredicate = TriggerIfBelow(99.9, 20, 30), + criticalPredicate = TriggerIfBelow(99.9, 30, 30), + ), + LatencyAlert( + notificationGroup = DefaultNotificationGroup, + percentile = P99, + warnPredicate = TriggerIfLatencyAbove(1100.millis, 15, 30), + criticalPredicate = TriggerIfLatencyAbove(1200.millis, 15, 30) + ), + ThroughputAlert( + notificationGroup = DefaultNotificationGroup, + warnPredicate = TriggerIfAbove(18000), + criticalPredicate = TriggerIfAbove(20000) + ), + EmptyResponseRateAlert( + notificationGroup = DefaultNotificationGroup, + warnPredicate = TriggerIfAbove(65), + criticalPredicate = TriggerIfAbove(80) + ) + ) + + override val debugAccessPolicies: Set[AccessPolicy] = DefaultHomeMixerAccessPolicy +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/following/FollowingWhoToFollowArmCandidatePipelineConfigBuilder.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/following/FollowingWhoToFollowArmCandidatePipelineConfigBuilder.scala new file mode 100644 index 0000000000..484a72d5f4 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/following/FollowingWhoToFollowArmCandidatePipelineConfigBuilder.scala @@ -0,0 +1,65 @@ +package com.twitter.home_mixer.product.following + +import com.twitter.home_mixer.functional_component.decorator.urt.builder.HomeWhoToFollowFeedbackActionInfoBuilder +import com.twitter.home_mixer.functional_component.gate.DismissFatigueGate +import com.twitter.home_mixer.functional_component.gate.TimelinesPersistenceStoreLastInjectionGate +import com.twitter.home_mixer.model.HomeFeatures.DismissInfoFeature +import com.twitter.home_mixer.model.HomeFeatures.PersistenceEntriesFeature +import com.twitter.home_mixer.model.HomeFeatures.WhoToFollowExcludedUserIdsFeature +import com.twitter.home_mixer.product.following.model.FollowingQuery +import com.twitter.home_mixer.product.following.param.FollowingParam.EnableWhoToFollowCandidatePipelineParam +import com.twitter.home_mixer.product.following.param.FollowingParam.WhoToFollowDisplayLocationParam +import com.twitter.home_mixer.product.following.param.FollowingParam.WhoToFollowDisplayTypeIdParam +import com.twitter.home_mixer.product.following.param.FollowingParam.WhoToFollowMinInjectionIntervalParam +import com.twitter.home_mixer.service.HomeMixerAlertConfig +import com.twitter.product_mixer.component_library.decorator.urt.builder.timeline_module.ParamWhoToFollowModuleDisplayTypeBuilder +import com.twitter.product_mixer.component_library.gate.NonEmptyCandidatesGate +import com.twitter.product_mixer.component_library.pipeline.candidate.who_to_follow_module.WhoToFollowArmCandidatePipelineConfig +import com.twitter.product_mixer.component_library.pipeline.candidate.who_to_follow_module.WhoToFollowArmDependentCandidatePipelineConfig +import com.twitter.product_mixer.component_library.pipeline.candidate.who_to_follow_module.WhoToFollowArmDependentCandidatePipelineConfigBuilder +import com.twitter.product_mixer.core.functional_component.common.CandidateScope +import com.twitter.product_mixer.core.functional_component.configapi.StaticParam +import com.twitter.product_mixer.core.functional_component.gate.BaseGate +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.timelineservice.model.rich.EntityIdType +import com.twitter.timelineservice.suggests.thriftscala.SuggestType +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class FollowingWhoToFollowArmCandidatePipelineConfigBuilder @Inject() ( + whoToFollowArmDependentCandidatePipelineConfigBuilder: WhoToFollowArmDependentCandidatePipelineConfigBuilder, + homeWhoToFollowFeedbackActionInfoBuilder: HomeWhoToFollowFeedbackActionInfoBuilder) { + + def build( + requiredNonEmptyPipelines: CandidateScope + ): WhoToFollowArmDependentCandidatePipelineConfig[FollowingQuery] = { + val gates: Seq[BaseGate[PipelineQuery]] = Seq( + TimelinesPersistenceStoreLastInjectionGate( + WhoToFollowMinInjectionIntervalParam, + PersistenceEntriesFeature, + EntityIdType.WhoToFollow + ), + DismissFatigueGate(SuggestType.WhoToFollow, DismissInfoFeature), + NonEmptyCandidatesGate(requiredNonEmptyPipelines) + ) + + whoToFollowArmDependentCandidatePipelineConfigBuilder.build[FollowingQuery]( + identifier = WhoToFollowArmCandidatePipelineConfig.identifier, + supportedClientParam = Some(EnableWhoToFollowCandidatePipelineParam), + alerts = alerts, + gates = gates, + moduleDisplayTypeBuilder = + ParamWhoToFollowModuleDisplayTypeBuilder(WhoToFollowDisplayTypeIdParam), + feedbackActionInfoBuilder = Some(homeWhoToFollowFeedbackActionInfoBuilder), + displayLocationParam = StaticParam(WhoToFollowDisplayLocationParam.default), + excludedUserIdsFeature = Some(WhoToFollowExcludedUserIdsFeature), + profileUserIdFeature = None + ) + } + + private val alerts = Seq( + HomeMixerAlertConfig.BusinessHours.defaultSuccessRateAlert(70), + HomeMixerAlertConfig.BusinessHours.defaultEmptyResponseRateAlert() + ) +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/following/model/BUILD.bazel b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/following/model/BUILD.bazel new file mode 100644 index 0000000000..70d5e5af93 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/following/model/BUILD.bazel @@ -0,0 +1,22 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + strict_deps = True, + tags = ["bazel-compatible"], + dependencies = [ + "home-mixer/server/src/main/scala/com/twitter/home_mixer/model", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/model/request", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/product/following/param", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/cursor", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/query/ads", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/pipeline/candidate/flexible_injection_pipeline", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline", + "stringcenter/client", + "stringcenter/client/src/main/java", + ], + exports = [ + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/cursor", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/query/ads", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline", + ], +) diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/following/model/FollowingQuery.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/following/model/FollowingQuery.scala new file mode 100644 index 0000000000..39442a61bc --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/following/model/FollowingQuery.scala @@ -0,0 +1,50 @@ +package com.twitter.home_mixer.product.following.model + +import com.twitter.adserver.thriftscala.HomeTimelineType +import com.twitter.adserver.thriftscala.TimelineRequestParams +import com.twitter.home_mixer.model.HomeAdsQuery +import com.twitter.dspbidder.commons.{thriftscala => dsp} +import com.twitter.home_mixer.model.request.DeviceContext +import com.twitter.home_mixer.model.request.HasDeviceContext +import com.twitter.home_mixer.model.request.HasSeenTweetIds +import com.twitter.home_mixer.model.request.FollowingProduct +import com.twitter.onboarding.task.service.{thriftscala => ots} +import com.twitter.product_mixer.component_library.model.cursor.UrtOrderedCursor +import com.twitter.product_mixer.component_library.pipeline.candidate.flexible_injection_pipeline.transformer.HasFlipInjectionParams +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.model.marshalling.request._ +import com.twitter.product_mixer.core.pipeline.HasPipelineCursor +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.timelines.configapi.Params + +case class FollowingQuery( + override val params: Params, + override val clientContext: ClientContext, + override val pipelineCursor: Option[UrtOrderedCursor], + override val requestedMaxResults: Option[Int], + override val debugOptions: Option[DebugOptions], + override val features: Option[FeatureMap], + override val deviceContext: Option[DeviceContext], + override val seenTweetIds: Option[Seq[Long]], + override val dspClientContext: Option[dsp.DspClientContext]) + extends PipelineQuery + with HasPipelineCursor[UrtOrderedCursor] + with HasDeviceContext + with HasSeenTweetIds + with HasFlipInjectionParams + with HomeAdsQuery { + override val product: Product = FollowingProduct + + override def withFeatureMap(features: FeatureMap): FollowingQuery = + copy(features = Some(features)) + + override val timelineRequestParams: Option[TimelineRequestParams] = + Some(TimelineRequestParams(homeTimelineType = Some(HomeTimelineType.HomeLatest))) + + // Fields below are used for FLIP Injection in Onboarding Task Service (OTS) + override val displayLocation: ots.DisplayLocation = ots.DisplayLocation.HomeLatestTimeline + override val rankingDisablerWithLatestControlsAvailable: Option[Boolean] = None + override val isEmptyState: Option[Boolean] = None + override val isFirstRequestAfterSignup: Option[Boolean] = None + override val isEndOfTimeline: Option[Boolean] = None +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/following/model/HomeMixerExternalStrings.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/following/model/HomeMixerExternalStrings.scala new file mode 100644 index 0000000000..fb07bc12c1 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/following/model/HomeMixerExternalStrings.scala @@ -0,0 +1,66 @@ +package com.twitter.home_mixer.product.following.model + +import com.twitter.product_mixer.core.product.guice.scope.ProductScoped +import com.twitter.stringcenter.client.ExternalStringRegistry +import javax.inject.Inject +import javax.inject.Provider +import javax.inject.Singleton + +@Singleton +class HomeMixerExternalStrings @Inject() ( + @ProductScoped externalStringRegistryProvider: Provider[ExternalStringRegistry]) { + val seeNewTweetsString = + externalStringRegistryProvider.get().createProdString("SeeNewTweets") + val tweetedString = + externalStringRegistryProvider.get().createProdString("Tweeted") + val muteUserString = + externalStringRegistryProvider.get().createProdString("Feedback.muteUser") + val blockUserString = externalStringRegistryProvider.get().createProdString("Feedback.blockUser") + val unfollowUserString = + externalStringRegistryProvider.get().createProdString("Feedback.unfollowUser") + val unfollowUserConfirmationString = + externalStringRegistryProvider.get().createProdString("Feedback.unfollowUserConfirmation") + val reportTweetString = + externalStringRegistryProvider.get().createProdString("Feedback.reportTweet") + val dontLikeString = externalStringRegistryProvider.get().createProdString("Feedback.dontLike") + val dontLikeConfirmationString = + externalStringRegistryProvider.get().createProdString("Feedback.dontLikeConfirmation") + val showFewerTweetsString = + externalStringRegistryProvider.get().createProdString("Feedback.showFewerTweets") + val showFewerTweetsConfirmationString = + externalStringRegistryProvider.get().createProdString("Feedback.showFewerTweetsConfirmation") + val showFewerRetweetsString = + externalStringRegistryProvider.get().createProdString("Feedback.showFewerRetweets") + val showFewerRetweetsConfirmationString = + externalStringRegistryProvider.get().createProdString("Feedback.showFewerRetweetsConfirmation") + val notRelevantString = + externalStringRegistryProvider.get().createProdString("Feedback.notRelevant") + val notRelevantConfirmationString = + externalStringRegistryProvider.get().createProdString("Feedback.notRelevantConfirmation") + + val socialContextOneUserLikedString = + externalStringRegistryProvider.get().createProdString("SocialContext.oneUserLiked") + val socialContextTwoUsersLikedString = + externalStringRegistryProvider.get().createProdString("SocialContext.twoUsersLiked") + val socialContextMoreUsersLikedString = + externalStringRegistryProvider.get().createProdString("SocialContext.moreUsersLiked") + val socialContextLikedByTimelineTitle = + externalStringRegistryProvider.get().createProdString("SocialContext.likedByTimelineTitle") + + val socialContextOneUserFollowsString = + externalStringRegistryProvider.get().createProdString("SocialContext.oneUserFollows") + val socialContextTwoUsersFollowString = + externalStringRegistryProvider.get().createProdString("SocialContext.twoUsersFollow") + val socialContextMoreUsersFollowString = + externalStringRegistryProvider.get().createProdString("SocialContext.moreUsersFollow") + val socialContextFollowedByTimelineTitle = + externalStringRegistryProvider.get().createProdString("SocialContext.followedByTimelineTitle") + + val socialContextYouMightLikeString = + externalStringRegistryProvider.get().createProdString("SocialContext.youMightLike") + + val socialContextExtendedReply = + externalStringRegistryProvider.get().createProdString("SocialContext.extendedReply") + val socialContextReceivedReply = + externalStringRegistryProvider.get().createProdString("SocialContext.receivedReply") +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/following/param/BUILD.bazel b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/following/param/BUILD.bazel new file mode 100644 index 0000000000..a56e3a1fdc --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/following/param/BUILD.bazel @@ -0,0 +1,14 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + strict_deps = True, + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/javax/inject:javax.inject", + "configapi/configapi-core/src/main/scala/com/twitter/timelines/configapi", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/param/decider", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/product", + "util/util-core/src/main/scala/com/twitter/conversions", + ], +) diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/following/param/FollowingParam.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/following/param/FollowingParam.scala new file mode 100644 index 0000000000..e43990507e --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/following/param/FollowingParam.scala @@ -0,0 +1,85 @@ +package com.twitter.home_mixer.product.following.param + +import com.twitter.conversions.DurationOps._ +import com.twitter.product_mixer.component_library.decorator.urt.builder.timeline_module.WhoToFollowModuleDisplayType +import com.twitter.timelines.configapi.DurationConversion +import com.twitter.timelines.configapi.FSBoundedParam +import com.twitter.timelines.configapi.FSEnumParam +import com.twitter.timelines.configapi.FSParam +import com.twitter.timelines.configapi.HasDurationConversion +import com.twitter.util.Duration + +object FollowingParam { + val SupportedClientFSName = "following_supported_client" + + object ServerMaxResultsParam + extends FSBoundedParam[Int]( + name = "following_server_max_results", + default = 100, + min = 1, + max = 500 + ) + + object EnableWhoToFollowCandidatePipelineParam + extends FSParam[Boolean]( + name = "following_enable_who_to_follow", + default = true + ) + + object EnableAdsCandidatePipelineParam + extends FSParam[Boolean]( + name = "following_enable_ads", + default = true + ) + + object EnableFlipInjectionModuleCandidatePipelineParam + extends FSParam[Boolean]( + name = "following_enable_flip_inline_injection_module", + default = true + ) + + object FlipInlineInjectionModulePosition + extends FSBoundedParam[Int]( + name = "following_flip_inline_injection_module_position", + default = 0, + min = 0, + max = 1000 + ) + + object WhoToFollowPositionParam + extends FSBoundedParam[Int]( + name = "following_who_to_follow_position", + default = 5, + min = 0, + max = 99 + ) + + object WhoToFollowMinInjectionIntervalParam + extends FSBoundedParam[Duration]( + "following_who_to_follow_min_injection_interval_in_minutes", + default = 1800.minutes, + min = 0.minutes, + max = 6000.minutes) + with HasDurationConversion { + override val durationConversion: DurationConversion = DurationConversion.FromMinutes + } + + object WhoToFollowDisplayTypeIdParam + extends FSEnumParam[WhoToFollowModuleDisplayType.type]( + name = "following_enable_who_to_follow_display_type_id", + default = WhoToFollowModuleDisplayType.Vertical, + enum = WhoToFollowModuleDisplayType + ) + + object WhoToFollowDisplayLocationParam + extends FSParam[String]( + name = "following_who_to_follow_display_location", + default = "timeline_reverse_chron" + ) + + object EnableFastAds + extends FSParam[Boolean]( + name = "following_enable_fast_ads", + default = true + ) +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/following/param/FollowingParamConfig.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/following/param/FollowingParamConfig.scala new file mode 100644 index 0000000000..df3b548017 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/following/param/FollowingParamConfig.scala @@ -0,0 +1,34 @@ +package com.twitter.home_mixer.product.following.param + +import com.twitter.home_mixer.param.decider.DeciderKey +import com.twitter.home_mixer.product.following.param.FollowingParam._ +import com.twitter.product_mixer.core.product.ProductParamConfig +import com.twitter.servo.decider.DeciderKeyName +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class FollowingParamConfig @Inject() () extends ProductParamConfig { + override val enabledDeciderKey: DeciderKeyName = DeciderKey.EnableFollowingProduct + override val supportedClientFSName: String = SupportedClientFSName + + override val booleanFSOverrides = + Seq( + EnableFlipInjectionModuleCandidatePipelineParam, + EnableWhoToFollowCandidatePipelineParam, + EnableAdsCandidatePipelineParam, + EnableFastAds, + ) + + override val boundedIntFSOverrides = Seq( + FlipInlineInjectionModulePosition, + WhoToFollowPositionParam, + ServerMaxResultsParam + ) + + override val stringFSOverrides = Seq(WhoToFollowDisplayLocationParam) + + override val boundedDurationFSOverrides = Seq(WhoToFollowMinInjectionIntervalParam) + + override val enumFSOverrides = Seq(WhoToFollowDisplayTypeIdParam) +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/for_you/BUILD.bazel b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/for_you/BUILD.bazel new file mode 100644 index 0000000000..8371cbdf1b --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/for_you/BUILD.bazel @@ -0,0 +1,87 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + strict_deps = True, + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/javax/inject:javax.inject", + "ads-injection/lib/src/main/scala/com/twitter/goldfinch/api", + "finagle/finagle-memcached/src/main/scala", + "finatra/inject/inject-core/src/main/scala", + "finatra/inject/inject-core/src/main/scala/com/twitter/inject", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/candidate_pipeline", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/candidate_source", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/builder", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/urt/builder", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/filter", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/gate", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/scorer", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/selector", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/side_effect", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/marshaller/timelines", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/model", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/model/request", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/param", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/product/following/model", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/product/for_you/candidate_source", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/product/for_you/model", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/product/for_you/param", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/model", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/service", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/util", + "home-mixer/thrift/src/main/thrift:thrift-scala", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/candidate_source/ads", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/candidate_source/people_discovery", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/candidate_source/timeline_scorer", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/candidate_source/timeline_service", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/feature_hydrator/candidate/ads", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/feature_hydrator/candidate/param_gated", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/feature_hydrator/candidate/tweet_is_nsfw", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/feature_hydrator/query/async", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/feature_hydrator/query/impressed_tweets", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/filter", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/gate", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/candidate/ads", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/pipeline/candidate/ads", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/pipeline/candidate/flexible_injection_pipeline", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/pipeline/candidate/who_to_follow_module", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/premarshaller/urt", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/scorer/tweet_tlx", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/selector", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/selector/ads", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/side_effect", + "product-mixer/core/src/main/java/com/twitter/product_mixer/core/product/guice/scope", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/candidate_source/product_pipeline", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/configapi", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/decorator", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/decorator/urt/builder", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/request", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/common/presentation", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/request", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/item", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/candidate", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/mixer", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/product", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/recommendation", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/scoring", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/product", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/product/guice", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/quality_factor", + "src/java/com/twitter/search/common/util/lang", + "src/scala/com/twitter/suggests/controller_data", + "src/thrift/com/twitter/search/common:constants-java", + "src/thrift/com/twitter/timelines/render:thrift-scala", + "src/thrift/com/twitter/timelines/suggests/common:poly_data_record-java", + "src/thrift/com/twitter/timelineservice:thrift-scala", + "timelinemixer/server/src/main/scala/com/twitter/timelinemixer/injection/model/candidate", + "timelines/src/main/scala/com/twitter/timelines/injection/scribe", + "timelines/src/main/scala/com/twitter/timelines/model/candidate", + ], + exports = [ + "src/thrift/com/twitter/timelines/render:thrift-scala", + ], +) diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/for_you/ForYouAdsCandidatePipelineBuilder.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/for_you/ForYouAdsCandidatePipelineBuilder.scala new file mode 100644 index 0000000000..82963f5bb4 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/for_you/ForYouAdsCandidatePipelineBuilder.scala @@ -0,0 +1,94 @@ +package com.twitter.home_mixer.product.for_you + +import com.twitter.product_mixer.component_library.feature_hydrator.candidate.param_gated.ParamGatedCandidateFeatureHydrator +import com.twitter.adserver.{thriftscala => ads} +import com.twitter.home_mixer.functional_component.decorator.HomeAdsClientEventDetailsBuilder +import com.twitter.home_mixer.functional_component.gate.ExcludeSoftUserGate +import com.twitter.home_mixer.param.HomeGlobalParams +import com.twitter.home_mixer.param.HomeGlobalParams.EnableAdvertiserBrandSafetySettingsFeatureHydratorParam +import com.twitter.home_mixer.product.for_you.model.ForYouQuery +import com.twitter.home_mixer.product.for_you.param.ForYouParam.AdsNumOrganicItemsParam +import com.twitter.home_mixer.service.HomeMixerAlertConfig +import com.twitter.product_mixer.component_library.candidate_source.ads.AdsProdThriftCandidateSource +import com.twitter.product_mixer.component_library.decorator.urt.UrtItemCandidateDecorator +import com.twitter.product_mixer.component_library.decorator.urt.builder.contextual_ref.ContextualTweetRefBuilder +import com.twitter.product_mixer.component_library.decorator.urt.builder.item.ad.AdsCandidateUrtItemBuilder +import com.twitter.product_mixer.component_library.decorator.urt.builder.metadata.ClientEventInfoBuilder +import com.twitter.product_mixer.component_library.feature_hydrator.candidate.ads.AdvertiserBrandSafetySettingsFeatureHydrator +import com.twitter.product_mixer.component_library.model.candidate.ads.AdsCandidate +import com.twitter.product_mixer.component_library.pipeline.candidate.ads.AdsCandidatePipelineConfig +import com.twitter.product_mixer.component_library.pipeline.candidate.ads.AdsCandidatePipelineConfigBuilder +import com.twitter.product_mixer.component_library.pipeline.candidate.ads.StaticAdsDisplayLocationBuilder +import com.twitter.product_mixer.component_library.pipeline.candidate.ads.ValidAdImpressionIdFilter +import com.twitter.product_mixer.core.functional_component.common.CandidateScope +import com.twitter.product_mixer.core.gate.ParamNotGate +import com.twitter.product_mixer.core.model.common.identifier.CandidatePipelineIdentifier +import com.twitter.product_mixer.core.model.marshalling.response.rtf.safety_level.TimelineHomePromotedHydrationSafetyLevel +import com.twitter.product_mixer.core.model.marshalling.response.urt.contextual_ref.TweetHydrationContext +import com.twitter.timelines.injection.scribe.InjectionScribeUtil +import com.twitter.timelineservice.suggests.{thriftscala => st} +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class ForYouAdsCandidatePipelineBuilder @Inject() ( + adsCandidatePipelineConfigBuilder: AdsCandidatePipelineConfigBuilder, + adsCandidateSource: AdsProdThriftCandidateSource, + advertiserBrandSafetySettingsFeatureHydrator: AdvertiserBrandSafetySettingsFeatureHydrator[ + ForYouQuery, + AdsCandidate + ]) { + + private val identifier: CandidatePipelineIdentifier = CandidatePipelineIdentifier("ForYouAds") + + private val suggestType = st.SuggestType.Promoted + + private val clientEventInfoBuilder = ClientEventInfoBuilder( + component = InjectionScribeUtil.scribeComponent(suggestType).get, + detailsBuilder = Some(HomeAdsClientEventDetailsBuilder(Some(suggestType.name))) + ) + + private val contextualTweetRefBuilder = ContextualTweetRefBuilder( + TweetHydrationContext( + safetyLevelOverride = Some(TimelineHomePromotedHydrationSafetyLevel), + outerTweetContext = None + )) + + private val decorator = UrtItemCandidateDecorator( + AdsCandidateUrtItemBuilder( + tweetClientEventInfoBuilder = Some(clientEventInfoBuilder), + contextualTweetRefBuilder = Some(contextualTweetRefBuilder) + )) + + private val alerts = Seq( + HomeMixerAlertConfig.BusinessHours.defaultSuccessRateAlert(), + HomeMixerAlertConfig.BusinessHours.defaultEmptyResponseRateAlert() + ) + + def build( + organicCandidatePipelines: Option[CandidateScope] = None + ): AdsCandidatePipelineConfig[ForYouQuery] = + adsCandidatePipelineConfigBuilder.build[ForYouQuery]( + adsCandidateSource = adsCandidateSource, + identifier = identifier, + adsDisplayLocationBuilder = StaticAdsDisplayLocationBuilder(ads.DisplayLocation.TimelineHome), + estimateNumOrganicItems = _.params(AdsNumOrganicItemsParam).toShort, + gates = Seq( + ParamNotGate( + name = "AdsDisableInjectionBasedOnUserRole", + param = HomeGlobalParams.AdsDisableInjectionBasedOnUserRoleParam + ), + ExcludeSoftUserGate + ), + filters = Seq(ValidAdImpressionIdFilter), + postFilterFeatureHydration = Seq( + ParamGatedCandidateFeatureHydrator( + EnableAdvertiserBrandSafetySettingsFeatureHydratorParam, + advertiserBrandSafetySettingsFeatureHydrator + ) + ), + decorator = Some(decorator), + alerts = alerts, + urtRequest = Some(true), + ) +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/for_you/ForYouConversationServiceCandidatePipelineConfig.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/for_you/ForYouConversationServiceCandidatePipelineConfig.scala new file mode 100644 index 0000000000..3f69ad888e --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/for_you/ForYouConversationServiceCandidatePipelineConfig.scala @@ -0,0 +1,135 @@ +package com.twitter.home_mixer.product.for_you + +import com.twitter.home_mixer.candidate_pipeline.ConversationServiceResponseFeatureTransformer +import com.twitter.home_mixer.functional_component.decorator.HomeConversationServiceCandidateDecorator +import com.twitter.home_mixer.functional_component.decorator.HomeFeedbackActionInfoBuilder +import com.twitter.home_mixer.functional_component.feature_hydrator.NamesFeatureHydrator +import com.twitter.home_mixer.functional_component.feature_hydrator.SocialGraphServiceFeatureHydrator +import com.twitter.home_mixer.functional_component.feature_hydrator.TimelineServiceTweetsFeature +import com.twitter.home_mixer.functional_component.feature_hydrator.TweetypieFeatureHydrator +import com.twitter.home_mixer.functional_component.filter.InvalidConversationModuleFilter +import com.twitter.home_mixer.functional_component.filter.PredicateFeatureFilter +import com.twitter.home_mixer.functional_component.filter.PreviouslySeenTweetsFilter +import com.twitter.home_mixer.functional_component.filter.PreviouslyServedTweetsFilter +import com.twitter.home_mixer.functional_component.filter.RetweetDeduplicationFilter +import com.twitter.home_mixer.functional_component.gate.NonEmptySeqFeatureGate +import com.twitter.home_mixer.model.HomeFeatures.IsHydratedFeature +import com.twitter.home_mixer.model.HomeFeatures.QuotedTweetDroppedFeature +import com.twitter.home_mixer.product.for_you.model.ForYouQuery +import com.twitter.home_mixer.service.HomeMixerAlertConfig +import com.twitter.product_mixer.component_library.candidate_source.tweetconvosvc.ConversationServiceCandidateSource +import com.twitter.product_mixer.component_library.candidate_source.tweetconvosvc.ConversationServiceCandidateSourceRequest +import com.twitter.product_mixer.component_library.candidate_source.tweetconvosvc.TweetWithConversationMetadata +import com.twitter.product_mixer.component_library.filter.FeatureFilter +import com.twitter.product_mixer.component_library.gate.NoCandidatesGate +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.functional_component.candidate_source.BaseCandidateSource +import com.twitter.product_mixer.core.functional_component.common.SpecificPipelines +import com.twitter.product_mixer.core.functional_component.decorator.CandidateDecorator +import com.twitter.product_mixer.core.functional_component.feature_hydrator.BaseCandidateFeatureHydrator +import com.twitter.product_mixer.core.functional_component.filter.Filter +import com.twitter.product_mixer.core.functional_component.gate.BaseGate +import com.twitter.product_mixer.core.functional_component.transformer.CandidateFeatureTransformer +import com.twitter.product_mixer.core.functional_component.transformer.CandidatePipelineResultsTransformer +import com.twitter.product_mixer.core.functional_component.transformer.DependentCandidatePipelineQueryTransformer +import com.twitter.product_mixer.core.model.common.identifier.CandidatePipelineIdentifier +import com.twitter.product_mixer.core.model.common.identifier.FilterIdentifier +import com.twitter.product_mixer.core.pipeline.candidate.DependentCandidatePipelineConfig +import javax.inject.Inject +import javax.inject.Singleton + +/** + * Candidate Pipeline Config that fetches Tweet ancestors from Conversation Service Candidate Source + */ +@Singleton +class ForYouConversationServiceCandidatePipelineConfig @Inject() ( + forYouScoredTweetsCandidatePipelineConfig: ForYouScoredTweetsCandidatePipelineConfig, + forYouTimelineScorerCandidatePipelineConfig: ForYouTimelineScorerCandidatePipelineConfig, + conversationServiceCandidateSource: ConversationServiceCandidateSource, + tweetypieFeatureHydrator: TweetypieFeatureHydrator, + socialGraphServiceFeatureHydrator: SocialGraphServiceFeatureHydrator, + namesFeatureHydrator: NamesFeatureHydrator, + homeFeedbackActionInfoBuilder: HomeFeedbackActionInfoBuilder) + extends DependentCandidatePipelineConfig[ + ForYouQuery, + ConversationServiceCandidateSourceRequest, + TweetWithConversationMetadata, + TweetCandidate + ] { + + override val identifier: CandidatePipelineIdentifier = + CandidatePipelineIdentifier("ForYouConversationService") + + override val gates: Seq[BaseGate[ForYouQuery]] = Seq( + NoCandidatesGate( + SpecificPipelines( + forYouTimelineScorerCandidatePipelineConfig.identifier, + forYouScoredTweetsCandidatePipelineConfig.identifier + ) + ), + NonEmptySeqFeatureGate(TimelineServiceTweetsFeature) + ) + + override val candidateSource: BaseCandidateSource[ + ConversationServiceCandidateSourceRequest, + TweetWithConversationMetadata + ] = conversationServiceCandidateSource + + override val queryTransformer: DependentCandidatePipelineQueryTransformer[ + ForYouQuery, + ConversationServiceCandidateSourceRequest + ] = { (query, candidates) => + val timelineServiceTweets = query.features + .map(_.getOrElse(TimelineServiceTweetsFeature, Seq.empty)).getOrElse(Seq.empty) + + val tweetsWithConversationMetadata = timelineServiceTweets.map { id => + TweetWithConversationMetadata( + tweetId = id, + userId = None, + sourceTweetId = None, + sourceUserId = None, + inReplyToTweetId = None, + conversationId = None, + ancestors = Seq.empty + ) + } + ConversationServiceCandidateSourceRequest(tweetsWithConversationMetadata) + } + + override val featuresFromCandidateSourceTransformers: Seq[ + CandidateFeatureTransformer[TweetWithConversationMetadata] + ] = Seq(ConversationServiceResponseFeatureTransformer) + + override val resultTransformer: CandidatePipelineResultsTransformer[ + TweetWithConversationMetadata, + TweetCandidate + ] = { sourceResult => TweetCandidate(id = sourceResult.tweetId) } + + override val preFilterFeatureHydrationPhase1: Seq[ + BaseCandidateFeatureHydrator[ForYouQuery, TweetCandidate, _] + ] = Seq(tweetypieFeatureHydrator, socialGraphServiceFeatureHydrator) + + override def filters: Seq[Filter[ForYouQuery, TweetCandidate]] = Seq( + PreviouslyServedTweetsFilter, + PreviouslySeenTweetsFilter, + RetweetDeduplicationFilter, + FeatureFilter.fromFeature(FilterIdentifier("TweetypieHydrated"), IsHydratedFeature), + PredicateFeatureFilter.fromPredicate( + FilterIdentifier("QuotedTweetDropped"), + shouldKeepCandidate = { features => !features.getOrElse(QuotedTweetDroppedFeature, false) } + ), + InvalidConversationModuleFilter + ) + + override val postFilterFeatureHydration: Seq[ + BaseCandidateFeatureHydrator[ForYouQuery, TweetCandidate, _] + ] = Seq(namesFeatureHydrator) + + override val decorator: Option[CandidateDecorator[ForYouQuery, TweetCandidate]] = + HomeConversationServiceCandidateDecorator(homeFeedbackActionInfoBuilder) + + override val alerts = Seq( + HomeMixerAlertConfig.BusinessHours.defaultSuccessRateAlert(), + HomeMixerAlertConfig.BusinessHours.defaultEmptyResponseRateAlert() + ) +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/for_you/ForYouProductPipelineConfig.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/for_you/ForYouProductPipelineConfig.scala new file mode 100644 index 0000000000..e84b05389b --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/for_you/ForYouProductPipelineConfig.scala @@ -0,0 +1,135 @@ +package com.twitter.home_mixer.product.for_you + +import com.twitter.conversions.DurationOps._ +import com.twitter.home_mixer.marshaller.timelines.ChronologicalCursorUnmarshaller +import com.twitter.home_mixer.model.request.HomeMixerRequest +import com.twitter.home_mixer.model.request.ForYouProduct +import com.twitter.home_mixer.model.request.ForYouProductContext +import com.twitter.home_mixer.product.for_you.model.ForYouQuery +import com.twitter.home_mixer.product.for_you.param.ForYouParam.EnableScoredTweetsMixerPipelineParam +import com.twitter.home_mixer.product.for_you.param.ForYouParam.ServerMaxResultsParam +import com.twitter.home_mixer.product.for_you.param.ForYouParamConfig +import com.twitter.home_mixer.service.HomeMixerAccessPolicy.DefaultHomeMixerAccessPolicy +import com.twitter.home_mixer.service.HomeMixerAlertConfig.DefaultNotificationGroup +import com.twitter.product_mixer.component_library.model.cursor.UrtOrderedCursor +import com.twitter.product_mixer.component_library.premarshaller.cursor.UrtCursorSerializer +import com.twitter.product_mixer.core.functional_component.common.access_policy.AccessPolicy +import com.twitter.product_mixer.core.functional_component.common.alert.Alert +import com.twitter.product_mixer.core.functional_component.common.alert.EmptyResponseRateAlert +import com.twitter.product_mixer.core.functional_component.common.alert.LatencyAlert +import com.twitter.product_mixer.core.functional_component.common.alert.P99 +import com.twitter.product_mixer.core.functional_component.common.alert.SuccessRateAlert +import com.twitter.product_mixer.core.functional_component.common.alert.ThroughputAlert +import com.twitter.product_mixer.core.functional_component.common.alert.predicate.TriggerIfAbove +import com.twitter.product_mixer.core.functional_component.common.alert.predicate.TriggerIfBelow +import com.twitter.product_mixer.core.functional_component.common.alert.predicate.TriggerIfLatencyAbove +import com.twitter.product_mixer.core.model.common.identifier.ComponentIdentifier +import com.twitter.product_mixer.core.model.common.identifier.ProductPipelineIdentifier +import com.twitter.product_mixer.core.model.marshalling.request.Product +import com.twitter.product_mixer.core.model.marshalling.response.urt.operation.TopCursor +import com.twitter.product_mixer.core.pipeline.PipelineConfig +import com.twitter.product_mixer.core.pipeline.pipeline_failure.BadRequest +import com.twitter.product_mixer.core.pipeline.pipeline_failure.PipelineFailure +import com.twitter.product_mixer.core.pipeline.product.ProductPipelineConfig +import com.twitter.product_mixer.core.product.ProductParamConfig +import com.twitter.product_mixer.core.util.SortIndexBuilder +import com.twitter.timelines.configapi.Params +import com.twitter.timelines.render.{thriftscala => urt} +import com.twitter.timelines.util.RequestCursorSerializer +import com.twitter.util.Time +import com.twitter.util.Try +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class ForYouProductPipelineConfig @Inject() ( + forYouTimelineScorerMixerPipelineConfig: ForYouTimelineScorerMixerPipelineConfig, + forYouScoredTweetsMixerPipelineConfig: ForYouScoredTweetsMixerPipelineConfig, + forYouParamConfig: ForYouParamConfig) + extends ProductPipelineConfig[HomeMixerRequest, ForYouQuery, urt.TimelineResponse] { + + override val identifier: ProductPipelineIdentifier = ProductPipelineIdentifier("ForYou") + + override val product: Product = ForYouProduct + + override val paramConfig: ProductParamConfig = forYouParamConfig + + override def pipelineQueryTransformer( + request: HomeMixerRequest, + params: Params + ): ForYouQuery = { + val context = request.productContext match { + case Some(context: ForYouProductContext) => context + case _ => throw PipelineFailure(BadRequest, "ForYouProductContext not found") + } + + val debugOptions = request.debugParams.flatMap(_.debugOptions) + + /** + * Unlike other clients, newly created tweets on Android have the sort index set to the current + * time instead of the top sort index + 1, so these tweets get stuck at the top of the timeline + * if subsequent timeline responses use the sort index from the previous response instead of + * the current time. + */ + val pipelineCursor = request.serializedRequestCursor.flatMap { cursor => + Try(UrtCursorSerializer.deserializeOrderedCursor(cursor)) + .getOrElse(ChronologicalCursorUnmarshaller(RequestCursorSerializer.deserialize(cursor))) + .map { + case topCursor @ UrtOrderedCursor(_, _, Some(TopCursor), _) => + val queryTime = debugOptions.flatMap(_.requestTimeOverride).getOrElse(Time.now) + topCursor.copy(initialSortIndex = SortIndexBuilder.timeToId(queryTime)) + case cursor => cursor + } + } + + ForYouQuery( + params = params, + clientContext = request.clientContext, + features = None, + pipelineCursor = pipelineCursor, + requestedMaxResults = Some(params(ServerMaxResultsParam)), + debugOptions = debugOptions, + deviceContext = context.deviceContext, + seenTweetIds = context.seenTweetIds, + dspClientContext = context.dspClientContext + ) + } + + override val pipelines: Seq[PipelineConfig] = + Seq(forYouTimelineScorerMixerPipelineConfig, forYouScoredTweetsMixerPipelineConfig) + + override def pipelineSelector( + query: ForYouQuery + ): ComponentIdentifier = { + if (query.params.getBoolean(EnableScoredTweetsMixerPipelineParam)) + forYouScoredTweetsMixerPipelineConfig.identifier + else + forYouTimelineScorerMixerPipelineConfig.identifier + } + + override val alerts: Seq[Alert] = Seq( + SuccessRateAlert( + notificationGroup = DefaultNotificationGroup, + warnPredicate = TriggerIfBelow(99.9, 20, 30), + criticalPredicate = TriggerIfBelow(99.9, 30, 30), + ), + LatencyAlert( + notificationGroup = DefaultNotificationGroup, + percentile = P99, + warnPredicate = TriggerIfLatencyAbove(2000.millis, 15, 30), + criticalPredicate = TriggerIfLatencyAbove(2100.millis, 15, 30) + ), + ThroughputAlert( + notificationGroup = DefaultNotificationGroup, + warnPredicate = TriggerIfAbove(70000), + criticalPredicate = TriggerIfAbove(80000) + ), + EmptyResponseRateAlert( + notificationGroup = DefaultNotificationGroup, + warnPredicate = TriggerIfAbove(2), + criticalPredicate = TriggerIfAbove(3) + ) + ) + + override val debugAccessPolicies: Set[AccessPolicy] = DefaultHomeMixerAccessPolicy +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/for_you/ForYouScoredTweetsCandidatePipelineConfig.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/for_you/ForYouScoredTweetsCandidatePipelineConfig.scala new file mode 100644 index 0000000000..a8e2c36daa --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/for_you/ForYouScoredTweetsCandidatePipelineConfig.scala @@ -0,0 +1,166 @@ +package com.twitter.home_mixer.product.for_you + +import com.twitter.home_mixer.functional_component.decorator.HomeFeedbackActionInfoBuilder +import com.twitter.home_mixer.functional_component.decorator.HomeTimelinesScoreInfoBuilder +import com.twitter.home_mixer.functional_component.decorator.HomeTweetSocialContextBuilder +import com.twitter.home_mixer.functional_component.decorator.builder.HomeClientEventInfoBuilder +import com.twitter.home_mixer.functional_component.decorator.builder.HomeConversationModuleMetadataBuilder +import com.twitter.home_mixer.functional_component.feature_hydrator.FocalTweetFeatureHydrator +import com.twitter.home_mixer.functional_component.feature_hydrator.NamesFeatureHydrator +import com.twitter.home_mixer.functional_component.feature_hydrator.PerspectiveFilteredSocialContextFeatureHydrator +import com.twitter.home_mixer.functional_component.feature_hydrator.SGSValidSocialContextFeatureHydrator +import com.twitter.home_mixer.functional_component.feature_hydrator.TweetypieFeatureHydrator +import com.twitter.home_mixer.functional_component.filter.FeedbackFatigueFilter +import com.twitter.home_mixer.functional_component.filter.InvalidConversationModuleFilter +import com.twitter.home_mixer.functional_component.filter.PredicateFeatureFilter +import com.twitter.home_mixer.functional_component.filter.SocialContextFilter +import com.twitter.home_mixer.functional_component.gate.SupportedLanguagesGate +import com.twitter.home_mixer.functional_component.scorer.FeedbackFatigueScorer +import com.twitter.home_mixer.model.HomeFeatures.ConversationModuleFocalTweetIdFeature +import com.twitter.home_mixer.model.HomeFeatures.InNetworkFeature +import com.twitter.home_mixer.model.HomeFeatures.IsHydratedFeature +import com.twitter.home_mixer.model.HomeFeatures.IsNsfwFeature +import com.twitter.home_mixer.model.HomeFeatures.QuotedTweetDroppedFeature +import com.twitter.home_mixer.product.for_you.candidate_source.ScoredTweetWithConversationMetadata +import com.twitter.home_mixer.product.for_you.candidate_source.ScoredTweetsProductCandidateSource +import com.twitter.home_mixer.product.for_you.model.ForYouQuery +import com.twitter.home_mixer.product.for_you.param.ForYouParam.EnableScoredTweetsCandidatePipelineParam +import com.twitter.home_mixer.service.HomeMixerAlertConfig +import com.twitter.product_mixer.component_library.decorator.urt.UrtItemCandidateDecorator +import com.twitter.product_mixer.component_library.decorator.urt.UrtMultipleModulesDecorator +import com.twitter.product_mixer.component_library.decorator.urt.builder.item.tweet.TweetCandidateUrtItemBuilder +import com.twitter.product_mixer.component_library.decorator.urt.builder.timeline_module.ManualModuleId +import com.twitter.product_mixer.component_library.decorator.urt.builder.timeline_module.StaticModuleDisplayTypeBuilder +import com.twitter.product_mixer.component_library.decorator.urt.builder.timeline_module.TimelineModuleBuilder +import com.twitter.product_mixer.component_library.filter.FeatureFilter +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.functional_component.candidate_source.CandidateSource +import com.twitter.product_mixer.core.functional_component.decorator.CandidateDecorator +import com.twitter.product_mixer.core.functional_component.feature_hydrator.BaseCandidateFeatureHydrator +import com.twitter.product_mixer.core.functional_component.filter.Filter +import com.twitter.product_mixer.core.functional_component.gate.Gate +import com.twitter.product_mixer.core.functional_component.scorer.Scorer +import com.twitter.product_mixer.core.functional_component.transformer.CandidateFeatureTransformer +import com.twitter.product_mixer.core.functional_component.transformer.CandidatePipelineQueryTransformer +import com.twitter.product_mixer.core.functional_component.transformer.CandidatePipelineResultsTransformer +import com.twitter.product_mixer.core.model.common.identifier.CandidatePipelineIdentifier +import com.twitter.product_mixer.core.model.common.identifier.FilterIdentifier +import com.twitter.product_mixer.core.model.marshalling.response.urt.EntryNamespace +import com.twitter.product_mixer.core.model.marshalling.response.urt.timeline_module.VerticalConversation +import com.twitter.product_mixer.core.pipeline.candidate.CandidatePipelineConfig +import com.twitter.timelines.configapi.decider.DeciderParam +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class ForYouScoredTweetsCandidatePipelineConfig @Inject() ( + scoredTweetsProductCandidateSource: ScoredTweetsProductCandidateSource, + tweetypieFeatureHydrator: TweetypieFeatureHydrator, + namesFeatureHydrator: NamesFeatureHydrator, + sgsValidSocialContextFeatureHydrator: SGSValidSocialContextFeatureHydrator, + perspectiveFilteredSocialContextFeatureHydrator: PerspectiveFilteredSocialContextFeatureHydrator, + focalTweetFeatureHydrator: FocalTweetFeatureHydrator, + homeFeedbackActionInfoBuilder: HomeFeedbackActionInfoBuilder, + homeTweetSocialContextBuilder: HomeTweetSocialContextBuilder) + extends CandidatePipelineConfig[ + ForYouQuery, + ForYouQuery, + ScoredTweetWithConversationMetadata, + TweetCandidate + ] { + + override val identifier: CandidatePipelineIdentifier = + CandidatePipelineIdentifier("ForYouScoredTweets") + + private val TweetypieHydratedFilterId = "TweetypieHydrated" + private val QuotedTweetDroppedFilterId = "QuotedTweetDropped" + private val OutOfNetworkNSFWFilterId = "OutOfNetworkNSFW" + private val ConversationModuleNamespace = EntryNamespace("home-conversation") + + override val gates: Seq[Gate[ForYouQuery]] = Seq(SupportedLanguagesGate) + + override val candidateSource: CandidateSource[ForYouQuery, ScoredTweetWithConversationMetadata] = + scoredTweetsProductCandidateSource + + override val enabledDeciderParam: Option[DeciderParam[Boolean]] = + Some(EnableScoredTweetsCandidatePipelineParam) + + override val queryTransformer: CandidatePipelineQueryTransformer[ForYouQuery, ForYouQuery] = + identity + + override val featuresFromCandidateSourceTransformers: Seq[ + CandidateFeatureTransformer[ScoredTweetWithConversationMetadata] + ] = Seq(ForYouScoredTweetsResponseFeatureTransformer) + + override val resultTransformer: CandidatePipelineResultsTransformer[ + ScoredTweetWithConversationMetadata, + TweetCandidate + ] = { sourceResults => TweetCandidate(sourceResults.tweetId) } + + override val preFilterFeatureHydrationPhase1: Seq[ + BaseCandidateFeatureHydrator[ForYouQuery, TweetCandidate, _] + ] = Seq( + namesFeatureHydrator, + tweetypieFeatureHydrator, + sgsValidSocialContextFeatureHydrator, + perspectiveFilteredSocialContextFeatureHydrator, + ) + + override val filters: Seq[Filter[ForYouQuery, TweetCandidate]] = Seq( + FeatureFilter.fromFeature(FilterIdentifier(TweetypieHydratedFilterId), IsHydratedFeature), + PredicateFeatureFilter.fromPredicate( + FilterIdentifier(QuotedTweetDroppedFilterId), + shouldKeepCandidate = { features => !features.getOrElse(QuotedTweetDroppedFeature, false) } + ), + PredicateFeatureFilter.fromPredicate( + FilterIdentifier(OutOfNetworkNSFWFilterId), + shouldKeepCandidate = { features => + features.getOrElse(InNetworkFeature, false) || + !features.getOrElse(IsNsfwFeature, false) + } + ), + FeedbackFatigueFilter, + SocialContextFilter, + InvalidConversationModuleFilter + ) + + override val postFilterFeatureHydration: Seq[ + BaseCandidateFeatureHydrator[ForYouQuery, TweetCandidate, _] + ] = Seq(focalTweetFeatureHydrator) + + override val scorers: Seq[Scorer[ForYouQuery, TweetCandidate]] = Seq(FeedbackFatigueScorer) + + override val decorator: Option[CandidateDecorator[ForYouQuery, TweetCandidate]] = { + val clientEventInfoBuilder = HomeClientEventInfoBuilder() + + val tweetItemBuilder = TweetCandidateUrtItemBuilder( + clientEventInfoBuilder = clientEventInfoBuilder, + socialContextBuilder = Some(homeTweetSocialContextBuilder), + timelinesScoreInfoBuilder = Some(HomeTimelinesScoreInfoBuilder), + feedbackActionInfoBuilder = Some(homeFeedbackActionInfoBuilder) + ) + + val tweetDecorator = UrtItemCandidateDecorator(tweetItemBuilder) + + val moduleBuilder = TimelineModuleBuilder( + entryNamespace = ConversationModuleNamespace, + clientEventInfoBuilder = clientEventInfoBuilder, + moduleIdGeneration = ManualModuleId(0L), + displayTypeBuilder = StaticModuleDisplayTypeBuilder(VerticalConversation), + metadataBuilder = Some(HomeConversationModuleMetadataBuilder()) + ) + + Some( + UrtMultipleModulesDecorator( + urtItemCandidateDecorator = tweetDecorator, + moduleBuilder = moduleBuilder, + groupByKey = (_, _, candidateFeatures) => + candidateFeatures.getOrElse(ConversationModuleFocalTweetIdFeature, None) + )) + } + + override val alerts = Seq( + HomeMixerAlertConfig.BusinessHours.defaultSuccessRateAlert(), + HomeMixerAlertConfig.BusinessHours.defaultEmptyResponseRateAlert(10, 20) + ) +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/for_you/ForYouScoredTweetsMixerPipelineConfig.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/for_you/ForYouScoredTweetsMixerPipelineConfig.scala new file mode 100644 index 0000000000..29cb636787 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/for_you/ForYouScoredTweetsMixerPipelineConfig.scala @@ -0,0 +1,317 @@ +package com.twitter.home_mixer.product.for_you + +import com.twitter.clientapp.{thriftscala => ca} +import com.twitter.goldfinch.api.AdsInjectionSurfaceAreas +import com.twitter.home_mixer.candidate_pipeline.EditedTweetsCandidatePipelineConfig +import com.twitter.home_mixer.candidate_pipeline.NewTweetsPillCandidatePipelineConfig +import com.twitter.home_mixer.functional_component.decorator.urt.builder.AddEntriesWithReplaceAndShowAlertAndCoverInstructionBuilder +import com.twitter.home_mixer.functional_component.feature_hydrator._ +import com.twitter.home_mixer.functional_component.selector.DebunchCandidates +import com.twitter.home_mixer.functional_component.selector.UpdateConversationModuleId +import com.twitter.home_mixer.functional_component.selector.UpdateHomeClientEventDetails +import com.twitter.home_mixer.functional_component.selector.UpdateNewTweetsPillDecoration +import com.twitter.home_mixer.functional_component.side_effect._ +import com.twitter.home_mixer.model.ClearCacheIncludeInstruction +import com.twitter.home_mixer.model.HomeFeatures.InNetworkFeature +import com.twitter.home_mixer.param.HomeGlobalParams.MaxNumberReplaceInstructionsParam +import com.twitter.home_mixer.product.following.model.HomeMixerExternalStrings +import com.twitter.home_mixer.product.for_you.model.ForYouQuery +import com.twitter.home_mixer.product.for_you.param.ForYouParam.ClearCacheOnPtr +import com.twitter.home_mixer.product.for_you.param.ForYouParam.EnableFlipInjectionModuleCandidatePipelineParam +import com.twitter.home_mixer.product.for_you.param.ForYouParam.FlipInlineInjectionModulePosition +import com.twitter.home_mixer.product.for_you.param.ForYouParam.ServerMaxResultsParam +import com.twitter.home_mixer.product.for_you.param.ForYouParam.WhoToFollowPositionParam +import com.twitter.home_mixer.util.CandidatesUtil +import com.twitter.logpipeline.client.common.EventPublisher +import com.twitter.product_mixer.component_library.feature_hydrator.query.async.AsyncQueryFeatureHydrator +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.component_library.pipeline.candidate.flexible_injection_pipeline.FlipPromptCandidatePipelineConfigBuilder +import com.twitter.product_mixer.component_library.pipeline.candidate.who_to_follow_module.WhoToFollowCandidatePipelineConfig +import com.twitter.product_mixer.component_library.premarshaller.urt.UrtDomainMarshaller +import com.twitter.product_mixer.component_library.premarshaller.urt.builder.ClearCacheInstructionBuilder +import com.twitter.product_mixer.component_library.premarshaller.urt.builder.OrderedBottomCursorBuilder +import com.twitter.product_mixer.component_library.premarshaller.urt.builder.OrderedTopCursorBuilder +import com.twitter.product_mixer.component_library.premarshaller.urt.builder.ReplaceAllEntries +import com.twitter.product_mixer.component_library.premarshaller.urt.builder.ReplaceEntryInstructionBuilder +import com.twitter.product_mixer.component_library.premarshaller.urt.builder.ShowAlertInstructionBuilder +import com.twitter.product_mixer.component_library.premarshaller.urt.builder.ShowCoverInstructionBuilder +import com.twitter.product_mixer.component_library.premarshaller.urt.builder.StaticTimelineScribeConfigBuilder +import com.twitter.product_mixer.component_library.premarshaller.urt.builder.UrtMetadataBuilder +import com.twitter.product_mixer.component_library.selector.DropMaxCandidates +import com.twitter.product_mixer.component_library.selector.DropMaxModuleItemCandidates +import com.twitter.product_mixer.component_library.selector.DropModuleTooFewModuleItemResults +import com.twitter.product_mixer.component_library.selector.InsertAppendResults +import com.twitter.product_mixer.component_library.selector.InsertFixedPositionResults +import com.twitter.product_mixer.component_library.selector.SelectConditionally +import com.twitter.product_mixer.component_library.selector.UpdateSortCandidates +import com.twitter.product_mixer.component_library.selector.UpdateSortModuleItemCandidates +import com.twitter.product_mixer.component_library.selector.ads.AdsInjector +import com.twitter.product_mixer.component_library.selector.ads.InsertAdResults +import com.twitter.product_mixer.core.functional_component.common.SpecificPipeline +import com.twitter.product_mixer.core.functional_component.common.SpecificPipelines +import com.twitter.product_mixer.core.functional_component.configapi.StaticParam +import com.twitter.product_mixer.core.functional_component.feature_hydrator.QueryFeatureHydrator +import com.twitter.product_mixer.core.functional_component.marshaller.TransportMarshaller +import com.twitter.product_mixer.core.functional_component.marshaller.response.urt.UrtTransportMarshaller +import com.twitter.product_mixer.core.functional_component.premarshaller.DomainMarshaller +import com.twitter.product_mixer.core.functional_component.selector.Selector +import com.twitter.product_mixer.core.functional_component.side_effect.PipelineResultSideEffect +import com.twitter.product_mixer.core.model.common.UniversalNoun +import com.twitter.product_mixer.core.model.common.identifier.CandidatePipelineIdentifier +import com.twitter.product_mixer.core.model.common.identifier.MixerPipelineIdentifier +import com.twitter.product_mixer.core.model.common.presentation.ItemCandidateWithDetails +import com.twitter.product_mixer.core.model.common.presentation.ModuleCandidateWithDetails +import com.twitter.product_mixer.core.model.marshalling.response.urt.Timeline +import com.twitter.product_mixer.core.model.marshalling.response.urt.TimelineModule +import com.twitter.product_mixer.core.model.marshalling.response.urt.TimelineScribeConfig +import com.twitter.product_mixer.core.model.marshalling.response.urt.item.tweet.TweetItem +import com.twitter.product_mixer.core.pipeline.FailOpenPolicy +import com.twitter.product_mixer.core.pipeline.candidate.CandidatePipelineConfig +import com.twitter.product_mixer.core.pipeline.candidate.DependentCandidatePipelineConfig +import com.twitter.product_mixer.core.pipeline.mixer.MixerPipelineConfig +import com.twitter.product_mixer.core.product.guice.scope.ProductScoped +import com.twitter.stringcenter.client.StringCenter +import com.twitter.timelines.render.{thriftscala => urt} +import javax.inject.Inject +import javax.inject.Provider +import javax.inject.Singleton + +@Singleton +class ForYouScoredTweetsMixerPipelineConfig @Inject() ( + forYouScoredTweetsCandidatePipelineConfig: ForYouScoredTweetsCandidatePipelineConfig, + forYouConversationServiceCandidatePipelineConfig: ForYouConversationServiceCandidatePipelineConfig, + forYouAdsCandidatePipelineBuilder: ForYouAdsCandidatePipelineBuilder, + forYouWhoToFollowCandidatePipelineConfigBuilder: ForYouWhoToFollowCandidatePipelineConfigBuilder, + flipPromptCandidatePipelineConfigBuilder: FlipPromptCandidatePipelineConfigBuilder, + editedTweetsCandidatePipelineConfig: EditedTweetsCandidatePipelineConfig, + newTweetsPillCandidatePipelineConfig: NewTweetsPillCandidatePipelineConfig[ForYouQuery], + dismissInfoQueryFeatureHydrator: DismissInfoQueryFeatureHydrator, + gizmoduckUserQueryFeatureHydrator: GizmoduckUserQueryFeatureHydrator, + persistenceStoreQueryFeatureHydrator: PersistenceStoreQueryFeatureHydrator, + requestQueryFeatureHydrator: RequestQueryFeatureHydrator[ForYouQuery], + feedbackHistoryQueryFeatureHydrator: FeedbackHistoryQueryFeatureHydrator, + timelineServiceTweetsQueryFeatureHydrator: TimelineServiceTweetsQueryFeatureHydrator, + adsInjector: AdsInjector, + servedCandidateKeysKafkaSideEffectBuilder: ServedCandidateKeysKafkaSideEffectBuilder, + servedCandidateFeatureKeysKafkaSideEffectBuilder: ServedCandidateFeatureKeysKafkaSideEffectBuilder, + updateTimelinesPersistenceStoreSideEffect: UpdateTimelinesPersistenceStoreSideEffect, + truncateTimelinesPersistenceStoreSideEffect: TruncateTimelinesPersistenceStoreSideEffect, + homeScribeServedEntriesSideEffect: HomeScribeServedEntriesSideEffect, + servedStatsSideEffect: ServedStatsSideEffect, + clientEventsScribeEventPublisher: EventPublisher[ca.LogEvent], + externalStrings: HomeMixerExternalStrings, + @ProductScoped stringCenterProvider: Provider[StringCenter], + urtTransportMarshaller: UrtTransportMarshaller) + extends MixerPipelineConfig[ForYouQuery, Timeline, urt.TimelineResponse] { + + override val identifier: MixerPipelineIdentifier = MixerPipelineIdentifier("ForYouScoredTweets") + + private val MaxConsecutiveOutOfNetworkCandidates = 2 + + private val dependentCandidatesStep = MixerPipelineConfig.dependentCandidatePipelinesStep + + override val fetchQueryFeatures: Seq[QueryFeatureHydrator[ForYouQuery]] = Seq( + requestQueryFeatureHydrator, + persistenceStoreQueryFeatureHydrator, + timelineServiceTweetsQueryFeatureHydrator, + feedbackHistoryQueryFeatureHydrator, + AsyncQueryFeatureHydrator(dependentCandidatesStep, dismissInfoQueryFeatureHydrator), + AsyncQueryFeatureHydrator(dependentCandidatesStep, gizmoduckUserQueryFeatureHydrator), + ) + + private val forYouAdsCandidatePipelineConfig = forYouAdsCandidatePipelineBuilder.build() + + private val forYouWhoToFollowCandidatePipelineConfig = + forYouWhoToFollowCandidatePipelineConfigBuilder.build() + + private val flipPromptCandidatePipelineConfig = + flipPromptCandidatePipelineConfigBuilder.build[ForYouQuery]( + supportedClientParam = Some(EnableFlipInjectionModuleCandidatePipelineParam) + ) + + override val candidatePipelines: Seq[CandidatePipelineConfig[ForYouQuery, _, _, _]] = Seq( + forYouScoredTweetsCandidatePipelineConfig, + forYouAdsCandidatePipelineConfig, + forYouWhoToFollowCandidatePipelineConfig, + flipPromptCandidatePipelineConfig + ) + + override val dependentCandidatePipelines: Seq[ + DependentCandidatePipelineConfig[ForYouQuery, _, _, _] + ] = Seq( + forYouConversationServiceCandidatePipelineConfig, + editedTweetsCandidatePipelineConfig, + newTweetsPillCandidatePipelineConfig + ) + + override val failOpenPolicies: Map[CandidatePipelineIdentifier, FailOpenPolicy] = Map( + forYouScoredTweetsCandidatePipelineConfig.identifier -> FailOpenPolicy.Always, + forYouAdsCandidatePipelineConfig.identifier -> FailOpenPolicy.Always, + forYouWhoToFollowCandidatePipelineConfig.identifier -> FailOpenPolicy.Always, + flipPromptCandidatePipelineConfig.identifier -> FailOpenPolicy.Always, + editedTweetsCandidatePipelineConfig.identifier -> FailOpenPolicy.Always, + newTweetsPillCandidatePipelineConfig.identifier -> FailOpenPolicy.Always, + ) + + override val resultSelectors: Seq[Selector[ForYouQuery]] = Seq( + UpdateSortCandidates( + ordering = CandidatesUtil.reverseChronTweetsOrdering, + candidatePipeline = forYouConversationServiceCandidatePipelineConfig.identifier + ), + UpdateSortCandidates( + ordering = CandidatesUtil.scoreOrdering, + candidatePipeline = forYouScoredTweetsCandidatePipelineConfig.identifier + ), + UpdateSortModuleItemCandidates( + candidatePipeline = forYouScoredTweetsCandidatePipelineConfig.identifier, + ordering = CandidatesUtil.conversationModuleTweetsOrdering + ), + DebunchCandidates( + pipelineScope = SpecificPipeline(forYouScoredTweetsCandidatePipelineConfig.identifier), + mustDebunch = { + case item: ItemCandidateWithDetails => + !item.features.getOrElse(InNetworkFeature, false) + case module: ModuleCandidateWithDetails => + !module.candidates.last.features.getOrElse(InNetworkFeature, false) + }, + maxBunchSize = MaxConsecutiveOutOfNetworkCandidates + ), + UpdateConversationModuleId( + pipelineScope = SpecificPipeline(forYouScoredTweetsCandidatePipelineConfig.identifier) + ), + DropMaxCandidates( + candidatePipeline = forYouConversationServiceCandidatePipelineConfig.identifier, + maxSelectionsParam = ServerMaxResultsParam + ), + DropMaxCandidates( + candidatePipeline = forYouScoredTweetsCandidatePipelineConfig.identifier, + maxSelectionsParam = ServerMaxResultsParam + ), + DropMaxCandidates( + candidatePipeline = editedTweetsCandidatePipelineConfig.identifier, + maxSelectionsParam = MaxNumberReplaceInstructionsParam + ), + DropModuleTooFewModuleItemResults( + candidatePipeline = forYouWhoToFollowCandidatePipelineConfig.identifier, + minModuleItemsParam = StaticParam(WhoToFollowCandidatePipelineConfig.MinCandidatesSize) + ), + DropMaxModuleItemCandidates( + candidatePipeline = forYouWhoToFollowCandidatePipelineConfig.identifier, + maxModuleItemsParam = StaticParam(WhoToFollowCandidatePipelineConfig.MaxCandidatesSize) + ), + // The Conversation Service pipeline will only run if the Scored Tweets pipeline returned nothing + InsertAppendResults(candidatePipeline = + forYouConversationServiceCandidatePipelineConfig.identifier), + InsertAppendResults(candidatePipeline = forYouScoredTweetsCandidatePipelineConfig.identifier), + InsertFixedPositionResults( + candidatePipeline = forYouWhoToFollowCandidatePipelineConfig.identifier, + positionParam = WhoToFollowPositionParam + ), + InsertFixedPositionResults( + candidatePipeline = flipPromptCandidatePipelineConfig.identifier, + positionParam = FlipInlineInjectionModulePosition + ), + InsertAdResults( + surfaceAreaName = AdsInjectionSurfaceAreas.HomeTimeline, + adsInjector = adsInjector.forSurfaceArea(AdsInjectionSurfaceAreas.HomeTimeline), + adsCandidatePipeline = forYouAdsCandidatePipelineConfig.identifier + ), + // This selector must come after the tweets are inserted into the results + UpdateNewTweetsPillDecoration( + pipelineScope = SpecificPipelines( + forYouConversationServiceCandidatePipelineConfig.identifier, + forYouScoredTweetsCandidatePipelineConfig.identifier, + newTweetsPillCandidatePipelineConfig.identifier + ), + stringCenter = stringCenterProvider.get(), + seeNewTweetsString = externalStrings.seeNewTweetsString, + tweetedString = externalStrings.tweetedString + ), + InsertAppendResults(candidatePipeline = editedTweetsCandidatePipelineConfig.identifier), + SelectConditionally( + selector = + InsertAppendResults(candidatePipeline = newTweetsPillCandidatePipelineConfig.identifier), + includeSelector = (_, _, results) => CandidatesUtil.containsType[TweetCandidate](results) + ), + UpdateHomeClientEventDetails( + candidatePipelines = Set( + forYouConversationServiceCandidatePipelineConfig.identifier, + forYouScoredTweetsCandidatePipelineConfig.identifier + ) + ), + ) + + private val servedCandidateKeysKafkaSideEffect = + servedCandidateKeysKafkaSideEffectBuilder.build( + Set(forYouScoredTweetsCandidatePipelineConfig.identifier)) + + private val servedCandidateFeatureKeysKafkaSideEffect = + servedCandidateFeatureKeysKafkaSideEffectBuilder.build( + Set(forYouScoredTweetsCandidatePipelineConfig.identifier)) + + private val homeScribeClientEventSideEffect = HomeScribeClientEventSideEffect( + logPipelinePublisher = clientEventsScribeEventPublisher, + injectedTweetsCandidatePipelineIdentifiers = Seq( + forYouScoredTweetsCandidatePipelineConfig.identifier, + forYouConversationServiceCandidatePipelineConfig.identifier + ), + adsCandidatePipelineIdentifier = forYouAdsCandidatePipelineConfig.identifier, + whoToFollowCandidatePipelineIdentifier = + Some(forYouWhoToFollowCandidatePipelineConfig.identifier), + ) + + override val resultSideEffects: Seq[PipelineResultSideEffect[ForYouQuery, Timeline]] = Seq( + servedCandidateKeysKafkaSideEffect, + servedCandidateFeatureKeysKafkaSideEffect, + updateTimelinesPersistenceStoreSideEffect, + truncateTimelinesPersistenceStoreSideEffect, + homeScribeClientEventSideEffect, + homeScribeServedEntriesSideEffect, + servedStatsSideEffect + ) + + override val domainMarshaller: DomainMarshaller[ForYouQuery, Timeline] = { + val instructionBuilders = Seq( + ClearCacheInstructionBuilder( + ClearCacheIncludeInstruction( + ClearCacheOnPtr.EnableParam, + ClearCacheOnPtr.MinEntriesParam, + ) + ), + ReplaceEntryInstructionBuilder(ReplaceAllEntries), + // excludes alert, cover, and replace candidates + AddEntriesWithReplaceAndShowAlertAndCoverInstructionBuilder(), + ShowAlertInstructionBuilder(), + ShowCoverInstructionBuilder(), + ) + + val idSelector: PartialFunction[UniversalNoun[_], Long] = { + // exclude ads while determining tweet cursor values + case item: TweetItem if item.promotedMetadata.isEmpty => item.id + case module: TimelineModule + if module.items.headOption.exists(_.item.isInstanceOf[TweetItem]) => + module.items.last.item match { case item: TweetItem => item.id } + } + val topCursorBuilder = OrderedTopCursorBuilder(idSelector) + val bottomCursorBuilder = OrderedBottomCursorBuilder(idSelector) + + val metadataBuilder = UrtMetadataBuilder( + title = None, + scribeConfigBuilder = Some( + StaticTimelineScribeConfigBuilder( + TimelineScribeConfig( + page = Some("for_you_scored_tweets"), + section = None, + entityToken = None))) + ) + + UrtDomainMarshaller( + instructionBuilders = instructionBuilders, + metadataBuilder = Some(metadataBuilder), + cursorBuilders = Seq(topCursorBuilder, bottomCursorBuilder) + ) + } + + override val transportMarshaller: TransportMarshaller[Timeline, urt.TimelineResponse] = + urtTransportMarshaller +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/for_you/ForYouScoredTweetsResponseFeatureTransformer.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/for_you/ForYouScoredTweetsResponseFeatureTransformer.scala new file mode 100644 index 0000000000..9d8b43f732 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/for_you/ForYouScoredTweetsResponseFeatureTransformer.scala @@ -0,0 +1,78 @@ +package com.twitter.home_mixer.product.for_you + +import com.twitter.timelines.render.{thriftscala => tl} +import com.twitter.home_mixer.model.HomeFeatures._ +import com.twitter.home_mixer.product.for_you.candidate_source.ScoredTweetWithConversationMetadata +import com.twitter.product_mixer.core.feature.Feature +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.feature.featuremap.FeatureMapBuilder +import com.twitter.product_mixer.core.functional_component.transformer.CandidateFeatureTransformer +import com.twitter.product_mixer.core.model.common.identifier.TransformerIdentifier +import com.twitter.product_mixer.core.model.marshalling.response.urt.metadata.BasicTopicContextFunctionalityType +import com.twitter.product_mixer.core.model.marshalling.response.urt.metadata.RecWithEducationTopicContextFunctionalityType +import com.twitter.product_mixer.core.model.marshalling.response.urt.metadata.RecommendationTopicContextFunctionalityType + +object ForYouScoredTweetsResponseFeatureTransformer + extends CandidateFeatureTransformer[ScoredTweetWithConversationMetadata] { + + override val identifier: TransformerIdentifier = + TransformerIdentifier("ForYouScoredTweetsResponse") + + override val features: Set[Feature[_, _]] = Set( + AncestorsFeature, + AuthorIdFeature, + ConversationModuleIdFeature, + ConversationModuleFocalTweetIdFeature, + DirectedAtUserIdFeature, + FavoritedByUserIdsFeature, + FollowedByUserIdsFeature, + InNetworkFeature, + InReplyToTweetIdFeature, + InReplyToUserIdFeature, + IsReadFromCacheFeature, + IsRetweetFeature, + QuotedTweetIdFeature, + QuotedUserIdFeature, + ScoreFeature, + SourceTweetIdFeature, + SourceUserIdFeature, + StreamToKafkaFeature, + SuggestTypeFeature, + TopicContextFunctionalityTypeFeature, + TopicIdSocialContextFeature + ) + + override def transform(input: ScoredTweetWithConversationMetadata): FeatureMap = + FeatureMapBuilder() + .add(AncestorsFeature, input.ancestors.getOrElse(Seq.empty)) + .add(AuthorIdFeature, Some(input.authorId)) + .add(ConversationModuleIdFeature, input.conversationId) + .add(ConversationModuleFocalTweetIdFeature, input.conversationFocalTweetId) + .add(DirectedAtUserIdFeature, input.directedAtUserId) + .add(FavoritedByUserIdsFeature, input.favoritedByUserIds.getOrElse(Seq.empty)) + .add(FollowedByUserIdsFeature, input.followedByUserIds.getOrElse(Seq.empty)) + .add(InNetworkFeature, input.inNetwork.getOrElse(false)) + .add(InReplyToTweetIdFeature, input.inReplyToTweetId) + .add(InReplyToUserIdFeature, input.inReplyToUserId) + .add(IsReadFromCacheFeature, input.isReadFromCache.getOrElse(false)) + .add(IsRetweetFeature, input.sourceTweetId.isDefined) + .add(QuotedTweetIdFeature, input.quotedTweetId) + .add(QuotedUserIdFeature, input.quotedUserId) + .add(ScoreFeature, input.score) + .add(SourceTweetIdFeature, input.sourceTweetId) + .add(SourceUserIdFeature, input.sourceUserId) + .add(StreamToKafkaFeature, input.streamToKafka.getOrElse(false)) + .add(SuggestTypeFeature, input.suggestType) + .add( + TopicContextFunctionalityTypeFeature, + input.topicFunctionalityType.collect { + case tl.TopicContextFunctionalityType.Basic => BasicTopicContextFunctionalityType + case tl.TopicContextFunctionalityType.Recommendation => + RecommendationTopicContextFunctionalityType + case tl.TopicContextFunctionalityType.RecWithEducation => + RecWithEducationTopicContextFunctionalityType + } + ) + .add(TopicIdSocialContextFeature, input.topicId) + .build() +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/for_you/ForYouTimelineScorerCandidatePipelineConfig.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/for_you/ForYouTimelineScorerCandidatePipelineConfig.scala new file mode 100644 index 0000000000..37f93c2a19 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/for_you/ForYouTimelineScorerCandidatePipelineConfig.scala @@ -0,0 +1,237 @@ +package com.twitter.home_mixer.product.for_you + +import com.twitter.home_mixer.functional_component.decorator.HomeFeedbackActionInfoBuilder +import com.twitter.home_mixer.functional_component.decorator.HomeTimelinesScoreInfoBuilder +import com.twitter.home_mixer.functional_component.decorator.HomeTweetSocialContextBuilder +import com.twitter.home_mixer.functional_component.decorator.builder.HomeClientEventInfoBuilder +import com.twitter.home_mixer.functional_component.decorator.builder.HomeConversationModuleMetadataBuilder +import com.twitter.home_mixer.functional_component.feature_hydrator.FocalTweetFeatureHydrator +import com.twitter.home_mixer.functional_component.feature_hydrator.NamesFeatureHydrator +import com.twitter.home_mixer.functional_component.feature_hydrator.PerspectiveFilteredSocialContextFeatureHydrator +import com.twitter.home_mixer.functional_component.feature_hydrator.SGSValidSocialContextFeatureHydrator +import com.twitter.home_mixer.functional_component.feature_hydrator.SocialGraphServiceFeatureHydrator +import com.twitter.home_mixer.functional_component.feature_hydrator.TimelineServiceTweetsFeature +import com.twitter.home_mixer.functional_component.feature_hydrator.TweetypieFeatureHydrator +import com.twitter.home_mixer.functional_component.filter.FeedbackFatigueFilter +import com.twitter.home_mixer.functional_component.filter.InvalidConversationModuleFilter +import com.twitter.home_mixer.functional_component.filter.PredicateFeatureFilter +import com.twitter.home_mixer.functional_component.filter.RejectTweetFromViewerFilter +import com.twitter.home_mixer.functional_component.filter.RetweetDeduplicationFilter +import com.twitter.home_mixer.functional_component.filter.SocialContextFilter +import com.twitter.home_mixer.functional_component.scorer.FeedbackFatigueScorer +import com.twitter.home_mixer.functional_component.scorer.OONTweetScalingScorer +import com.twitter.home_mixer.marshaller.timelines.DeviceContextMarshaller +import com.twitter.home_mixer.marshaller.timelines.TimelineServiceCursorMarshaller +import com.twitter.home_mixer.model.HomeFeatures.ConversationModuleFocalTweetIdFeature +import com.twitter.home_mixer.model.HomeFeatures.InNetworkFeature +import com.twitter.home_mixer.model.HomeFeatures.IsHydratedFeature +import com.twitter.home_mixer.model.HomeFeatures.IsNsfwFeature +import com.twitter.home_mixer.model.HomeFeatures.QuotedTweetDroppedFeature +import com.twitter.home_mixer.model.request.DeviceContext +import com.twitter.home_mixer.product.for_you.model.ForYouQuery +import com.twitter.home_mixer.product.for_you.param.ForYouParam.EnableTimelineScorerCandidatePipelineParam +import com.twitter.home_mixer.service.HomeMixerAlertConfig +import com.twitter.product_mixer.component_library.candidate_source.timeline_scorer.ScoredTweetCandidateWithFocalTweet +import com.twitter.product_mixer.component_library.candidate_source.timeline_scorer.TimelineScorerCandidateSource +import com.twitter.product_mixer.component_library.decorator.urt.UrtItemCandidateDecorator +import com.twitter.product_mixer.component_library.decorator.urt.UrtMultipleModulesDecorator +import com.twitter.product_mixer.component_library.decorator.urt.builder.item.tweet.TweetCandidateUrtItemBuilder +import com.twitter.product_mixer.component_library.decorator.urt.builder.timeline_module.ManualModuleId +import com.twitter.product_mixer.component_library.decorator.urt.builder.timeline_module.StaticModuleDisplayTypeBuilder +import com.twitter.product_mixer.component_library.decorator.urt.builder.timeline_module.TimelineModuleBuilder +import com.twitter.product_mixer.component_library.filter.FeatureFilter +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.functional_component.candidate_source.BaseCandidateSource +import com.twitter.product_mixer.core.functional_component.common.alert.Alert +import com.twitter.product_mixer.core.functional_component.decorator.CandidateDecorator +import com.twitter.product_mixer.core.functional_component.feature_hydrator.BaseCandidateFeatureHydrator +import com.twitter.product_mixer.core.functional_component.filter.Filter +import com.twitter.product_mixer.core.functional_component.scorer.Scorer +import com.twitter.product_mixer.core.functional_component.transformer.CandidateFeatureTransformer +import com.twitter.product_mixer.core.functional_component.transformer.CandidatePipelineQueryTransformer +import com.twitter.product_mixer.core.functional_component.transformer.CandidatePipelineResultsTransformer +import com.twitter.product_mixer.core.model.common.identifier.CandidatePipelineIdentifier +import com.twitter.product_mixer.core.model.common.identifier.FilterIdentifier +import com.twitter.product_mixer.core.model.marshalling.response.urt.EntryNamespace +import com.twitter.product_mixer.core.model.marshalling.response.urt.timeline_module.VerticalConversation +import com.twitter.product_mixer.core.pipeline.candidate.CandidatePipelineConfig +import com.twitter.timelines.configapi.FSParam +import com.twitter.timelines.model.candidate.CandidateTweetSourceId +import com.twitter.timelines.service.{thriftscala => tst} +import com.twitter.timelinescorer.{thriftscala => t} +import com.twitter.timelineservice.{thriftscala => tlst} +import javax.inject.Inject +import javax.inject.Singleton + +/** + * Candidate Pipeline Config that fetches tweets from the Timeline Scorer Candidate Source + */ +@Singleton +class ForYouTimelineScorerCandidatePipelineConfig @Inject() ( + timelineScorerCandidateSource: TimelineScorerCandidateSource, + deviceContextMarshaller: DeviceContextMarshaller, + tweetypieFeatureHydrator: TweetypieFeatureHydrator, + sgsFeatureHydrator: SocialGraphServiceFeatureHydrator, + sgsValidSocialContextFeatureHydrator: SGSValidSocialContextFeatureHydrator, + perspectiveFilteredSocialContextFeatureHydrator: PerspectiveFilteredSocialContextFeatureHydrator, + namesFeatureHydrator: NamesFeatureHydrator, + focalTweetFeatureHydrator: FocalTweetFeatureHydrator, + homeFeedbackActionInfoBuilder: HomeFeedbackActionInfoBuilder, + homeTweetSocialContextBuilder: HomeTweetSocialContextBuilder) + extends CandidatePipelineConfig[ + ForYouQuery, + t.ScoredTweetsRequest, + ScoredTweetCandidateWithFocalTweet, + TweetCandidate + ] { + + override val identifier: CandidatePipelineIdentifier = + CandidatePipelineIdentifier("ForYouTimelineScorerTweets") + + private val TweetypieHydratedFilterId = "TweetypieHydrated" + private val QuotedTweetDroppedFilterId = "QuotedTweetDropped" + private val OutOfNetworkNSFWFilterId = "OutOfNetworkNSFW" + private val ConversationModuleNamespace = EntryNamespace("home-conversation") + + override val supportedClientParam: Option[FSParam[Boolean]] = + Some(EnableTimelineScorerCandidatePipelineParam) + + override val candidateSource: BaseCandidateSource[ + t.ScoredTweetsRequest, + ScoredTweetCandidateWithFocalTweet + ] = timelineScorerCandidateSource + + override val queryTransformer: CandidatePipelineQueryTransformer[ + ForYouQuery, + t.ScoredTweetsRequest + ] = { query => + val deviceContext = query.deviceContext.getOrElse(DeviceContext.Empty) + + val scoredTweetsRequestContext = t.v1.ScoredTweetsRequestContext( + contextualUserId = query.clientContext.userId, + timelineId = query.clientContext.userId.map(tlst.TimelineId(tlst.TimelineType.Home, _, None)), + deviceContext = Some(deviceContextMarshaller(deviceContext, query.clientContext)), + seenTweetIds = query.seenTweetIds, + contextualUserContext = Some(tst.ContextualUserContext(query.clientContext.userRoles)), + timelineRequestCursor = query.pipelineCursor.flatMap(TimelineServiceCursorMarshaller(_)) + ) + + val candidateTweetSourceIds = Seq( + CandidateTweetSourceId.RecycledTweet, + CandidateTweetSourceId.OrganicTweet, + CandidateTweetSourceId.AncestorsOnlyOrganicTweet, + CandidateTweetSourceId.BackfillOrganicTweet, + CandidateTweetSourceId.CroonTweet, + CandidateTweetSourceId.RecommendedTweet, + CandidateTweetSourceId.FrsTweet, + CandidateTweetSourceId.ListTweet + ) + + val timelineServiceTweets = + query.features.map(_.getOrElse(TimelineServiceTweetsFeature, Seq.empty)).getOrElse(Seq.empty) + + val timelineEntries = timelineServiceTweets.map { id => + tlst.TimelineEntry.Tweet(tlst.Tweet(statusId = id, sortIndex = id)) + } + + t.ScoredTweetsRequest.V1( + t.v1.ScoredTweetsRequest( + scoredTweetsRequestContext = Some(scoredTweetsRequestContext), + candidateTweetSourceIds = + Some(candidateTweetSourceIds.flatMap(CandidateTweetSourceId.toThrift)), + maxResultsCount = query.requestedMaxResults, + organicTimeline = Some( + tlst.Timeline( + timelineId = tlst.TimelineId( + timelineType = tlst.TimelineType.Home, + id = query.getRequiredUserId, + canonicalTimelineId = None), + entries = timelineEntries, + modules = tlst.TimelineModules() + )) + ) + ) + } + + override val featuresFromCandidateSourceTransformers: Seq[ + CandidateFeatureTransformer[ScoredTweetCandidateWithFocalTweet] + ] = Seq(ForYouTimelineScorerResponseFeatureTransformer) + + override val resultTransformer: CandidatePipelineResultsTransformer[ + ScoredTweetCandidateWithFocalTweet, + TweetCandidate + ] = { candidateWithFocalTweetId => + TweetCandidate(id = candidateWithFocalTweetId.candidate.tweetId) + } + + override val preFilterFeatureHydrationPhase1: Seq[ + BaseCandidateFeatureHydrator[ForYouQuery, TweetCandidate, _] + ] = Seq( + tweetypieFeatureHydrator, + sgsFeatureHydrator, + sgsValidSocialContextFeatureHydrator, + perspectiveFilteredSocialContextFeatureHydrator, + namesFeatureHydrator + ) + + override def filters: Seq[Filter[ForYouQuery, TweetCandidate]] = Seq( + RetweetDeduplicationFilter, + FeatureFilter.fromFeature(FilterIdentifier(TweetypieHydratedFilterId), IsHydratedFeature), + PredicateFeatureFilter.fromPredicate( + FilterIdentifier(QuotedTweetDroppedFilterId), + shouldKeepCandidate = { features => !features.getOrElse(QuotedTweetDroppedFeature, false) } + ), + PredicateFeatureFilter.fromPredicate( + FilterIdentifier(OutOfNetworkNSFWFilterId), + shouldKeepCandidate = { features => + features.getOrElse(InNetworkFeature, false) || + !features.getOrElse(IsNsfwFeature, false) + } + ), + FeedbackFatigueFilter, + RejectTweetFromViewerFilter, + SocialContextFilter, + InvalidConversationModuleFilter + ) + + override val postFilterFeatureHydration: Seq[ + BaseCandidateFeatureHydrator[ForYouQuery, TweetCandidate, _] + ] = Seq(focalTweetFeatureHydrator) + + override val scorers: Seq[Scorer[ForYouQuery, TweetCandidate]] = + Seq(OONTweetScalingScorer, FeedbackFatigueScorer) + + override val decorator: Option[CandidateDecorator[ForYouQuery, TweetCandidate]] = { + val clientEventInfoBuilder = HomeClientEventInfoBuilder() + + val tweetItemBuilder = TweetCandidateUrtItemBuilder( + clientEventInfoBuilder = clientEventInfoBuilder, + socialContextBuilder = Some(homeTweetSocialContextBuilder), + timelinesScoreInfoBuilder = Some(HomeTimelinesScoreInfoBuilder), + feedbackActionInfoBuilder = Some(homeFeedbackActionInfoBuilder) + ) + + val tweetDecorator = UrtItemCandidateDecorator(tweetItemBuilder) + + val moduleBuilder = TimelineModuleBuilder( + entryNamespace = ConversationModuleNamespace, + clientEventInfoBuilder = clientEventInfoBuilder, + moduleIdGeneration = ManualModuleId(0L), + displayTypeBuilder = StaticModuleDisplayTypeBuilder(VerticalConversation), + metadataBuilder = Some(HomeConversationModuleMetadataBuilder()) + ) + + Some( + UrtMultipleModulesDecorator( + urtItemCandidateDecorator = tweetDecorator, + moduleBuilder = moduleBuilder, + groupByKey = (_, _, candidateFeatures) => + candidateFeatures.getOrElse(ConversationModuleFocalTweetIdFeature, None) + )) + } + + override val alerts: Seq[Alert] = Seq( + HomeMixerAlertConfig.BusinessHours.defaultSuccessRateAlert(), + HomeMixerAlertConfig.BusinessHours.defaultEmptyResponseRateAlert(10, 20) + ) +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/for_you/ForYouTimelineScorerMixerPipelineConfig.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/for_you/ForYouTimelineScorerMixerPipelineConfig.scala new file mode 100644 index 0000000000..b43c6ed673 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/for_you/ForYouTimelineScorerMixerPipelineConfig.scala @@ -0,0 +1,336 @@ +package com.twitter.home_mixer.product.for_you + +import com.twitter.clientapp.{thriftscala => ca} +import com.twitter.goldfinch.api.AdsInjectionSurfaceAreas +import com.twitter.home_mixer.candidate_pipeline.EditedTweetsCandidatePipelineConfig +import com.twitter.home_mixer.candidate_pipeline.NewTweetsPillCandidatePipelineConfig +import com.twitter.home_mixer.functional_component.decorator.urt.builder.AddEntriesWithReplaceAndShowAlertAndCoverInstructionBuilder +import com.twitter.home_mixer.functional_component.feature_hydrator._ +import com.twitter.home_mixer.functional_component.selector.DebunchCandidates +import com.twitter.home_mixer.functional_component.selector.UpdateConversationModuleId +import com.twitter.home_mixer.functional_component.selector.UpdateHomeClientEventDetails +import com.twitter.home_mixer.functional_component.selector.UpdateNewTweetsPillDecoration +import com.twitter.home_mixer.functional_component.side_effect._ +import com.twitter.home_mixer.model.ClearCacheIncludeInstruction +import com.twitter.home_mixer.model.HomeFeatures.InNetworkFeature +import com.twitter.home_mixer.param.HomeGlobalParams.MaxNumberReplaceInstructionsParam +import com.twitter.home_mixer.product.following.model.HomeMixerExternalStrings +import com.twitter.home_mixer.product.for_you.model.ForYouQuery +import com.twitter.home_mixer.product.for_you.param.ForYouParam.ClearCacheOnPtr +import com.twitter.home_mixer.product.for_you.param.ForYouParam.EnableFlipInjectionModuleCandidatePipelineParam +import com.twitter.home_mixer.product.for_you.param.ForYouParam.FlipInlineInjectionModulePosition +import com.twitter.home_mixer.product.for_you.param.ForYouParam.ServerMaxResultsParam +import com.twitter.home_mixer.product.for_you.param.ForYouParam.WhoToFollowPositionParam +import com.twitter.home_mixer.util.CandidatesUtil +import com.twitter.logpipeline.client.common.EventPublisher +import com.twitter.product_mixer.component_library.feature_hydrator.query.async.AsyncQueryFeatureHydrator +import com.twitter.product_mixer.component_library.feature_hydrator.query.impressed_tweets.ImpressedTweetsQueryFeatureHydrator +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.component_library.pipeline.candidate.flexible_injection_pipeline.FlipPromptCandidatePipelineConfigBuilder +import com.twitter.product_mixer.component_library.pipeline.candidate.who_to_follow_module.WhoToFollowCandidatePipelineConfig +import com.twitter.product_mixer.component_library.premarshaller.urt.UrtDomainMarshaller +import com.twitter.product_mixer.component_library.premarshaller.urt.builder.ClearCacheInstructionBuilder +import com.twitter.product_mixer.component_library.premarshaller.urt.builder.OrderedBottomCursorBuilder +import com.twitter.product_mixer.component_library.premarshaller.urt.builder.OrderedTopCursorBuilder +import com.twitter.product_mixer.component_library.premarshaller.urt.builder.ReplaceAllEntries +import com.twitter.product_mixer.component_library.premarshaller.urt.builder.ReplaceEntryInstructionBuilder +import com.twitter.product_mixer.component_library.premarshaller.urt.builder.ShowAlertInstructionBuilder +import com.twitter.product_mixer.component_library.premarshaller.urt.builder.ShowCoverInstructionBuilder +import com.twitter.product_mixer.component_library.premarshaller.urt.builder.StaticTimelineScribeConfigBuilder +import com.twitter.product_mixer.component_library.premarshaller.urt.builder.UrtMetadataBuilder +import com.twitter.product_mixer.component_library.selector.DropMaxCandidates +import com.twitter.product_mixer.component_library.selector.DropMaxModuleItemCandidates +import com.twitter.product_mixer.component_library.selector.DropModuleTooFewModuleItemResults +import com.twitter.product_mixer.component_library.selector.InsertAppendResults +import com.twitter.product_mixer.component_library.selector.InsertFixedPositionResults +import com.twitter.product_mixer.component_library.selector.SelectConditionally +import com.twitter.product_mixer.component_library.selector.UpdateSortCandidates +import com.twitter.product_mixer.component_library.selector.UpdateSortModuleItemCandidates +import com.twitter.product_mixer.component_library.selector.ads.AdsInjector +import com.twitter.product_mixer.component_library.selector.ads.InsertAdResults +import com.twitter.product_mixer.core.functional_component.common.SpecificPipeline +import com.twitter.product_mixer.core.functional_component.common.SpecificPipelines +import com.twitter.product_mixer.core.functional_component.configapi.StaticParam +import com.twitter.product_mixer.core.functional_component.feature_hydrator.QueryFeatureHydrator +import com.twitter.product_mixer.core.functional_component.marshaller.TransportMarshaller +import com.twitter.product_mixer.core.functional_component.marshaller.response.urt.UrtTransportMarshaller +import com.twitter.product_mixer.core.functional_component.premarshaller.DomainMarshaller +import com.twitter.product_mixer.core.functional_component.selector.Selector +import com.twitter.product_mixer.core.functional_component.side_effect.PipelineResultSideEffect +import com.twitter.product_mixer.core.model.common.UniversalNoun +import com.twitter.product_mixer.core.model.common.identifier.CandidatePipelineIdentifier +import com.twitter.product_mixer.core.model.common.identifier.MixerPipelineIdentifier +import com.twitter.product_mixer.core.model.common.presentation.ItemCandidateWithDetails +import com.twitter.product_mixer.core.model.common.presentation.ModuleCandidateWithDetails +import com.twitter.product_mixer.core.model.marshalling.response.urt.Timeline +import com.twitter.product_mixer.core.model.marshalling.response.urt.TimelineModule +import com.twitter.product_mixer.core.model.marshalling.response.urt.TimelineScribeConfig +import com.twitter.product_mixer.core.model.marshalling.response.urt.item.tweet.TweetItem +import com.twitter.product_mixer.core.pipeline.FailOpenPolicy +import com.twitter.product_mixer.core.pipeline.candidate.CandidatePipelineConfig +import com.twitter.product_mixer.core.pipeline.candidate.DependentCandidatePipelineConfig +import com.twitter.product_mixer.core.pipeline.mixer.MixerPipelineConfig +import com.twitter.product_mixer.core.product.guice.scope.ProductScoped +import com.twitter.stringcenter.client.StringCenter +import com.twitter.timelines.render.{thriftscala => urt} +import javax.inject.Inject +import javax.inject.Provider +import javax.inject.Singleton + +@Singleton +class ForYouTimelineScorerMixerPipelineConfig @Inject() ( + forYouTimelineScorerCandidatePipelineConfig: ForYouTimelineScorerCandidatePipelineConfig, + forYouConversationServiceCandidatePipelineConfig: ForYouConversationServiceCandidatePipelineConfig, + forYouAdsCandidatePipelineBuilder: ForYouAdsCandidatePipelineBuilder, + forYouWhoToFollowCandidatePipelineConfigBuilder: ForYouWhoToFollowCandidatePipelineConfigBuilder, + flipPromptCandidatePipelineConfigBuilder: FlipPromptCandidatePipelineConfigBuilder, + editedTweetsCandidatePipelineConfig: EditedTweetsCandidatePipelineConfig, + newTweetsPillCandidatePipelineConfig: NewTweetsPillCandidatePipelineConfig[ForYouQuery], + dismissInfoQueryFeatureHydrator: DismissInfoQueryFeatureHydrator, + gizmoduckUserQueryFeatureHydrator: GizmoduckUserQueryFeatureHydrator, + manhattanTweetImpressionsQueryFeatureHydrator: TweetImpressionsQueryFeatureHydrator[ForYouQuery], + memcacheTweetImpressionsQueryFeatureHydrator: ImpressedTweetsQueryFeatureHydrator, + persistenceStoreQueryFeatureHydrator: PersistenceStoreQueryFeatureHydrator, + requestQueryFeatureHydrator: RequestQueryFeatureHydrator[ForYouQuery], + timelineServiceTweetsQueryFeatureHydrator: TimelineServiceTweetsQueryFeatureHydrator, + lastNonPollingTimeQueryFeatureHydrator: LastNonPollingTimeQueryFeatureHydrator, + feedbackHistoryQueryFeatureHydrator: FeedbackHistoryQueryFeatureHydrator, + adsInjector: AdsInjector, + servedCandidateKeysKafkaSideEffectBuilder: ServedCandidateKeysKafkaSideEffectBuilder, + servedCandidateFeatureKeysKafkaSideEffectBuilder: ServedCandidateFeatureKeysKafkaSideEffectBuilder, + updateLastNonPollingTimeSideEffect: UpdateLastNonPollingTimeSideEffect[ForYouQuery, Timeline], + publishClientSentImpressionsEventBusSideEffect: PublishClientSentImpressionsEventBusSideEffect, + publishClientSentImpressionsManhattanSideEffect: PublishClientSentImpressionsManhattanSideEffect, + updateTimelinesPersistenceStoreSideEffect: UpdateTimelinesPersistenceStoreSideEffect, + truncateTimelinesPersistenceStoreSideEffect: TruncateTimelinesPersistenceStoreSideEffect, + homeScribeServedEntriesSideEffect: HomeScribeServedEntriesSideEffect, + servedStatsSideEffect: ServedStatsSideEffect, + clientEventsScribeEventPublisher: EventPublisher[ca.LogEvent], + externalStrings: HomeMixerExternalStrings, + @ProductScoped stringCenterProvider: Provider[StringCenter], + urtTransportMarshaller: UrtTransportMarshaller) + extends MixerPipelineConfig[ForYouQuery, Timeline, urt.TimelineResponse] { + + override val identifier: MixerPipelineIdentifier = MixerPipelineIdentifier("ForYouTimelineScorer") + + private val MaxConsecutiveOutOfNetworkCandidates = 2 + + private val dependentCandidatesStep = MixerPipelineConfig.dependentCandidatePipelinesStep + private val resultSelectorsStep = MixerPipelineConfig.resultSelectorsStep + + override def fetchQueryFeatures: Seq[QueryFeatureHydrator[ForYouQuery]] = Seq( + requestQueryFeatureHydrator, + persistenceStoreQueryFeatureHydrator, + timelineServiceTweetsQueryFeatureHydrator, + feedbackHistoryQueryFeatureHydrator, + AsyncQueryFeatureHydrator(dependentCandidatesStep, dismissInfoQueryFeatureHydrator), + AsyncQueryFeatureHydrator(dependentCandidatesStep, gizmoduckUserQueryFeatureHydrator), + AsyncQueryFeatureHydrator(dependentCandidatesStep, lastNonPollingTimeQueryFeatureHydrator), + AsyncQueryFeatureHydrator(resultSelectorsStep, manhattanTweetImpressionsQueryFeatureHydrator), + AsyncQueryFeatureHydrator(resultSelectorsStep, memcacheTweetImpressionsQueryFeatureHydrator) + ) + + private val forYouAdsCandidatePipelineConfig = forYouAdsCandidatePipelineBuilder.build() + + private val forYouWhoToFollowCandidatePipelineConfig = + forYouWhoToFollowCandidatePipelineConfigBuilder.build() + + private val flipPromptCandidatePipelineConfig = + flipPromptCandidatePipelineConfigBuilder.build[ForYouQuery]( + supportedClientParam = Some(EnableFlipInjectionModuleCandidatePipelineParam) + ) + + override val candidatePipelines: Seq[CandidatePipelineConfig[ForYouQuery, _, _, _]] = Seq( + forYouTimelineScorerCandidatePipelineConfig, + forYouAdsCandidatePipelineConfig, + forYouWhoToFollowCandidatePipelineConfig, + flipPromptCandidatePipelineConfig + ) + + override val dependentCandidatePipelines: Seq[ + DependentCandidatePipelineConfig[ForYouQuery, _, _, _] + ] = Seq( + forYouConversationServiceCandidatePipelineConfig, + editedTweetsCandidatePipelineConfig, + newTweetsPillCandidatePipelineConfig + ) + + override val failOpenPolicies: Map[CandidatePipelineIdentifier, FailOpenPolicy] = Map( + forYouTimelineScorerCandidatePipelineConfig.identifier -> FailOpenPolicy.Always, + forYouAdsCandidatePipelineConfig.identifier -> FailOpenPolicy.Always, + forYouWhoToFollowCandidatePipelineConfig.identifier -> FailOpenPolicy.Always, + flipPromptCandidatePipelineConfig.identifier -> FailOpenPolicy.Always, + editedTweetsCandidatePipelineConfig.identifier -> FailOpenPolicy.Always, + newTweetsPillCandidatePipelineConfig.identifier -> FailOpenPolicy.Always, + ) + + override val resultSelectors: Seq[Selector[ForYouQuery]] = Seq( + UpdateSortCandidates( + ordering = CandidatesUtil.reverseChronTweetsOrdering, + candidatePipeline = forYouConversationServiceCandidatePipelineConfig.identifier + ), + UpdateSortCandidates( + ordering = CandidatesUtil.scoreOrdering, + candidatePipeline = forYouTimelineScorerCandidatePipelineConfig.identifier + ), + UpdateSortModuleItemCandidates( + candidatePipeline = forYouTimelineScorerCandidatePipelineConfig.identifier, + ordering = CandidatesUtil.conversationModuleTweetsOrdering + ), + DebunchCandidates( + pipelineScope = SpecificPipeline(forYouTimelineScorerCandidatePipelineConfig.identifier), + mustDebunch = { + case item: ItemCandidateWithDetails => + !item.features.getOrElse(InNetworkFeature, false) + case module: ModuleCandidateWithDetails => + !module.candidates.last.features.getOrElse(InNetworkFeature, false) + }, + maxBunchSize = MaxConsecutiveOutOfNetworkCandidates + ), + UpdateConversationModuleId( + pipelineScope = SpecificPipeline(forYouTimelineScorerCandidatePipelineConfig.identifier) + ), + DropMaxCandidates( + candidatePipeline = forYouConversationServiceCandidatePipelineConfig.identifier, + maxSelectionsParam = ServerMaxResultsParam + ), + DropMaxCandidates( + candidatePipeline = forYouTimelineScorerCandidatePipelineConfig.identifier, + maxSelectionsParam = ServerMaxResultsParam + ), + DropMaxCandidates( + candidatePipeline = editedTweetsCandidatePipelineConfig.identifier, + maxSelectionsParam = MaxNumberReplaceInstructionsParam + ), + DropModuleTooFewModuleItemResults( + candidatePipeline = forYouWhoToFollowCandidatePipelineConfig.identifier, + minModuleItemsParam = StaticParam(WhoToFollowCandidatePipelineConfig.MinCandidatesSize) + ), + DropMaxModuleItemCandidates( + candidatePipeline = forYouWhoToFollowCandidatePipelineConfig.identifier, + maxModuleItemsParam = StaticParam(WhoToFollowCandidatePipelineConfig.MaxCandidatesSize) + ), + // Add Conversation Service tweets to results only if the scored pipeline doesn't return any + SelectConditionally( + selector = InsertAppendResults( + candidatePipeline = forYouConversationServiceCandidatePipelineConfig.identifier), + includeSelector = (_, candidates, _) => + !candidates.exists(candidate => + forYouTimelineScorerCandidatePipelineConfig.identifier == candidate.source) + ), + InsertAppendResults(candidatePipeline = forYouTimelineScorerCandidatePipelineConfig.identifier), + InsertFixedPositionResults( + candidatePipeline = forYouWhoToFollowCandidatePipelineConfig.identifier, + positionParam = WhoToFollowPositionParam + ), + InsertFixedPositionResults( + candidatePipeline = flipPromptCandidatePipelineConfig.identifier, + positionParam = FlipInlineInjectionModulePosition + ), + InsertAdResults( + surfaceAreaName = AdsInjectionSurfaceAreas.HomeTimeline, + adsInjector = adsInjector.forSurfaceArea(AdsInjectionSurfaceAreas.HomeTimeline), + adsCandidatePipeline = forYouAdsCandidatePipelineConfig.identifier + ), + // This selector must come after the tweets are inserted into the results + UpdateNewTweetsPillDecoration( + pipelineScope = SpecificPipelines( + forYouConversationServiceCandidatePipelineConfig.identifier, + forYouTimelineScorerCandidatePipelineConfig.identifier, + newTweetsPillCandidatePipelineConfig.identifier + ), + stringCenter = stringCenterProvider.get(), + seeNewTweetsString = externalStrings.seeNewTweetsString, + tweetedString = externalStrings.tweetedString + ), + InsertAppendResults(candidatePipeline = editedTweetsCandidatePipelineConfig.identifier), + SelectConditionally( + selector = + InsertAppendResults(candidatePipeline = newTweetsPillCandidatePipelineConfig.identifier), + includeSelector = (_, _, results) => CandidatesUtil.containsType[TweetCandidate](results) + ), + UpdateHomeClientEventDetails( + candidatePipelines = Set( + forYouConversationServiceCandidatePipelineConfig.identifier, + forYouTimelineScorerCandidatePipelineConfig.identifier + ) + ), + ) + + private val servedCandidateKeysKafkaSideEffect = + servedCandidateKeysKafkaSideEffectBuilder.build( + Set(forYouTimelineScorerCandidatePipelineConfig.identifier)) + + private val servedCandidateFeatureKeysKafkaSideEffect = + servedCandidateFeatureKeysKafkaSideEffectBuilder.build( + Set(forYouTimelineScorerCandidatePipelineConfig.identifier)) + + private val homeScribeClientEventSideEffect = HomeScribeClientEventSideEffect( + logPipelinePublisher = clientEventsScribeEventPublisher, + injectedTweetsCandidatePipelineIdentifiers = Seq( + forYouTimelineScorerCandidatePipelineConfig.identifier, + forYouConversationServiceCandidatePipelineConfig.identifier + ), + adsCandidatePipelineIdentifier = forYouAdsCandidatePipelineConfig.identifier, + whoToFollowCandidatePipelineIdentifier = + Some(forYouWhoToFollowCandidatePipelineConfig.identifier), + ) + + override val resultSideEffects: Seq[PipelineResultSideEffect[ForYouQuery, Timeline]] = Seq( + servedCandidateKeysKafkaSideEffect, + servedCandidateFeatureKeysKafkaSideEffect, + updateLastNonPollingTimeSideEffect, + publishClientSentImpressionsEventBusSideEffect, + publishClientSentImpressionsManhattanSideEffect, + updateTimelinesPersistenceStoreSideEffect, + truncateTimelinesPersistenceStoreSideEffect, + homeScribeClientEventSideEffect, + homeScribeServedEntriesSideEffect, + servedStatsSideEffect + ) + + override val domainMarshaller: DomainMarshaller[ForYouQuery, Timeline] = { + val instructionBuilders = Seq( + ClearCacheInstructionBuilder( + ClearCacheIncludeInstruction( + ClearCacheOnPtr.EnableParam, + ClearCacheOnPtr.MinEntriesParam + ) + ), + ReplaceEntryInstructionBuilder(ReplaceAllEntries), + // excludes alert, cover, and replace candidates + AddEntriesWithReplaceAndShowAlertAndCoverInstructionBuilder(), + ShowAlertInstructionBuilder(), + ShowCoverInstructionBuilder(), + ) + + val idSelector: PartialFunction[UniversalNoun[_], Long] = { + // exclude ads while determining tweet cursor values + case item: TweetItem if item.promotedMetadata.isEmpty => item.id + case module: TimelineModule + if module.items.headOption.exists(_.item.isInstanceOf[TweetItem]) => + module.items.last.item match { case item: TweetItem => item.id } + } + val topCursorBuilder = OrderedTopCursorBuilder(idSelector) + val bottomCursorBuilder = OrderedBottomCursorBuilder(idSelector) + + val metadataBuilder = UrtMetadataBuilder( + title = None, + scribeConfigBuilder = Some( + StaticTimelineScribeConfigBuilder( + TimelineScribeConfig( + page = Some("for_you_timeline_scorer"), + section = None, + entityToken = None))) + ) + + UrtDomainMarshaller( + instructionBuilders = instructionBuilders, + metadataBuilder = Some(metadataBuilder), + cursorBuilders = Seq(topCursorBuilder, bottomCursorBuilder) + ) + } + + override val transportMarshaller: TransportMarshaller[Timeline, urt.TimelineResponse] = + urtTransportMarshaller +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/for_you/ForYouTimelineScorerResponseFeatureTransformer.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/for_you/ForYouTimelineScorerResponseFeatureTransformer.scala new file mode 100644 index 0000000000..dfae6a10fd --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/for_you/ForYouTimelineScorerResponseFeatureTransformer.scala @@ -0,0 +1,189 @@ +package com.twitter.home_mixer.product.for_you + +import com.twitter.tweetconvosvc.tweet_ancestor.{thriftscala => ta} +import com.twitter.home_mixer.model.HomeFeatures._ +import com.twitter.mediaservices.commons.tweetmedia.{thriftscala => mt} +import com.twitter.product_mixer.component_library.candidate_source.timeline_scorer.ScoredTweetCandidateWithFocalTweet +import com.twitter.product_mixer.core.feature.Feature +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.feature.featuremap.FeatureMapBuilder +import com.twitter.product_mixer.core.functional_component.transformer.CandidateFeatureTransformer +import com.twitter.product_mixer.core.model.common.identifier.TransformerIdentifier +import com.twitter.product_mixer.core.model.marshalling.response.urt.metadata.BasicTopicContextFunctionalityType +import com.twitter.product_mixer.core.model.marshalling.response.urt.metadata.RecWithEducationTopicContextFunctionalityType +import com.twitter.product_mixer.core.model.marshalling.response.urt.metadata.RecommendationTopicContextFunctionalityType +import com.twitter.search.common.constants.thriftjava.ThriftLanguage +import com.twitter.search.common.util.lang.ThriftLanguageUtil +import com.twitter.snowflake.id.SnowflakeId +import com.twitter.timelinemixer.injection.model.candidate.AudioSpaceMetaData +import com.twitter.timelines.conversation_features.{thriftscala => cvt} +import com.twitter.timelinescorer.common.scoredtweetcandidate.{thriftscala => stc} +import com.twitter.timelineservice.suggests.{thriftscala => tls} + +object ForYouTimelineScorerResponseFeatureTransformer + extends CandidateFeatureTransformer[ScoredTweetCandidateWithFocalTweet] { + + override val identifier: TransformerIdentifier = + TransformerIdentifier("ForYouTimelineScorerResponse") + + override val features: Set[Feature[_, _]] = Set( + AncestorsFeature, + AudioSpaceMetaDataFeature, + AuthorIdFeature, + AuthorIsEligibleForConnectBoostFeature, + AuthoredByContextualUserFeature, + CandidateSourceIdFeature, + ConversationFeature, + ConversationModuleFocalTweetIdFeature, + ConversationModuleIdFeature, + DirectedAtUserIdFeature, + EarlybirdFeature, + EntityTokenFeature, + ExclusiveConversationAuthorIdFeature, + FavoritedByUserIdsFeature, + FollowedByUserIdsFeature, + TopicIdSocialContextFeature, + TopicContextFunctionalityTypeFeature, + FromInNetworkSourceFeature, + FullScoringSucceededFeature, + HasDisplayedTextFeature, + InReplyToTweetIdFeature, + IsAncestorCandidateFeature, + IsExtendedReplyFeature, + IsRandomTweetFeature, + IsReadFromCacheFeature, + IsRetweetFeature, + IsRetweetedReplyFeature, + NonSelfFavoritedByUserIdsFeature, + NumImagesFeature, + OriginalTweetCreationTimeFromSnowflakeFeature, + PredictionRequestIdFeature, + QuotedTweetIdFeature, + ScoreFeature, + SimclustersTweetTopKClustersWithScoresFeature, + SourceTweetIdFeature, + SourceUserIdFeature, + StreamToKafkaFeature, + SuggestTypeFeature, + TweetLanguageFeature, + VideoDurationMsFeature, + ) + + // Convert language code to ISO 639-3 format + private def getLanguageISOFormatByValue(languageCodeValue: Int): String = + ThriftLanguageUtil.getLanguageCodeOf(ThriftLanguage.findByValue(languageCodeValue)) + + override def transform( + candidateWithFocalTweet: ScoredTweetCandidateWithFocalTweet + ): FeatureMap = { + val candidate: stc.v1.ScoredTweetCandidate = candidateWithFocalTweet.candidate + val focalTweetId = candidateWithFocalTweet.focalTweetIdOpt + + val originalTweetId = candidate.sourceTweetId.getOrElse(candidate.tweetId) + val tweetFeatures = candidate.tweetFeaturesMap.flatMap(_.get(originalTweetId)) + val earlybirdFeatures = tweetFeatures.flatMap(_.recapFeatures.flatMap(_.tweetFeatures)) + val directedAtUserIsInFirstDegree = + earlybirdFeatures.flatMap(_.directedAtUserIdIsInFirstDegree) + val isReply = candidate.inReplyToTweetId.nonEmpty + val isRetweet = candidate.isRetweet.getOrElse(false) + val isInNetwork = candidate.isInNetwork.getOrElse(true) + val conversationFeatures = candidate.conversationFeatures.flatMap { + case cvt.ConversationFeatures.V1(candidate) => Some(candidate) + case _ => None + } + val numImages = candidate.mediaMetaData + .map( + _.count(mediaEntity => + mediaEntity.mediaInfo.exists(_.isInstanceOf[mt.MediaInfo.ImageInfo]) || + mediaEntity.mediaInfo.isEmpty)) + val hasImage = earlybirdFeatures.exists(_.hasImage) + val hasVideo = earlybirdFeatures.exists(_.hasVideo) + val hasCard = earlybirdFeatures.exists(_.hasCard) + val hasQuote = earlybirdFeatures.exists(_.hasQuote.contains(true)) + val hasDisplayedText = earlybirdFeatures.exists(_.tweetLength.exists(length => { + val numMedia = Seq(hasVideo, (hasImage || hasCard), hasQuote).count(b => b) + val tcoLengthsPlusSpaces = 23 * numMedia + (if (numMedia > 0) numMedia - 1 else 0) + length > tcoLengthsPlusSpaces + })) + val suggestType = Some( + candidate.overrideSuggestType.getOrElse(tls.SuggestType.RankedTimelineTweet)) + + val topicSocialProofMetadataOpt = candidate.entityData.flatMap(_.topicSocialProofMetadata) + val topicIdSocialContextOpt = topicSocialProofMetadataOpt.map(_.topicId) + val topicContextFunctionalityTypeOpt = + topicSocialProofMetadataOpt.map(_.topicContextFunctionalityType).collect { + case stc.v1.TopicContextFunctionalityType.Basic => BasicTopicContextFunctionalityType + case stc.v1.TopicContextFunctionalityType.Recommendation => + RecommendationTopicContextFunctionalityType + case stc.v1.TopicContextFunctionalityType.RecWithEducation => + RecWithEducationTopicContextFunctionalityType + } + + FeatureMapBuilder() + .add( + AncestorsFeature, + candidate.ancestors + .getOrElse(Seq.empty) + .map(ancestor => ta.TweetAncestor(ancestor.tweetId, ancestor.userId.getOrElse(0L)))) + .add( + AudioSpaceMetaDataFeature, + candidate.audioSpaceMetaDatalist.map(_.head).map(AudioSpaceMetaData.fromThrift)) + .add(AuthorIdFeature, Some(candidate.authorId)) + .add( + AuthorIsEligibleForConnectBoostFeature, + candidate.authorIsEligibleForConnectBoost.getOrElse(false)) + .add( + AuthoredByContextualUserFeature, + candidate.viewerId.contains(candidate.authorId) || + candidate.viewerId.exists(candidate.sourceUserId.contains)) + .add(CandidateSourceIdFeature, candidate.candidateTweetSourceId) + .add(ConversationFeature, conversationFeatures) + .add(ConversationModuleIdFeature, candidate.conversationId) + .add(ConversationModuleFocalTweetIdFeature, focalTweetId) + .add(DirectedAtUserIdFeature, candidate.directedAtUserId) + .add(EarlybirdFeature, earlybirdFeatures) + // This is temporary, will need to be updated with the encoded string. + .add(EntityTokenFeature, Some("test_EntityTokenForYou")) + .add(ExclusiveConversationAuthorIdFeature, candidate.exclusiveConversationAuthorId) + .add(FavoritedByUserIdsFeature, candidate.favoritedByUserIds.getOrElse(Seq.empty)) + .add(FollowedByUserIdsFeature, candidate.followedByUserIds.getOrElse(Seq.empty)) + .add(TopicIdSocialContextFeature, topicIdSocialContextOpt) + .add(TopicContextFunctionalityTypeFeature, topicContextFunctionalityTypeOpt) + .add(FullScoringSucceededFeature, candidate.fullScoringSucceeded.getOrElse(false)) + .add(HasDisplayedTextFeature, hasDisplayedText) + .add(InReplyToTweetIdFeature, candidate.inReplyToTweetId) + .add(IsAncestorCandidateFeature, candidate.isAncestorCandidate.getOrElse(false)) + .add( + IsExtendedReplyFeature, + isInNetwork && isReply && !isRetweet && directedAtUserIsInFirstDegree.contains(false)) + .add(FromInNetworkSourceFeature, candidate.isInNetwork.getOrElse(true)) + .add(IsRandomTweetFeature, candidate.isRandomTweet.getOrElse(false)) + .add(IsReadFromCacheFeature, candidate.isReadFromCache.getOrElse(false)) + .add(IsRetweetFeature, candidate.isRetweet.getOrElse(false)) + .add(IsRetweetedReplyFeature, isReply && isRetweet) + .add( + NonSelfFavoritedByUserIdsFeature, + candidate.favoritedByUserIds.getOrElse(Seq.empty).filterNot(_ == candidate.authorId)) + .add(NumImagesFeature, numImages) + .add( + OriginalTweetCreationTimeFromSnowflakeFeature, + SnowflakeId.timeFromIdOpt(originalTweetId)) + .add(PredictionRequestIdFeature, candidate.predictionRequestId) + .add(ScoreFeature, Some(candidate.score)) + .add( + SimclustersTweetTopKClustersWithScoresFeature, + candidate.simclustersTweetTopKClustersWithScores.map(_.toMap).getOrElse(Map.empty)) + .add( + StreamToKafkaFeature, + candidate.predictionRequestId.nonEmpty && candidate.fullScoringSucceeded.getOrElse(false)) + .add(SourceTweetIdFeature, candidate.sourceTweetId) + .add(SourceUserIdFeature, candidate.sourceUserId) + .add(SuggestTypeFeature, suggestType) + .add(QuotedTweetIdFeature, candidate.quotedTweetId) + .add( + TweetLanguageFeature, + earlybirdFeatures.flatMap(_.language.map(_.value)).map(getLanguageISOFormatByValue)) + .add(VideoDurationMsFeature, earlybirdFeatures.flatMap(_.videoDurationMs)) + .build() + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/for_you/ForYouWhoToFollowCandidatePipelineConfigBuilder.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/for_you/ForYouWhoToFollowCandidatePipelineConfigBuilder.scala new file mode 100644 index 0000000000..0f01c9e148 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/for_you/ForYouWhoToFollowCandidatePipelineConfigBuilder.scala @@ -0,0 +1,59 @@ +package com.twitter.home_mixer.product.for_you + +import com.twitter.home_mixer.functional_component.decorator.urt.builder.HomeWhoToFollowFeedbackActionInfoBuilder +import com.twitter.home_mixer.functional_component.gate.DismissFatigueGate +import com.twitter.home_mixer.functional_component.gate.TimelinesPersistenceStoreLastInjectionGate +import com.twitter.home_mixer.model.HomeFeatures.DismissInfoFeature +import com.twitter.home_mixer.model.HomeFeatures.PersistenceEntriesFeature +import com.twitter.home_mixer.model.HomeFeatures.WhoToFollowExcludedUserIdsFeature +import com.twitter.home_mixer.product.for_you.model.ForYouQuery +import com.twitter.home_mixer.product.for_you.param.ForYouParam.EnableWhoToFollowCandidatePipelineParam +import com.twitter.home_mixer.product.for_you.param.ForYouParam.WhoToFollowDisplayTypeIdParam +import com.twitter.home_mixer.product.for_you.param.ForYouParam.WhoToFollowMinInjectionIntervalParam +import com.twitter.home_mixer.service.HomeMixerAlertConfig +import com.twitter.product_mixer.component_library.decorator.urt.builder.timeline_module.ParamWhoToFollowModuleDisplayTypeBuilder +import com.twitter.product_mixer.component_library.pipeline.candidate.who_to_follow_module.WhoToFollowCandidatePipelineConfig +import com.twitter.product_mixer.component_library.pipeline.candidate.who_to_follow_module.WhoToFollowCandidatePipelineConfigBuilder +import com.twitter.product_mixer.core.functional_component.configapi.StaticParam +import com.twitter.product_mixer.core.functional_component.gate.Gate +import com.twitter.timelineservice.model.rich.EntityIdType +import com.twitter.timelineservice.suggests.thriftscala.SuggestType +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class ForYouWhoToFollowCandidatePipelineConfigBuilder @Inject() ( + whoToFollowCandidatePipelineConfigBuilder: WhoToFollowCandidatePipelineConfigBuilder, + homeWhoToFollowFeedbackActionInfoBuilder: HomeWhoToFollowFeedbackActionInfoBuilder) { + + // People Discovery module timeout is set to 350ms currently so use faster display location here + private val DisplayLocation = "timeline_reverse_chron" + + def build(): WhoToFollowCandidatePipelineConfig[ForYouQuery] = { + val gates: Seq[Gate[ForYouQuery]] = Seq( + TimelinesPersistenceStoreLastInjectionGate( + WhoToFollowMinInjectionIntervalParam, + PersistenceEntriesFeature, + EntityIdType.WhoToFollow + ), + DismissFatigueGate(SuggestType.WhoToFollow, DismissInfoFeature) + ) + + whoToFollowCandidatePipelineConfigBuilder.build[ForYouQuery]( + identifier = WhoToFollowCandidatePipelineConfig.identifier, + supportedClientParam = Some(EnableWhoToFollowCandidatePipelineParam), + alerts = alerts, + gates = gates, + moduleDisplayTypeBuilder = + ParamWhoToFollowModuleDisplayTypeBuilder(WhoToFollowDisplayTypeIdParam), + feedbackActionInfoBuilder = Some(homeWhoToFollowFeedbackActionInfoBuilder), + excludedUserIdsFeature = Some(WhoToFollowExcludedUserIdsFeature), + displayLocationParam = StaticParam(DisplayLocation) + ) + } + + private val alerts = Seq( + HomeMixerAlertConfig.BusinessHours.defaultSuccessRateAlert(70), + HomeMixerAlertConfig.BusinessHours.defaultEmptyResponseRateAlert() + ) +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/for_you/candidate_source/BUILD.bazel b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/for_you/candidate_source/BUILD.bazel new file mode 100644 index 0000000000..ed317eb09d --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/for_you/candidate_source/BUILD.bazel @@ -0,0 +1,25 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + strict_deps = True, + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/javax/inject:javax.inject", + "finatra/inject/inject-core/src/main/scala", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/model", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/model/request", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/param", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/product/for_you/model", + "home-mixer/thrift/src/main/thrift:thrift-scala", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/cursor", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/candidate_source", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/candidate_source/product_pipeline", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline", + "src/thrift/com/twitter/search:earlybird-scala", + "src/thrift/com/twitter/timelinemixer:thrift-scala", + "stitch/stitch-timelineservice/src/main/scala", + ], + exports = [ + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/candidate_source", + ], +) diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/for_you/candidate_source/ScoredTweetsProductCandidateSource.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/for_you/candidate_source/ScoredTweetsProductCandidateSource.scala new file mode 100644 index 0000000000..db15a2cac3 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/for_you/candidate_source/ScoredTweetsProductCandidateSource.scala @@ -0,0 +1,154 @@ +package com.twitter.home_mixer.product.for_you.candidate_source + +import com.google.inject.Provider +import com.twitter.home_mixer.model.HomeFeatures.ServedTweetIdsFeature +import com.twitter.home_mixer.model.request.HomeMixerRequest +import com.twitter.home_mixer.model.request.ScoredTweetsProduct +import com.twitter.home_mixer.model.request.ScoredTweetsProductContext +import com.twitter.home_mixer.product.for_you.model.ForYouQuery +import com.twitter.home_mixer.{thriftscala => t} +import com.twitter.product_mixer.core.functional_component.candidate_source.product_pipeline.ProductPipelineCandidateSource +import com.twitter.product_mixer.core.functional_component.configapi.ParamsBuilder +import com.twitter.product_mixer.core.model.common.identifier.CandidateSourceIdentifier +import com.twitter.product_mixer.core.product.registry.ProductPipelineRegistry +import com.twitter.timelines.render.{thriftscala => tl} +import com.twitter.timelineservice.suggests.{thriftscala => st} +import com.twitter.tweetconvosvc.tweet_ancestor.{thriftscala => ta} +import javax.inject.Inject +import javax.inject.Singleton + +/** + * [[ScoredTweetWithConversationMetadata]] + **/ +case class ScoredTweetWithConversationMetadata( + tweetId: Long, + authorId: Long, + score: Option[Double] = None, + suggestType: Option[st.SuggestType] = None, + sourceTweetId: Option[Long] = None, + sourceUserId: Option[Long] = None, + quotedTweetId: Option[Long] = None, + quotedUserId: Option[Long] = None, + inReplyToTweetId: Option[Long] = None, + inReplyToUserId: Option[Long] = None, + directedAtUserId: Option[Long] = None, + inNetwork: Option[Boolean] = None, + favoritedByUserIds: Option[Seq[Long]] = None, + followedByUserIds: Option[Seq[Long]] = None, + ancestors: Option[Seq[ta.TweetAncestor]] = None, + topicId: Option[Long] = None, + topicFunctionalityType: Option[tl.TopicContextFunctionalityType] = None, + conversationId: Option[Long] = None, + conversationFocalTweetId: Option[Long] = None, + isReadFromCache: Option[Boolean] = None, + streamToKafka: Option[Boolean] = None) + +@Singleton +class ScoredTweetsProductCandidateSource @Inject() ( + override val productPipelineRegistry: Provider[ProductPipelineRegistry], + override val paramsBuilder: Provider[ParamsBuilder]) + extends ProductPipelineCandidateSource[ + ForYouQuery, + HomeMixerRequest, + t.ScoredTweetsResponse, + ScoredTweetWithConversationMetadata + ] { + + override val identifier: CandidateSourceIdentifier = + CandidateSourceIdentifier("ScoredTweetsProduct") + + private val MaxModuleSize = 3 + private val MaxAncestorsInConversation = 2 + + override def pipelineRequestTransformer(productPipelineQuery: ForYouQuery): HomeMixerRequest = { + HomeMixerRequest( + clientContext = productPipelineQuery.clientContext, + product = ScoredTweetsProduct, + productContext = Some( + ScoredTweetsProductContext( + productPipelineQuery.deviceContext, + productPipelineQuery.seenTweetIds, + productPipelineQuery.features.map(_.getOrElse(ServedTweetIdsFeature, Seq.empty)) + )), + serializedRequestCursor = None, + maxResults = productPipelineQuery.requestedMaxResults, + debugParams = None, + homeRequestParam = false + ) + } + + override def productPipelineResultTransformer( + productPipelineResult: t.ScoredTweetsResponse + ): Seq[ScoredTweetWithConversationMetadata] = { + val scoredTweets = productPipelineResult.scoredTweets.flatMap { focalTweet => + val parentTweets = focalTweet.ancestors.getOrElse(Seq.empty).sortBy(-_.tweetId) + val (intermediates, root) = parentTweets.splitAt(parentTweets.size - 1) + val truncatedIntermediates = + intermediates.take(MaxModuleSize - MaxAncestorsInConversation).reverse + val rootScoredTweet: Seq[ScoredTweetWithConversationMetadata] = root.map { ancestor => + ScoredTweetWithConversationMetadata( + tweetId = ancestor.tweetId, + authorId = ancestor.userId, + suggestType = focalTweet.suggestType, + conversationId = Some(ancestor.tweetId), + conversationFocalTweetId = Some(focalTweet.tweetId) + ) + } + val conversationId = rootScoredTweet.headOption.map(_.tweetId) + + val tweetsToParents = + if (parentTweets.nonEmpty) parentTweets.zip(parentTweets.tail).toMap + else Map.empty[ta.TweetAncestor, ta.TweetAncestor] + + val intermediateScoredTweets = truncatedIntermediates.map { ancestor => + ScoredTweetWithConversationMetadata( + tweetId = ancestor.tweetId, + authorId = ancestor.userId, + suggestType = focalTweet.suggestType, + inReplyToTweetId = tweetsToParents.get(ancestor).map(_.tweetId), + conversationId = conversationId, + conversationFocalTweetId = Some(focalTweet.tweetId) + ) + } + val parentScoredTweets = rootScoredTweet ++ intermediateScoredTweets + + val conversationFocalTweetId = + if (parentScoredTweets.nonEmpty) Some(focalTweet.tweetId) else None + + val focalScoredTweet = ScoredTweetWithConversationMetadata( + tweetId = focalTweet.tweetId, + authorId = focalTweet.authorId, + score = focalTweet.score, + suggestType = focalTweet.suggestType, + sourceTweetId = focalTweet.sourceTweetId, + sourceUserId = focalTweet.sourceUserId, + quotedTweetId = focalTweet.quotedTweetId, + quotedUserId = focalTweet.quotedUserId, + inReplyToTweetId = parentScoredTweets.lastOption.map(_.tweetId), + inReplyToUserId = focalTweet.inReplyToUserId, + directedAtUserId = focalTweet.directedAtUserId, + inNetwork = focalTweet.inNetwork, + favoritedByUserIds = focalTweet.favoritedByUserIds, + followedByUserIds = focalTweet.followedByUserIds, + topicId = focalTweet.topicId, + topicFunctionalityType = focalTweet.topicFunctionalityType, + ancestors = focalTweet.ancestors, + conversationId = conversationId, + conversationFocalTweetId = conversationFocalTweetId, + isReadFromCache = focalTweet.isReadFromCache, + streamToKafka = focalTweet.streamToKafka + ) + + parentScoredTweets :+ focalScoredTweet + } + + val dedupedTweets = scoredTweets.groupBy(_.tweetId).map { + case (_, duplicateAncestors) => duplicateAncestors.maxBy(_.score.getOrElse(0.0)) + } + + // Sort by tweet id to prevent issues with future assumptions of the root being the first + // tweet and the focal being the last tweet in a module. The tweets as a whole do not need + // to be sorted overall, only the relative order within modules must be kept. + dedupedTweets.toSeq.sortBy(_.tweetId) + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/for_you/model/BUILD.bazel b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/for_you/model/BUILD.bazel new file mode 100644 index 0000000000..bcf9519f5f --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/for_you/model/BUILD.bazel @@ -0,0 +1,21 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + strict_deps = True, + tags = ["bazel-compatible"], + dependencies = [ + "home-mixer/server/src/main/scala/com/twitter/home_mixer/model", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/model/request", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/cursor", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/query/ads", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/pipeline/candidate/flexible_injection_pipeline", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/quality_factor", + ], + exports = [ + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/cursor", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/query/ads", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/pipeline/candidate/flexible_injection_pipeline", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline", + ], +) diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/for_you/model/ForYouQuery.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/for_you/model/ForYouQuery.scala new file mode 100644 index 0000000000..ec701ac602 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/for_you/model/ForYouQuery.scala @@ -0,0 +1,50 @@ +package com.twitter.home_mixer.product.for_you.model + +import com.twitter.adserver.thriftscala.HomeTimelineType +import com.twitter.adserver.thriftscala.TimelineRequestParams +import com.twitter.dspbidder.commons.{thriftscala => dsp} +import com.twitter.home_mixer.model.HomeAdsQuery +import com.twitter.home_mixer.model.request.DeviceContext +import com.twitter.home_mixer.model.request.ForYouProduct +import com.twitter.home_mixer.model.request.HasDeviceContext +import com.twitter.home_mixer.model.request.HasSeenTweetIds +import com.twitter.onboarding.task.service.{thriftscala => ots} +import com.twitter.product_mixer.component_library.model.cursor.UrtOrderedCursor +import com.twitter.product_mixer.component_library.pipeline.candidate.flexible_injection_pipeline.transformer.HasFlipInjectionParams +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.model.marshalling.request._ +import com.twitter.product_mixer.core.pipeline.HasPipelineCursor +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.timelines.configapi.Params + +case class ForYouQuery( + override val params: Params, + override val clientContext: ClientContext, + override val pipelineCursor: Option[UrtOrderedCursor], + override val requestedMaxResults: Option[Int], + override val debugOptions: Option[DebugOptions], + override val features: Option[FeatureMap], + override val deviceContext: Option[DeviceContext], + override val seenTweetIds: Option[Seq[Long]], + override val dspClientContext: Option[dsp.DspClientContext]) + extends PipelineQuery + with HasPipelineCursor[UrtOrderedCursor] + with HasDeviceContext + with HasSeenTweetIds + with HasFlipInjectionParams + with HomeAdsQuery { + override val product: Product = ForYouProduct + + override def withFeatureMap(features: FeatureMap): ForYouQuery = + copy(features = Some(features)) + + override val timelineRequestParams: Option[TimelineRequestParams] = + Some(TimelineRequestParams(homeTimelineType = Some(HomeTimelineType.Home))) + + // Fields below are used for FLIP Injection in Onboarding Task Service (OTS) + override val displayLocation: ots.DisplayLocation = ots.DisplayLocation.HomeTimeline + override val rankingDisablerWithLatestControlsAvailable: Option[Boolean] = None + override val isEmptyState: Option[Boolean] = None + override val isFirstRequestAfterSignup: Option[Boolean] = None + override val isEndOfTimeline: Option[Boolean] = None +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/for_you/model/ForYouTweetsResponse.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/for_you/model/ForYouTweetsResponse.scala new file mode 100644 index 0000000000..68b0d67361 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/for_you/model/ForYouTweetsResponse.scala @@ -0,0 +1,5 @@ +package com.twitter.home_mixer.product.for_you.model + +import com.twitter.product_mixer.core.model.marshalling.HasMarshalling + +case class ForYouTweetsResponse(tweetCandidates: Seq[Long]) extends HasMarshalling diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/for_you/param/BUILD.bazel b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/for_you/param/BUILD.bazel new file mode 100644 index 0000000000..a56e3a1fdc --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/for_you/param/BUILD.bazel @@ -0,0 +1,14 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + strict_deps = True, + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/javax/inject:javax.inject", + "configapi/configapi-core/src/main/scala/com/twitter/timelines/configapi", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/param/decider", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/product", + "util/util-core/src/main/scala/com/twitter/conversions", + ], +) diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/for_you/param/ForYouParam.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/for_you/param/ForYouParam.scala new file mode 100644 index 0000000000..0ed56a3f88 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/for_you/param/ForYouParam.scala @@ -0,0 +1,117 @@ +package com.twitter.home_mixer.product.for_you.param + +import com.twitter.conversions.DurationOps._ +import com.twitter.home_mixer.param.decider.DeciderKey +import com.twitter.product_mixer.component_library.decorator.urt.builder.timeline_module.WhoToFollowModuleDisplayType +import com.twitter.timelines.configapi.DurationConversion +import com.twitter.timelines.configapi.FSBoundedParam +import com.twitter.timelines.configapi.FSEnumParam +import com.twitter.timelines.configapi.FSParam +import com.twitter.timelines.configapi.HasDurationConversion +import com.twitter.timelines.configapi.decider.BooleanDeciderParam +import com.twitter.util.Duration + +object ForYouParam { + val SupportedClientFSName = "for_you_supported_client" + + object EnableTimelineScorerCandidatePipelineParam + extends FSParam[Boolean]( + name = "for_you_enable_timeline_scorer_candidate_pipeline", + default = true + ) + + object EnableScoredTweetsCandidatePipelineParam + extends BooleanDeciderParam(DeciderKey.EnableForYouScoredTweetsCandidatePipeline) + + object EnableWhoToFollowCandidatePipelineParam + extends FSParam[Boolean]( + name = "for_you_enable_who_to_follow", + default = true + ) + + object EnableScoredTweetsMixerPipelineParam + extends FSParam[Boolean]( + name = "for_you_enable_scored_tweets_mixer_pipeline", + default = true + ) + + object ServerMaxResultsParam + extends FSBoundedParam[Int]( + name = "for_you_server_max_results", + default = 35, + min = 1, + max = 500 + ) + + object TimelineServiceMaxResultsParam + extends FSBoundedParam[Int]( + name = "for_you_timeline_service_max_results", + default = 800, + min = 1, + max = 800 + ) + + object AdsNumOrganicItemsParam + extends FSBoundedParam[Int]( + name = "for_you_ads_num_organic_items", + default = 35, + min = 1, + max = 100 + ) + + object WhoToFollowPositionParam + extends FSBoundedParam[Int]( + name = "for_you_who_to_follow_position", + default = 5, + min = 0, + max = 99 + ) + + object WhoToFollowMinInjectionIntervalParam + extends FSBoundedParam[Duration]( + "for_you_who_to_follow_min_injection_interval_in_minutes", + default = 1800.minutes, + min = 0.minutes, + max = 6000.minutes) + with HasDurationConversion { + override val durationConversion: DurationConversion = DurationConversion.FromMinutes + } + + object WhoToFollowDisplayTypeIdParam + extends FSEnumParam[WhoToFollowModuleDisplayType.type]( + name = "for_you_enable_who_to_follow_display_type_id", + default = WhoToFollowModuleDisplayType.Vertical, + enum = WhoToFollowModuleDisplayType + ) + + object EnableFlipInjectionModuleCandidatePipelineParam + extends FSParam[Boolean]( + name = "for_you_enable_flip_inline_injection_module", + default = true + ) + + object FlipInlineInjectionModulePosition + extends FSBoundedParam[Int]( + name = "for_you_flip_inline_injection_module_position", + default = 0, + min = 0, + max = 1000 + ) + + object ClearCacheOnPtr { + + object EnableParam + extends FSParam[Boolean]( + name = "for_you_clear_cache_ptr_enable", + default = false + ) + + case object MinEntriesParam + extends FSBoundedParam[Int]( + name = "for_you_clear_cache_ptr_min_entries", + default = 10, + min = 0, + max = 35 + ) + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/for_you/param/ForYouParamConfig.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/for_you/param/ForYouParamConfig.scala new file mode 100644 index 0000000000..ea8e389cc1 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/for_you/param/ForYouParamConfig.scala @@ -0,0 +1,43 @@ +package com.twitter.home_mixer.product.for_you.param + +import com.twitter.home_mixer.param.decider.DeciderKey +import com.twitter.home_mixer.product.for_you.param.ForYouParam._ +import com.twitter.product_mixer.core.product.ProductParamConfig +import com.twitter.servo.decider.DeciderKeyName +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class ForYouParamConfig @Inject() () extends ProductParamConfig { + override val enabledDeciderKey: DeciderKeyName = DeciderKey.EnableForYouProduct + override val supportedClientFSName: String = SupportedClientFSName + + override val booleanDeciderOverrides = Seq( + EnableScoredTweetsCandidatePipelineParam + ) + + override val booleanFSOverrides = Seq( + EnableFlipInjectionModuleCandidatePipelineParam, + EnableWhoToFollowCandidatePipelineParam, + EnableScoredTweetsMixerPipelineParam, + ClearCacheOnPtr.EnableParam, + EnableTimelineScorerCandidatePipelineParam + ) + + override val boundedIntFSOverrides = Seq( + ServerMaxResultsParam, + WhoToFollowPositionParam, + FlipInlineInjectionModulePosition, + TimelineServiceMaxResultsParam, + AdsNumOrganicItemsParam, + ClearCacheOnPtr.MinEntriesParam + ) + + override val boundedDurationFSOverrides = Seq( + WhoToFollowMinInjectionIntervalParam + ) + + override val enumFSOverrides = Seq( + WhoToFollowDisplayTypeIdParam + ) +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_recommended_users/BUILD.bazel b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_recommended_users/BUILD.bazel new file mode 100644 index 0000000000..4c69111d18 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_recommended_users/BUILD.bazel @@ -0,0 +1,49 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + strict_deps = True, + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/javax/inject:javax.inject", + "finatra/inject/inject-core/src/main/scala", + "finatra/inject/inject-core/src/main/scala/com/twitter/inject", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/candidate_source", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/builder", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/urt/builder", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/filter", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/gate", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/selector", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/side_effect", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/marshaller/timelines", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/model", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/model/request", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/param", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_recommended_users/feature_hydrator", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_recommended_users/filter", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_recommended_users/model", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_recommended_users/param", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/service", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/premarshaller/urt", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/selector", + "product-mixer/core/src/main/java/com/twitter/product_mixer/core/product/guice/scope", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/configapi", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/decorator", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/decorator/urt/builder", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/common/presentation", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/request", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/item", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/candidate", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/mixer", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/product", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/product", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/product/guice", + "src/thrift/com/twitter/hermit/candidate:hermit-candidate-scala", + "src/thrift/com/twitter/timelines/render:thrift-scala", + ], + exports = [ + "src/thrift/com/twitter/timelines/render:thrift-scala", + ], +) diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_recommended_users/ListMemberBasedUsersCandidatePipelineConfig.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_recommended_users/ListMemberBasedUsersCandidatePipelineConfig.scala new file mode 100644 index 0000000000..267c0b4f31 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_recommended_users/ListMemberBasedUsersCandidatePipelineConfig.scala @@ -0,0 +1,89 @@ +package com.twitter.home_mixer.product.list_recommended_users + +import com.twitter.hermit.candidate.{thriftscala => t} +import com.twitter.home_mixer.functional_component.candidate_source.SimilarityBasedUsersCandidateSource +import com.twitter.home_mixer.functional_component.feature_hydrator.ListMembersFeature +import com.twitter.home_mixer.functional_component.filter.PredicateFeatureFilter +import com.twitter.home_mixer.product.list_recommended_users.feature_hydrator.GizmoduckUserFeatureHydrator +import com.twitter.home_mixer.product.list_recommended_users.feature_hydrator.IsListMemberFeatureHydrator +import com.twitter.home_mixer.product.list_recommended_users.filter.DropMaxCandidatesByScoreFilter +import com.twitter.home_mixer.product.list_recommended_users.filter.PreviouslyServedUsersFilter +import com.twitter.home_mixer.product.list_recommended_users.model.ListFeatures.IsListMemberFeature +import com.twitter.home_mixer.product.list_recommended_users.model.ListRecommendedUsersQuery +import com.twitter.product_mixer.component_library.decorator.urt.UrtItemCandidateDecorator +import com.twitter.product_mixer.component_library.decorator.urt.builder.item.user.UserCandidateUrtItemBuilder +import com.twitter.product_mixer.component_library.decorator.urt.builder.metadata.ClientEventInfoBuilder +import com.twitter.product_mixer.component_library.model.candidate.UserCandidate +import com.twitter.product_mixer.core.functional_component.candidate_source.BaseCandidateSource +import com.twitter.product_mixer.core.functional_component.decorator.CandidateDecorator +import com.twitter.product_mixer.core.functional_component.feature_hydrator.BaseCandidateFeatureHydrator +import com.twitter.product_mixer.core.functional_component.filter.Filter +import com.twitter.product_mixer.core.functional_component.transformer.CandidateFeatureTransformer +import com.twitter.product_mixer.core.functional_component.transformer.CandidatePipelineQueryTransformer +import com.twitter.product_mixer.core.functional_component.transformer.CandidatePipelineResultsTransformer +import com.twitter.product_mixer.core.model.common.identifier.CandidatePipelineIdentifier +import com.twitter.product_mixer.core.model.common.identifier.FilterIdentifier +import com.twitter.product_mixer.core.pipeline.candidate.CandidatePipelineConfig + +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class ListMemberBasedUsersCandidatePipelineConfig @Inject() ( + similarityBasedUsersCandidateSource: SimilarityBasedUsersCandidateSource, + gizmoduckUserFeatureHydrator: GizmoduckUserFeatureHydrator, + isListMemberFeatureHydrator: IsListMemberFeatureHydrator) + extends CandidatePipelineConfig[ + ListRecommendedUsersQuery, + Seq[Long], + t.Candidate, + UserCandidate + ] { + + override val identifier: CandidatePipelineIdentifier = + CandidatePipelineIdentifier("ListMemberBasedUsers") + + override val queryTransformer: CandidatePipelineQueryTransformer[ListRecommendedUsersQuery, Seq[ + Long + ]] = { query => + query.features.map(_.getOrElse(ListMembersFeature, Seq.empty)).getOrElse(Seq.empty) + } + + override val candidateSource: BaseCandidateSource[Seq[Long], t.Candidate] = + similarityBasedUsersCandidateSource + + override val featuresFromCandidateSourceTransformers: Seq[ + CandidateFeatureTransformer[t.Candidate] + ] = Seq(ListMemberBasedUsersResponseFeatureTransfromer) + + override val resultTransformer: CandidatePipelineResultsTransformer[ + t.Candidate, + UserCandidate + ] = { candidate => + UserCandidate(id = candidate.userId) + } + + override val preFilterFeatureHydrationPhase1: Seq[ + BaseCandidateFeatureHydrator[ListRecommendedUsersQuery, UserCandidate, _] + ] = Seq(isListMemberFeatureHydrator) + + override val filters: Seq[Filter[ListRecommendedUsersQuery, UserCandidate]] = + Seq( + PreviouslyServedUsersFilter, + PredicateFeatureFilter.fromPredicate( + FilterIdentifier("IsListMember"), + shouldKeepCandidate = { features => !features.getOrElse(IsListMemberFeature, false) } + ), + DropMaxCandidatesByScoreFilter + ) + + override val postFilterFeatureHydration: Seq[ + BaseCandidateFeatureHydrator[ListRecommendedUsersQuery, UserCandidate, _] + ] = Seq(gizmoduckUserFeatureHydrator) + + override val decorator: Option[CandidateDecorator[ListRecommendedUsersQuery, UserCandidate]] = { + val clientEventInfoBuilder = ClientEventInfoBuilder("user") + val userItemBuilder = UserCandidateUrtItemBuilder(clientEventInfoBuilder) + Some(UrtItemCandidateDecorator(userItemBuilder)) + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_recommended_users/ListMemberBasedUsersResponseFeatureTransfromer.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_recommended_users/ListMemberBasedUsersResponseFeatureTransfromer.scala new file mode 100644 index 0000000000..bce2ed457b --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_recommended_users/ListMemberBasedUsersResponseFeatureTransfromer.scala @@ -0,0 +1,21 @@ +package com.twitter.home_mixer.product.list_recommended_users + +import com.twitter.hermit.candidate.{thriftscala => t} +import com.twitter.home_mixer.product.list_recommended_users.model.ListFeatures.ScoreFeature +import com.twitter.product_mixer.core.feature.Feature +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.feature.featuremap.FeatureMapBuilder +import com.twitter.product_mixer.core.functional_component.transformer.CandidateFeatureTransformer +import com.twitter.product_mixer.core.model.common.identifier.TransformerIdentifier + +object ListMemberBasedUsersResponseFeatureTransfromer + extends CandidateFeatureTransformer[t.Candidate] { + + override val identifier: TransformerIdentifier = TransformerIdentifier("ListMemberBasedUsers") + + override val features: Set[Feature[_, _]] = Set(ScoreFeature) + + override def transform(candidate: t.Candidate): FeatureMap = FeatureMapBuilder() + .add(ScoreFeature, candidate.score) + .build() +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_recommended_users/ListRecommendedUsersMixerPipelineConfig.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_recommended_users/ListRecommendedUsersMixerPipelineConfig.scala new file mode 100644 index 0000000000..d370c10578 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_recommended_users/ListRecommendedUsersMixerPipelineConfig.scala @@ -0,0 +1,100 @@ +package com.twitter.home_mixer.product.list_recommended_users + +import com.twitter.home_mixer.functional_component.feature_hydrator.ListMembersQueryFeatureHydrator +import com.twitter.home_mixer.functional_component.gate.ViewerIsListOwnerGate +import com.twitter.home_mixer.product.list_recommended_users.model.ListFeatures.GizmoduckUserFeature +import com.twitter.home_mixer.product.list_recommended_users.model.ListRecommendedUsersQuery +import com.twitter.home_mixer.product.list_recommended_users.param.ListRecommendedUsersParam.ExcludedIdsMaxLengthParam +import com.twitter.home_mixer.product.list_recommended_users.param.ListRecommendedUsersParam.ServerMaxResultsParam +import com.twitter.product_mixer.component_library.premarshaller.urt.UrtDomainMarshaller +import com.twitter.product_mixer.component_library.premarshaller.urt.builder.AddEntriesWithReplaceInstructionBuilder +import com.twitter.product_mixer.component_library.premarshaller.urt.builder.ReplaceAllEntries +import com.twitter.product_mixer.component_library.premarshaller.urt.builder.ReplaceEntryInstructionBuilder +import com.twitter.product_mixer.component_library.premarshaller.urt.builder.StaticTimelineScribeConfigBuilder +import com.twitter.product_mixer.component_library.premarshaller.urt.builder.UnorderedExcludeIdsBottomCursorBuilder +import com.twitter.product_mixer.component_library.premarshaller.urt.builder.UrtMetadataBuilder +import com.twitter.product_mixer.component_library.selector.DropFilteredCandidates +import com.twitter.product_mixer.component_library.selector.DropMaxCandidates +import com.twitter.product_mixer.component_library.selector.InsertAppendResults +import com.twitter.product_mixer.core.functional_component.feature_hydrator.QueryFeatureHydrator +import com.twitter.product_mixer.core.functional_component.marshaller.TransportMarshaller +import com.twitter.product_mixer.core.functional_component.marshaller.response.urt.UrtTransportMarshaller +import com.twitter.product_mixer.core.functional_component.premarshaller.DomainMarshaller +import com.twitter.product_mixer.core.functional_component.selector.Selector +import com.twitter.product_mixer.core.model.common.UniversalNoun +import com.twitter.product_mixer.core.model.common.identifier.MixerPipelineIdentifier +import com.twitter.product_mixer.core.model.marshalling.response.urt.Timeline +import com.twitter.product_mixer.core.model.marshalling.response.urt.TimelineScribeConfig +import com.twitter.product_mixer.core.model.marshalling.response.urt.item.user.UserItem +import com.twitter.product_mixer.core.pipeline.candidate.CandidatePipelineConfig +import com.twitter.product_mixer.core.pipeline.mixer.MixerPipelineConfig +import com.twitter.timelines.render.{thriftscala => urt} + +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class ListRecommendedUsersMixerPipelineConfig @Inject() ( + listMemberBasedUsersCandidatePipelineConfig: ListMemberBasedUsersCandidatePipelineConfig, + viewerIsListOwnerGate: ViewerIsListOwnerGate, + listMembersQueryFeatureHydrator: ListMembersQueryFeatureHydrator, + urtTransportMarshaller: UrtTransportMarshaller) + extends MixerPipelineConfig[ListRecommendedUsersQuery, Timeline, urt.TimelineResponse] { + + override val identifier: MixerPipelineIdentifier = MixerPipelineIdentifier("ListRecommendedUsers") + + override val gates = Seq(viewerIsListOwnerGate) + + override val fetchQueryFeatures: Seq[QueryFeatureHydrator[ListRecommendedUsersQuery]] = + Seq(listMembersQueryFeatureHydrator) + + override val candidatePipelines: Seq[ + CandidatePipelineConfig[ListRecommendedUsersQuery, _, _, _] + ] = + Seq(listMemberBasedUsersCandidatePipelineConfig) + + override val resultSelectors: Seq[Selector[ListRecommendedUsersQuery]] = Seq( + DropFilteredCandidates( + candidatePipeline = listMemberBasedUsersCandidatePipelineConfig.identifier, + filter = candidate => candidate.features.getOrElse(GizmoduckUserFeature, None).isDefined + ), + DropMaxCandidates( + candidatePipeline = listMemberBasedUsersCandidatePipelineConfig.identifier, + maxSelectionsParam = ServerMaxResultsParam), + InsertAppendResults(listMemberBasedUsersCandidatePipelineConfig.identifier) + ) + + override val domainMarshaller: DomainMarshaller[ListRecommendedUsersQuery, Timeline] = { + val instructionBuilders = Seq( + ReplaceEntryInstructionBuilder(ReplaceAllEntries), + AddEntriesWithReplaceInstructionBuilder() + ) + + val metadataBuilder = UrtMetadataBuilder( + title = None, + scribeConfigBuilder = Some( + StaticTimelineScribeConfigBuilder( + TimelineScribeConfig( + page = Some("list_recommended_users"), + section = None, + entityToken = None))) + ) + + val excludeIdsSelector: PartialFunction[UniversalNoun[_], Long] = { + case item: UserItem => item.id + } + + val cursorBuilder = UnorderedExcludeIdsBottomCursorBuilder( + excludedIdsMaxLengthParam = ExcludedIdsMaxLengthParam, + excludeIdsSelector = excludeIdsSelector) + + UrtDomainMarshaller( + instructionBuilders = instructionBuilders, + metadataBuilder = Some(metadataBuilder), + cursorBuilders = Seq(cursorBuilder) + ) + } + + override val transportMarshaller: TransportMarshaller[Timeline, urt.TimelineResponse] = + urtTransportMarshaller +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_recommended_users/ListRecommendedUsersProductPipelineConfig.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_recommended_users/ListRecommendedUsersProductPipelineConfig.scala new file mode 100644 index 0000000000..e20bb8aec3 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_recommended_users/ListRecommendedUsersProductPipelineConfig.scala @@ -0,0 +1,79 @@ +package com.twitter.home_mixer.product.list_recommended_users + +import com.twitter.home_mixer.marshaller.timelines.RecommendedUsersCursorUnmarshaller +import com.twitter.home_mixer.model.request.HomeMixerRequest +import com.twitter.home_mixer.model.request.ListRecommendedUsersProduct +import com.twitter.home_mixer.model.request.ListRecommendedUsersProductContext +import com.twitter.home_mixer.product.list_recommended_users.model.ListRecommendedUsersQuery +import com.twitter.home_mixer.product.list_recommended_users.param.ListRecommendedUsersParam.ServerMaxResultsParam +import com.twitter.home_mixer.product.list_recommended_users.param.ListRecommendedUsersParamConfig +import com.twitter.home_mixer.service.HomeMixerAccessPolicy.DefaultHomeMixerAccessPolicy +import com.twitter.product_mixer.component_library.premarshaller.cursor.UrtCursorSerializer +import com.twitter.product_mixer.core.functional_component.common.access_policy.AccessPolicy +import com.twitter.product_mixer.core.model.common.identifier.ComponentIdentifier +import com.twitter.product_mixer.core.model.common.identifier.ProductPipelineIdentifier +import com.twitter.product_mixer.core.model.marshalling.request +import com.twitter.product_mixer.core.pipeline.PipelineConfig +import com.twitter.product_mixer.core.pipeline.pipeline_failure.BadRequest +import com.twitter.product_mixer.core.pipeline.pipeline_failure.PipelineFailure +import com.twitter.product_mixer.core.pipeline.product.ProductPipelineConfig +import com.twitter.product_mixer.core.product.ProductParamConfig +import com.twitter.timelines.configapi.Params +import com.twitter.timelines.render.{thriftscala => urt} +import com.twitter.timelines.util.RequestCursorSerializer +import com.twitter.util.Try + +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class ListRecommendedUsersProductPipelineConfig @Inject() ( + listRecommendedUsersMixerPipelineConfig: ListRecommendedUsersMixerPipelineConfig, + listRecommendedUsersParamConfig: ListRecommendedUsersParamConfig) + extends ProductPipelineConfig[ + HomeMixerRequest, + ListRecommendedUsersQuery, + urt.TimelineResponse + ] { + + override val identifier: ProductPipelineIdentifier = + ProductPipelineIdentifier("ListRecommendedUsers") + override val product: request.Product = ListRecommendedUsersProduct + override val paramConfig: ProductParamConfig = listRecommendedUsersParamConfig + + override def pipelineQueryTransformer( + request: HomeMixerRequest, + params: Params + ): ListRecommendedUsersQuery = { + val context = request.productContext match { + case Some(context: ListRecommendedUsersProductContext) => context + case _ => throw PipelineFailure(BadRequest, "ListRecommendedUsersProductContext not found") + } + + val debugOptions = request.debugParams.flatMap(_.debugOptions) + + val pipelineCursor = request.serializedRequestCursor.flatMap { cursor => + Try(UrtCursorSerializer.deserializeUnorderedExcludeIdsCursor(cursor)) + .getOrElse(RecommendedUsersCursorUnmarshaller(RequestCursorSerializer.deserialize(cursor))) + } + + ListRecommendedUsersQuery( + listId = context.listId, + params = params, + clientContext = request.clientContext, + features = None, + pipelineCursor = pipelineCursor, + requestedMaxResults = Some(params(ServerMaxResultsParam)), + debugOptions = debugOptions, + selectedUserIds = context.selectedUserIds, + excludedUserIds = context.excludedUserIds + ) + } + + override def pipelines: Seq[PipelineConfig] = Seq(listRecommendedUsersMixerPipelineConfig) + + override def pipelineSelector(query: ListRecommendedUsersQuery): ComponentIdentifier = + listRecommendedUsersMixerPipelineConfig.identifier + + override val debugAccessPolicies: Set[AccessPolicy] = DefaultHomeMixerAccessPolicy +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_recommended_users/feature_hydrator/BUILD.bazel b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_recommended_users/feature_hydrator/BUILD.bazel new file mode 100644 index 0000000000..326d086a36 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_recommended_users/feature_hydrator/BUILD.bazel @@ -0,0 +1,16 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + strict_deps = True, + tags = ["bazel-compatible"], + dependencies = [ + "home-mixer/server/src/main/scala/com/twitter/home_mixer/model/request", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_recommended_users/model", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/feature_hydrator", + "src/thrift/com/twitter/gizmoduck:thrift-scala", + "src/thrift/com/twitter/socialgraph:thrift-scala", + "stitch/stitch-gizmoduck", + "stitch/stitch-socialgraph", + ], + exports = [], +) diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_recommended_users/feature_hydrator/GizmoduckUserFeatureHydrator.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_recommended_users/feature_hydrator/GizmoduckUserFeatureHydrator.scala new file mode 100644 index 0000000000..d1db2c3488 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_recommended_users/feature_hydrator/GizmoduckUserFeatureHydrator.scala @@ -0,0 +1,59 @@ +package com.twitter.home_mixer.product.list_recommended_users.feature_hydrator + +import com.twitter.gizmoduck.{thriftscala => gt} +import com.twitter.home_mixer.product.list_recommended_users.model.ListFeatures.GizmoduckUserFeature +import com.twitter.product_mixer.component_library.model.candidate.UserCandidate +import com.twitter.product_mixer.core.feature.Feature +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.feature.featuremap.FeatureMapBuilder +import com.twitter.product_mixer.core.functional_component.feature_hydrator.BulkCandidateFeatureHydrator +import com.twitter.product_mixer.core.model.common.CandidateWithFeatures +import com.twitter.product_mixer.core.model.common.identifier.FeatureHydratorIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.spam.rtf.{thriftscala => rtf} +import com.twitter.stitch.Stitch +import com.twitter.stitch.gizmoduck.Gizmoduck +import com.twitter.util.Return + +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class GizmoduckUserFeatureHydrator @Inject() (gizmoduck: Gizmoduck) + extends BulkCandidateFeatureHydrator[PipelineQuery, UserCandidate] { + + override val identifier: FeatureHydratorIdentifier = + FeatureHydratorIdentifier("GizmoduckUser") + + override val features: Set[Feature[_, _]] = Set(GizmoduckUserFeature) + + private val queryFields: Set[gt.QueryFields] = Set(gt.QueryFields.Safety) + + override def apply( + query: PipelineQuery, + candidates: Seq[CandidateWithFeatures[UserCandidate]] + ): Stitch[Seq[FeatureMap]] = { + val context = gt.LookupContext( + forUserId = query.getOptionalUserId, + includeProtected = true, + safetyLevel = Some(rtf.SafetyLevel.Recommendations) + ) + val userIds = candidates.map(_.candidate.id) + + Stitch + .collectToTry( + userIds.map(userId => gizmoduck.getUserById(userId, queryFields, context))).map { + userResults => + val idToUserMap = userResults + .collect { + case Return(user) => user + }.map(user => user.id -> user).toMap + + candidates.map { candidate => + FeatureMapBuilder() + .add(GizmoduckUserFeature, idToUserMap.get(candidate.candidate.id)) + .build() + } + } + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_recommended_users/feature_hydrator/IsListMemberFeatureHydrator.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_recommended_users/feature_hydrator/IsListMemberFeatureHydrator.scala new file mode 100644 index 0000000000..a7c51b4b12 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_recommended_users/feature_hydrator/IsListMemberFeatureHydrator.scala @@ -0,0 +1,53 @@ +package com.twitter.home_mixer.product.list_recommended_users.feature_hydrator + +import com.twitter.home_mixer.model.request.HasListId +import com.twitter.home_mixer.product.list_recommended_users.model.ListFeatures.IsListMemberFeature +import com.twitter.product_mixer.component_library.model.candidate.UserCandidate +import com.twitter.product_mixer.core.feature.Feature +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.feature.featuremap.FeatureMapBuilder +import com.twitter.product_mixer.core.functional_component.feature_hydrator.BulkCandidateFeatureHydrator +import com.twitter.product_mixer.core.model.common.CandidateWithFeatures +import com.twitter.product_mixer.core.model.common.identifier.FeatureHydratorIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.socialgraph.{thriftscala => sg} +import com.twitter.stitch.Stitch +import com.twitter.stitch.socialgraph.SocialGraph + +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class IsListMemberFeatureHydrator @Inject() (socialGraph: SocialGraph) + extends BulkCandidateFeatureHydrator[PipelineQuery with HasListId, UserCandidate] { + + override val identifier: FeatureHydratorIdentifier = + FeatureHydratorIdentifier("IsListMember") + + override val features: Set[Feature[_, _]] = Set(IsListMemberFeature) + + override def apply( + query: PipelineQuery with HasListId, + candidates: Seq[CandidateWithFeatures[UserCandidate]] + ): Stitch[Seq[FeatureMap]] = { + val userIds = candidates.map(_.candidate.id) + val request = sg.IdsRequest( + relationships = Seq( + sg.SrcRelationship( + source = query.listId, + relationshipType = sg.RelationshipType.ListHasMember, + hasRelationship = true, + targets = Some(userIds))), + pageRequest = Some(sg.PageRequest(selectAll = Some(true))) + ) + + socialGraph.ids(request).map(_.ids).map { listMembers => + val listMembersSet = listMembers.toSet + candidates.map { candidate => + FeatureMapBuilder() + .add(IsListMemberFeature, listMembersSet.contains(candidate.candidate.id)) + .build() + } + } + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_recommended_users/filter/BUILD.bazel b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_recommended_users/filter/BUILD.bazel new file mode 100644 index 0000000000..c54e5a7b30 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_recommended_users/filter/BUILD.bazel @@ -0,0 +1,13 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + strict_deps = True, + tags = ["bazel-compatible"], + dependencies = [ + "home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/model/request", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_recommended_users/model", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/filter", + ], + exports = [], +) diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_recommended_users/filter/DropMaxCandidatesByScoreFilter.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_recommended_users/filter/DropMaxCandidatesByScoreFilter.scala new file mode 100644 index 0000000000..d75b301e39 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_recommended_users/filter/DropMaxCandidatesByScoreFilter.scala @@ -0,0 +1,29 @@ +package com.twitter.home_mixer.product.list_recommended_users.filter + +import com.twitter.home_mixer.product.list_recommended_users.model.ListFeatures.ScoreFeature +import com.twitter.product_mixer.component_library.model.candidate.UserCandidate +import com.twitter.product_mixer.core.functional_component.filter.Filter +import com.twitter.product_mixer.core.functional_component.filter.FilterResult +import com.twitter.product_mixer.core.model.common.CandidateWithFeatures +import com.twitter.product_mixer.core.model.common.identifier.FilterIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.stitch.Stitch + +object DropMaxCandidatesByScoreFilter extends Filter[PipelineQuery, UserCandidate] { + + override val identifier: FilterIdentifier = FilterIdentifier("DropMaxCandidatesByScore") + + private val MaxSimilarUserCandidates = 1000 + + override def apply( + query: PipelineQuery, + candidates: Seq[CandidateWithFeatures[UserCandidate]] + ): Stitch[FilterResult[UserCandidate]] = { + + val sortedCandidates = candidates.sortBy(-_.features.getOrElse(ScoreFeature, 0.0)) + + val (kept, removed) = sortedCandidates.map(_.candidate).splitAt(MaxSimilarUserCandidates) + + Stitch.value(FilterResult(kept, removed)) + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_recommended_users/filter/PreviouslyServedUsersFilter.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_recommended_users/filter/PreviouslyServedUsersFilter.scala new file mode 100644 index 0000000000..97cd6a5c15 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_recommended_users/filter/PreviouslyServedUsersFilter.scala @@ -0,0 +1,35 @@ +package com.twitter.home_mixer.product.list_recommended_users.filter + +import com.twitter.home_mixer.functional_component.feature_hydrator.ListMembersFeature +import com.twitter.home_mixer.product.list_recommended_users.model.ListRecommendedUsersQuery +import com.twitter.product_mixer.component_library.model.candidate.UserCandidate +import com.twitter.product_mixer.core.functional_component.filter.Filter +import com.twitter.product_mixer.core.functional_component.filter.FilterResult +import com.twitter.product_mixer.core.model.common.CandidateWithFeatures +import com.twitter.product_mixer.core.model.common.identifier.FilterIdentifier +import com.twitter.stitch.Stitch + +object PreviouslyServedUsersFilter extends Filter[ListRecommendedUsersQuery, UserCandidate] { + + override val identifier: FilterIdentifier = FilterIdentifier("PreviouslyServedUsers") + + override def apply( + query: ListRecommendedUsersQuery, + candidates: Seq[CandidateWithFeatures[UserCandidate]] + ): Stitch[FilterResult[UserCandidate]] = { + + val recentListMembers = query.features.map(_.getOrElse(ListMembersFeature, Seq.empty)) + + val servedUserIds = query.pipelineCursor.map(_.excludedIds) + + val excludedUserIds = (recentListMembers.getOrElse(Seq.empty) ++ + query.selectedUserIds.getOrElse(Seq.empty) ++ + query.excludedUserIds.getOrElse(Seq.empty) ++ + servedUserIds.getOrElse(Seq.empty)).toSet + + val (removed, kept) = + candidates.map(_.candidate).partition(candidate => excludedUserIds.contains(candidate.id)) + + Stitch.value(FilterResult(kept = kept, removed = removed)) + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_recommended_users/model/BUILD.bazel b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_recommended_users/model/BUILD.bazel new file mode 100644 index 0000000000..78302f75b4 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_recommended_users/model/BUILD.bazel @@ -0,0 +1,16 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + strict_deps = True, + tags = ["bazel-compatible"], + dependencies = [ + "home-mixer/server/src/main/scala/com/twitter/home_mixer/model", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/model/request", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/cursor", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline", + ], + exports = [ + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/cursor", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline", + ], +) diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_recommended_users/model/ListFeatures.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_recommended_users/model/ListFeatures.scala new file mode 100644 index 0000000000..9abcd6fb2d --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_recommended_users/model/ListFeatures.scala @@ -0,0 +1,12 @@ +package com.twitter.home_mixer.product.list_recommended_users.model + +import com.twitter.gizmoduck.{thriftscala => gt} +import com.twitter.product_mixer.component_library.model.candidate.UserCandidate +import com.twitter.product_mixer.core.feature.Feature + +object ListFeatures { + // Candidate features + object GizmoduckUserFeature extends Feature[UserCandidate, Option[gt.User]] + object IsListMemberFeature extends Feature[UserCandidate, Boolean] + object ScoreFeature extends Feature[UserCandidate, Double] +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_recommended_users/model/ListRecommendedUsersQuery.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_recommended_users/model/ListRecommendedUsersQuery.scala new file mode 100644 index 0000000000..8d8c2e6ea2 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_recommended_users/model/ListRecommendedUsersQuery.scala @@ -0,0 +1,30 @@ +package com.twitter.home_mixer.product.list_recommended_users.model + +import com.twitter.home_mixer.model.request.HasListId +import com.twitter.home_mixer.model.request.ListRecommendedUsersProduct +import com.twitter.product_mixer.component_library.model.cursor.UrtUnorderedExcludeIdsCursor +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.model.marshalling.request._ +import com.twitter.product_mixer.core.pipeline.HasPipelineCursor +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.timelines.configapi.Params + +case class ListRecommendedUsersQuery( + override val listId: Long, + override val params: Params, + override val clientContext: ClientContext, + override val pipelineCursor: Option[UrtUnorderedExcludeIdsCursor], + override val requestedMaxResults: Option[Int], + override val debugOptions: Option[DebugOptions], + override val features: Option[FeatureMap], + selectedUserIds: Option[Seq[Long]], + excludedUserIds: Option[Seq[Long]]) + extends PipelineQuery + with HasPipelineCursor[UrtUnorderedExcludeIdsCursor] + with HasListId { + + override val product: Product = ListRecommendedUsersProduct + + override def withFeatureMap(features: FeatureMap): ListRecommendedUsersQuery = + copy(features = Some(features)) +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_recommended_users/param/BUILD.bazel b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_recommended_users/param/BUILD.bazel new file mode 100644 index 0000000000..18867ad9e9 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_recommended_users/param/BUILD.bazel @@ -0,0 +1,12 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + strict_deps = True, + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/javax/inject:javax.inject", + "configapi/configapi-core/src/main/scala/com/twitter/timelines/configapi", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/param/decider", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/product", + ], +) diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_recommended_users/param/ListRecommendedUsersParam.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_recommended_users/param/ListRecommendedUsersParam.scala new file mode 100644 index 0000000000..3822a19876 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_recommended_users/param/ListRecommendedUsersParam.scala @@ -0,0 +1,23 @@ +package com.twitter.home_mixer.product.list_recommended_users.param + +import com.twitter.timelines.configapi.FSBoundedParam + +object ListRecommendedUsersParam { + val SupportedClientFSName = "list_recommended_users_supported_client" + + object ServerMaxResultsParam + extends FSBoundedParam[Int]( + name = "list_recommended_users_server_max_results", + default = 10, + min = 1, + max = 500 + ) + + object ExcludedIdsMaxLengthParam + extends FSBoundedParam[Int]( + name = "list_recommended_users_excluded_ids_max_length", + default = 2000, + min = 0, + max = 5000 + ) +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_recommended_users/param/ListRecommendedUsersParamConfig.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_recommended_users/param/ListRecommendedUsersParamConfig.scala new file mode 100644 index 0000000000..b162ef755b --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_recommended_users/param/ListRecommendedUsersParamConfig.scala @@ -0,0 +1,22 @@ +package com.twitter.home_mixer.product.list_recommended_users.param + +import com.twitter.home_mixer.param.decider.DeciderKey +import com.twitter.home_mixer.product.list_recommended_users.param.ListRecommendedUsersParam.ExcludedIdsMaxLengthParam +import com.twitter.home_mixer.product.list_recommended_users.param.ListRecommendedUsersParam.ServerMaxResultsParam +import com.twitter.home_mixer.product.list_recommended_users.param.ListRecommendedUsersParam.SupportedClientFSName +import com.twitter.product_mixer.core.product.ProductParamConfig +import com.twitter.servo.decider.DeciderKeyName + +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class ListRecommendedUsersParamConfig @Inject() () extends ProductParamConfig { + override val enabledDeciderKey: DeciderKeyName = DeciderKey.EnableListRecommendedUsersProduct + override val supportedClientFSName: String = SupportedClientFSName + + override val boundedIntFSOverrides = Seq( + ServerMaxResultsParam, + ExcludedIdsMaxLengthParam + ) +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_tweets/BUILD.bazel b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_tweets/BUILD.bazel new file mode 100644 index 0000000000..e60fc064e6 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_tweets/BUILD.bazel @@ -0,0 +1,59 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + strict_deps = True, + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/javax/inject:javax.inject", + "ads-injection/lib/src/main/scala/com/twitter/goldfinch/api", + "finatra/inject/inject-core/src/main/scala", + "finatra/inject/inject-core/src/main/scala/com/twitter/inject", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/candidate_pipeline", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/candidate_source", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/builder", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/decorator/urt/builder", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/filter", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/gate", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/selector", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/side_effect", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/marshaller/timelines", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/model", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/model/request", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/param", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_tweets/model", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_tweets/param", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/service", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/util", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/candidate_source/ads", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/candidate_source/timeline_service", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/feature_hydrator/candidate/ads", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/feature_hydrator/candidate/param_gated", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/gate", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/candidate/ads", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/pipeline/candidate/ads", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/premarshaller/urt", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/selector", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/selector/ads", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/side_effect", + "product-mixer/core/src/main/java/com/twitter/product_mixer/core/product/guice/scope", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/configapi", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/decorator", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/decorator/urt/builder", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/common/presentation", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/request", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/response/urt/item", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/candidate", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/mixer", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/product", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/product", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/product/guice", + "src/thrift/com/twitter/timelines/render:thrift-scala", + "timelines/src/main/scala/com/twitter/timelines/injection/scribe", + ], + exports = [ + "src/thrift/com/twitter/timelines/render:thrift-scala", + ], +) diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_tweets/ListTweetsAdsCandidatePipelineBuilder.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_tweets/ListTweetsAdsCandidatePipelineBuilder.scala new file mode 100644 index 0000000000..77c61710a4 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_tweets/ListTweetsAdsCandidatePipelineBuilder.scala @@ -0,0 +1,93 @@ +package com.twitter.home_mixer.product.list_tweets + +import com.twitter.adserver.{thriftscala => ads} +import com.twitter.home_mixer.functional_component.decorator.HomeAdsClientEventDetailsBuilder +import com.twitter.home_mixer.functional_component.gate.ExcludeSoftUserGate +import com.twitter.home_mixer.param.HomeGlobalParams +import com.twitter.home_mixer.param.HomeGlobalParams.EnableAdvertiserBrandSafetySettingsFeatureHydratorParam +import com.twitter.home_mixer.product.list_tweets.model.ListTweetsQuery +import com.twitter.home_mixer.product.list_tweets.param.ListTweetsParam.EnableAdsCandidatePipelineParam +import com.twitter.product_mixer.component_library.candidate_source.ads.AdsProdThriftCandidateSource +import com.twitter.product_mixer.component_library.decorator.urt.UrtItemCandidateDecorator +import com.twitter.product_mixer.component_library.decorator.urt.builder.contextual_ref.ContextualTweetRefBuilder +import com.twitter.product_mixer.component_library.decorator.urt.builder.item.ad.AdsCandidateUrtItemBuilder +import com.twitter.product_mixer.component_library.decorator.urt.builder.metadata.ClientEventInfoBuilder +import com.twitter.product_mixer.component_library.feature_hydrator.candidate.ads.AdvertiserBrandSafetySettingsFeatureHydrator +import com.twitter.product_mixer.component_library.feature_hydrator.candidate.param_gated.ParamGatedCandidateFeatureHydrator +import com.twitter.product_mixer.component_library.gate.NonEmptyCandidatesGate +import com.twitter.product_mixer.component_library.model.candidate.ads.AdsCandidate +import com.twitter.product_mixer.component_library.pipeline.candidate.ads.AdsDependentCandidatePipelineConfig +import com.twitter.product_mixer.component_library.pipeline.candidate.ads.AdsDependentCandidatePipelineConfigBuilder +import com.twitter.product_mixer.component_library.pipeline.candidate.ads.CountCandidatesFromPipelines +import com.twitter.product_mixer.component_library.pipeline.candidate.ads.StaticAdsDisplayLocationBuilder +import com.twitter.product_mixer.component_library.pipeline.candidate.ads.ValidAdImpressionIdFilter +import com.twitter.product_mixer.core.functional_component.common.CandidateScope +import com.twitter.product_mixer.core.gate.ParamNotGate +import com.twitter.product_mixer.core.model.common.identifier.CandidatePipelineIdentifier +import com.twitter.product_mixer.core.model.marshalling.response.rtf.safety_level.TimelineHomePromotedHydrationSafetyLevel +import com.twitter.product_mixer.core.model.marshalling.response.urt.contextual_ref.TweetHydrationContext +import com.twitter.timelines.injection.scribe.InjectionScribeUtil +import com.twitter.timelineservice.suggests.{thriftscala => st} +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class ListTweetsAdsCandidatePipelineBuilder @Inject() ( + adsCandidatePipelineConfigBuilder: AdsDependentCandidatePipelineConfigBuilder, + adsCandidateSource: AdsProdThriftCandidateSource, + advertiserBrandSafetySettingsFeatureHydrator: AdvertiserBrandSafetySettingsFeatureHydrator[ + ListTweetsQuery, + AdsCandidate + ]) { + + private val identifier: CandidatePipelineIdentifier = CandidatePipelineIdentifier("ListTweetsAds") + + private val suggestType = st.SuggestType.Promoted + + private val clientEventInfoBuilder = ClientEventInfoBuilder( + component = InjectionScribeUtil.scribeComponent(suggestType).get, + detailsBuilder = Some(HomeAdsClientEventDetailsBuilder(Some(suggestType.name))) + ) + + private val contextualTweetRefBuilder = ContextualTweetRefBuilder( + TweetHydrationContext( + safetyLevelOverride = Some(TimelineHomePromotedHydrationSafetyLevel), + outerTweetContext = None + )) + + private val decorator = UrtItemCandidateDecorator( + AdsCandidateUrtItemBuilder( + tweetClientEventInfoBuilder = Some(clientEventInfoBuilder), + contextualTweetRefBuilder = Some(contextualTweetRefBuilder) + ) + ) + + def build( + organicCandidatePipelines: CandidateScope + ): AdsDependentCandidatePipelineConfig[ListTweetsQuery] = + adsCandidatePipelineConfigBuilder.build[ListTweetsQuery]( + adsCandidateSource = adsCandidateSource, + identifier = identifier, + adsDisplayLocationBuilder = + StaticAdsDisplayLocationBuilder(ads.DisplayLocation.TimelineHomeReverseChron), + countNumOrganicItems = CountCandidatesFromPipelines(organicCandidatePipelines), + supportedClientParam = Some(EnableAdsCandidatePipelineParam), + gates = Seq( + ParamNotGate( + name = "AdsDisableInjectionBasedOnUserRole", + param = HomeGlobalParams.AdsDisableInjectionBasedOnUserRoleParam + ), + ExcludeSoftUserGate, + NonEmptyCandidatesGate(organicCandidatePipelines) + ), + filters = Seq(ValidAdImpressionIdFilter), + postFilterFeatureHydration = Seq( + ParamGatedCandidateFeatureHydrator( + EnableAdvertiserBrandSafetySettingsFeatureHydratorParam, + advertiserBrandSafetySettingsFeatureHydrator + ) + ), + decorator = Some(decorator), + urtRequest = Some(true), + ) +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_tweets/ListTweetsMixerPipelineConfig.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_tweets/ListTweetsMixerPipelineConfig.scala new file mode 100644 index 0000000000..f4e20ce7e9 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_tweets/ListTweetsMixerPipelineConfig.scala @@ -0,0 +1,155 @@ +package com.twitter.home_mixer.product.list_tweets + +import com.twitter.clientapp.{thriftscala => ca} +import com.twitter.goldfinch.api.AdsInjectionSurfaceAreas +import com.twitter.home_mixer.candidate_pipeline.ConversationServiceCandidatePipelineConfigBuilder +import com.twitter.home_mixer.functional_component.decorator.ListConversationServiceCandidateDecorator +import com.twitter.home_mixer.functional_component.feature_hydrator.RequestQueryFeatureHydrator +import com.twitter.home_mixer.functional_component.side_effect.HomeScribeClientEventSideEffect +import com.twitter.home_mixer.model.GapIncludeInstruction +import com.twitter.home_mixer.product.list_tweets.model.ListTweetsQuery +import com.twitter.home_mixer.util.CandidatesUtil +import com.twitter.logpipeline.client.common.EventPublisher +import com.twitter.product_mixer.component_library.gate.NonEmptyCandidatesGate +import com.twitter.product_mixer.component_library.premarshaller.urt.UrtDomainMarshaller +import com.twitter.product_mixer.component_library.premarshaller.urt.builder.AddEntriesWithReplaceAndShowAlertInstructionBuilder +import com.twitter.product_mixer.component_library.premarshaller.urt.builder.OrderedBottomCursorBuilder +import com.twitter.product_mixer.component_library.premarshaller.urt.builder.OrderedGapCursorBuilder +import com.twitter.product_mixer.component_library.premarshaller.urt.builder.OrderedTopCursorBuilder +import com.twitter.product_mixer.component_library.premarshaller.urt.builder.ReplaceAllEntries +import com.twitter.product_mixer.component_library.premarshaller.urt.builder.ReplaceEntryInstructionBuilder +import com.twitter.product_mixer.component_library.premarshaller.urt.builder.ShowAlertInstructionBuilder +import com.twitter.product_mixer.component_library.premarshaller.urt.builder.StaticTimelineScribeConfigBuilder +import com.twitter.product_mixer.component_library.premarshaller.urt.builder.UrtMetadataBuilder +import com.twitter.product_mixer.component_library.selector.InsertAppendResults +import com.twitter.product_mixer.component_library.selector.UpdateSortCandidates +import com.twitter.product_mixer.component_library.selector.ads.AdsInjector +import com.twitter.product_mixer.component_library.selector.ads.InsertAdResults +import com.twitter.product_mixer.core.functional_component.common.SpecificPipelines +import com.twitter.product_mixer.core.functional_component.feature_hydrator.QueryFeatureHydrator +import com.twitter.product_mixer.core.functional_component.marshaller.TransportMarshaller +import com.twitter.product_mixer.core.functional_component.marshaller.response.urt.UrtTransportMarshaller +import com.twitter.product_mixer.core.functional_component.premarshaller.DomainMarshaller +import com.twitter.product_mixer.core.functional_component.selector.Selector +import com.twitter.product_mixer.core.functional_component.side_effect.PipelineResultSideEffect +import com.twitter.product_mixer.core.model.common.UniversalNoun +import com.twitter.product_mixer.core.model.common.identifier.CandidatePipelineIdentifier +import com.twitter.product_mixer.core.model.common.identifier.MixerPipelineIdentifier +import com.twitter.product_mixer.core.model.marshalling.response.urt.Timeline +import com.twitter.product_mixer.core.model.marshalling.response.urt.TimelineModule +import com.twitter.product_mixer.core.model.marshalling.response.urt.TimelineScribeConfig +import com.twitter.product_mixer.core.model.marshalling.response.urt.item.tweet.TweetItem +import com.twitter.product_mixer.core.pipeline.FailOpenPolicy +import com.twitter.product_mixer.core.pipeline.candidate.CandidatePipelineConfig +import com.twitter.product_mixer.core.pipeline.candidate.DependentCandidatePipelineConfig +import com.twitter.product_mixer.core.pipeline.mixer.MixerPipelineConfig +import com.twitter.timelines.render.{thriftscala => urt} +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class ListTweetsMixerPipelineConfig @Inject() ( + listTweetsTimelineServiceCandidatePipelineConfig: ListTweetsTimelineServiceCandidatePipelineConfig, + conversationServiceCandidatePipelineConfigBuilder: ConversationServiceCandidatePipelineConfigBuilder[ + ListTweetsQuery + ], + listTweetsAdsCandidatePipelineBuilder: ListTweetsAdsCandidatePipelineBuilder, + requestQueryFeatureHydrator: RequestQueryFeatureHydrator[ListTweetsQuery], + adsInjector: AdsInjector, + clientEventsScribeEventPublisher: EventPublisher[ca.LogEvent], + urtTransportMarshaller: UrtTransportMarshaller) + extends MixerPipelineConfig[ListTweetsQuery, Timeline, urt.TimelineResponse] { + + override val identifier: MixerPipelineIdentifier = MixerPipelineIdentifier("ListTweets") + + private val conversationServiceCandidatePipelineConfig = + conversationServiceCandidatePipelineConfigBuilder.build( + Seq( + NonEmptyCandidatesGate( + SpecificPipelines(listTweetsTimelineServiceCandidatePipelineConfig.identifier)) + ), + ListConversationServiceCandidateDecorator() + ) + + private val listTweetsAdsCandidatePipelineConfig = listTweetsAdsCandidatePipelineBuilder.build( + SpecificPipelines(listTweetsTimelineServiceCandidatePipelineConfig.identifier) + ) + + override val candidatePipelines: Seq[CandidatePipelineConfig[ListTweetsQuery, _, _, _]] = + Seq(listTweetsTimelineServiceCandidatePipelineConfig) + + override val dependentCandidatePipelines: Seq[ + DependentCandidatePipelineConfig[ListTweetsQuery, _, _, _] + ] = + Seq(conversationServiceCandidatePipelineConfig, listTweetsAdsCandidatePipelineConfig) + + override val failOpenPolicies: Map[CandidatePipelineIdentifier, FailOpenPolicy] = Map( + conversationServiceCandidatePipelineConfig.identifier -> FailOpenPolicy.Always, + listTweetsAdsCandidatePipelineConfig.identifier -> FailOpenPolicy.Always) + + override val resultSelectors: Seq[Selector[ListTweetsQuery]] = Seq( + UpdateSortCandidates( + ordering = CandidatesUtil.reverseChronTweetsOrdering, + candidatePipeline = conversationServiceCandidatePipelineConfig.identifier + ), + InsertAppendResults(candidatePipeline = conversationServiceCandidatePipelineConfig.identifier), + InsertAdResults( + surfaceAreaName = AdsInjectionSurfaceAreas.HomeTimeline, + adsInjector = adsInjector.forSurfaceArea(AdsInjectionSurfaceAreas.HomeTimeline), + adsCandidatePipeline = listTweetsAdsCandidatePipelineConfig.identifier + ), + ) + + override val fetchQueryFeatures: Seq[QueryFeatureHydrator[ListTweetsQuery]] = Seq( + requestQueryFeatureHydrator + ) + + private val homeScribeClientEventSideEffect = HomeScribeClientEventSideEffect( + logPipelinePublisher = clientEventsScribeEventPublisher, + injectedTweetsCandidatePipelineIdentifiers = + Seq(conversationServiceCandidatePipelineConfig.identifier), + adsCandidatePipelineIdentifier = listTweetsAdsCandidatePipelineConfig.identifier, + ) + + override val resultSideEffects: Seq[PipelineResultSideEffect[ListTweetsQuery, Timeline]] = + Seq(homeScribeClientEventSideEffect) + + override val domainMarshaller: DomainMarshaller[ListTweetsQuery, Timeline] = { + val instructionBuilders = Seq( + ReplaceEntryInstructionBuilder(ReplaceAllEntries), + AddEntriesWithReplaceAndShowAlertInstructionBuilder(), + ShowAlertInstructionBuilder() + ) + + val idSelector: PartialFunction[UniversalNoun[_], Long] = { + // exclude ads while determining tweet cursor values + case item: TweetItem if item.promotedMetadata.isEmpty => item.id + case module: TimelineModule + if module.items.headOption.exists(_.item.isInstanceOf[TweetItem]) => + module.items.last.item match { + case item: TweetItem => item.id + } + } + + val topCursorBuilder = OrderedTopCursorBuilder(idSelector) + val bottomCursorBuilder = + OrderedBottomCursorBuilder(idSelector, GapIncludeInstruction.inverse()) + val gapCursorBuilder = OrderedGapCursorBuilder(idSelector, GapIncludeInstruction) + + val metadataBuilder = UrtMetadataBuilder( + title = None, + scribeConfigBuilder = Some( + StaticTimelineScribeConfigBuilder( + TimelineScribeConfig(page = Some("list_tweets"), section = None, entityToken = None))) + ) + + UrtDomainMarshaller( + instructionBuilders = instructionBuilders, + metadataBuilder = Some(metadataBuilder), + cursorBuilders = Seq(topCursorBuilder, bottomCursorBuilder, gapCursorBuilder) + ) + } + + override val transportMarshaller: TransportMarshaller[Timeline, urt.TimelineResponse] = + urtTransportMarshaller +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_tweets/ListTweetsProductPipelineConfig.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_tweets/ListTweetsProductPipelineConfig.scala new file mode 100644 index 0000000000..06afd9a92d --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_tweets/ListTweetsProductPipelineConfig.scala @@ -0,0 +1,94 @@ +package com.twitter.home_mixer.product.list_tweets + +import com.twitter.home_mixer.marshaller.timelines.ChronologicalCursorUnmarshaller +import com.twitter.home_mixer.model.request.HomeMixerRequest +import com.twitter.home_mixer.model.request.ListTweetsProduct +import com.twitter.home_mixer.model.request.ListTweetsProductContext +import com.twitter.home_mixer.product.list_tweets.model.ListTweetsQuery +import com.twitter.home_mixer.product.list_tweets.param.ListTweetsParam.ServerMaxResultsParam +import com.twitter.home_mixer.product.list_tweets.param.ListTweetsParamConfig +import com.twitter.home_mixer.service.HomeMixerAccessPolicy.DefaultHomeMixerAccessPolicy +import com.twitter.product_mixer.component_library.model.cursor.UrtOrderedCursor +import com.twitter.product_mixer.component_library.premarshaller.cursor.UrtCursorSerializer +import com.twitter.product_mixer.core.functional_component.common.access_policy.AccessPolicy +import com.twitter.product_mixer.core.model.common.identifier.ComponentIdentifier +import com.twitter.product_mixer.core.model.common.identifier.ProductPipelineIdentifier +import com.twitter.product_mixer.core.model.marshalling.request +import com.twitter.product_mixer.core.model.marshalling.response.urt.operation.GapCursor +import com.twitter.product_mixer.core.model.marshalling.response.urt.operation.TopCursor +import com.twitter.product_mixer.core.pipeline.PipelineConfig +import com.twitter.product_mixer.core.pipeline.pipeline_failure.BadRequest +import com.twitter.product_mixer.core.pipeline.pipeline_failure.MalformedCursor +import com.twitter.product_mixer.core.pipeline.pipeline_failure.PipelineFailure +import com.twitter.product_mixer.core.pipeline.product.ProductPipelineConfig +import com.twitter.product_mixer.core.product.ProductParamConfig +import com.twitter.product_mixer.core.util.SortIndexBuilder +import com.twitter.timelines.configapi.Params +import com.twitter.timelines.render.{thriftscala => urt} +import com.twitter.timelines.util.RequestCursorSerializer +import com.twitter.util.Time +import com.twitter.util.Try +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class ListTweetsProductPipelineConfig @Inject() ( + listTweetsMixerPipelineConfig: ListTweetsMixerPipelineConfig, + listTweetsParamConfig: ListTweetsParamConfig) + extends ProductPipelineConfig[HomeMixerRequest, ListTweetsQuery, urt.TimelineResponse] { + + override val identifier: ProductPipelineIdentifier = ProductPipelineIdentifier("ListTweets") + override val product: request.Product = ListTweetsProduct + override val paramConfig: ProductParamConfig = listTweetsParamConfig + + override def pipelineQueryTransformer( + request: HomeMixerRequest, + params: Params + ): ListTweetsQuery = { + val context = request.productContext match { + case Some(context: ListTweetsProductContext) => context + case _ => throw PipelineFailure(BadRequest, "ListTweetsProductContext not found") + } + + val debugOptions = request.debugParams.flatMap(_.debugOptions) + + /** + * Unlike other clients, newly created tweets on Android have the sort index set to the current + * time instead of the top sort index + 1, so these tweets get stuck at the top of the timeline + * if subsequent timeline responses use the sort index from the previous response instead of + * the current time. + */ + val pipelineCursor = request.serializedRequestCursor.flatMap { cursor => + Try(UrtCursorSerializer.deserializeOrderedCursor(cursor)) + .getOrElse(ChronologicalCursorUnmarshaller(RequestCursorSerializer.deserialize(cursor))) + .map { + case UrtOrderedCursor(_, id, Some(GapCursor), gapBoundaryId) + if id.isEmpty || gapBoundaryId.isEmpty => + throw PipelineFailure(MalformedCursor, "Gap Cursor bounds not defined") + case topCursor @ UrtOrderedCursor(_, _, Some(TopCursor), _) => + val queryTime = debugOptions.flatMap(_.requestTimeOverride).getOrElse(Time.now) + topCursor.copy(initialSortIndex = SortIndexBuilder.timeToId(queryTime)) + case cursor => cursor + } + } + + ListTweetsQuery( + params = params, + clientContext = request.clientContext, + features = None, + pipelineCursor = pipelineCursor, + requestedMaxResults = Some(params(ServerMaxResultsParam)), + debugOptions = debugOptions, + listId = context.listId, + deviceContext = context.deviceContext, + dspClientContext = context.dspClientContext + ) + } + + override def pipelines: Seq[PipelineConfig] = Seq(listTweetsMixerPipelineConfig) + + override def pipelineSelector(query: ListTweetsQuery): ComponentIdentifier = + listTweetsMixerPipelineConfig.identifier + + override val debugAccessPolicies: Set[AccessPolicy] = DefaultHomeMixerAccessPolicy +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_tweets/ListTweetsTimelineServiceCandidatePipelineConfig.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_tweets/ListTweetsTimelineServiceCandidatePipelineConfig.scala new file mode 100644 index 0000000000..aba9f47e0c --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_tweets/ListTweetsTimelineServiceCandidatePipelineConfig.scala @@ -0,0 +1,54 @@ +package com.twitter.home_mixer.product.list_tweets + +import com.twitter.home_mixer.candidate_pipeline.TimelineServiceResponseFeatureTransformer +import com.twitter.home_mixer.marshaller.timelines.TimelineServiceCursorMarshaller +import com.twitter.home_mixer.product.list_tweets.model.ListTweetsQuery +import com.twitter.home_mixer.product.list_tweets.param.ListTweetsParam.ServerMaxResultsParam +import com.twitter.product_mixer.component_library.candidate_source.timeline_service.TimelineServiceTweetCandidateSource +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.functional_component.candidate_source.BaseCandidateSource +import com.twitter.product_mixer.core.functional_component.transformer.CandidateFeatureTransformer +import com.twitter.product_mixer.core.functional_component.transformer.CandidatePipelineQueryTransformer +import com.twitter.product_mixer.core.functional_component.transformer.CandidatePipelineResultsTransformer +import com.twitter.product_mixer.core.model.common.identifier.CandidatePipelineIdentifier +import com.twitter.product_mixer.core.pipeline.candidate.CandidatePipelineConfig +import com.twitter.timelineservice.{thriftscala => t} +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class ListTweetsTimelineServiceCandidatePipelineConfig @Inject() ( + timelineServiceTweetCandidateSource: TimelineServiceTweetCandidateSource) + extends CandidatePipelineConfig[ListTweetsQuery, t.TimelineQuery, t.Tweet, TweetCandidate] { + + override val identifier: CandidatePipelineIdentifier = + CandidatePipelineIdentifier("ListTweetsTimelineServiceTweets") + + override val queryTransformer: CandidatePipelineQueryTransformer[ + ListTweetsQuery, + t.TimelineQuery + ] = { query => + val timelineQueryOptions = t.TimelineQueryOptions( + contextualUserId = query.clientContext.userId, + ) + + t.TimelineQuery( + timelineType = t.TimelineType.List, + timelineId = query.listId, + maxCount = query.maxResults(ServerMaxResultsParam).toShort, + cursor2 = query.pipelineCursor.flatMap(TimelineServiceCursorMarshaller(_)), + options = Some(timelineQueryOptions), + timelineId2 = Some(t.TimelineId(t.TimelineType.List, query.listId, None)) + ) + } + + override def candidateSource: BaseCandidateSource[t.TimelineQuery, t.Tweet] = + timelineServiceTweetCandidateSource + + override val resultTransformer: CandidatePipelineResultsTransformer[t.Tweet, TweetCandidate] = { + sourceResult => TweetCandidate(id = sourceResult.statusId) + } + + override val featuresFromCandidateSourceTransformers: Seq[CandidateFeatureTransformer[t.Tweet]] = + Seq(TimelineServiceResponseFeatureTransformer) +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_tweets/model/BUILD.bazel b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_tweets/model/BUILD.bazel new file mode 100644 index 0000000000..fac4ba9498 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_tweets/model/BUILD.bazel @@ -0,0 +1,19 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + strict_deps = True, + tags = ["bazel-compatible"], + dependencies = [ + "home-mixer/server/src/main/scala/com/twitter/home_mixer/model", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/model/request", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_tweets/param", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/cursor", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/query/ads", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline", + ], + exports = [ + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/cursor", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/query/ads", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline", + ], +) diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_tweets/model/ListTweetsQuery.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_tweets/model/ListTweetsQuery.scala new file mode 100644 index 0000000000..ce8c73c0a6 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_tweets/model/ListTweetsQuery.scala @@ -0,0 +1,38 @@ +package com.twitter.home_mixer.product.list_tweets.model + +import com.twitter.adserver.thriftscala.HomeTimelineType +import com.twitter.adserver.thriftscala.TimelineRequestParams +import com.twitter.dspbidder.commons.{thriftscala => dsp} +import com.twitter.home_mixer.model.HomeAdsQuery +import com.twitter.home_mixer.model.request.DeviceContext +import com.twitter.home_mixer.model.request.HasListId +import com.twitter.home_mixer.model.request.ListTweetsProduct +import com.twitter.product_mixer.component_library.model.cursor.UrtOrderedCursor +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.model.marshalling.request._ +import com.twitter.product_mixer.core.pipeline.HasPipelineCursor +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.timelines.configapi.Params + +case class ListTweetsQuery( + override val params: Params, + override val clientContext: ClientContext, + override val pipelineCursor: Option[UrtOrderedCursor], + override val requestedMaxResults: Option[Int], + override val debugOptions: Option[DebugOptions], + override val features: Option[FeatureMap], + override val listId: Long, + override val deviceContext: Option[DeviceContext], + override val dspClientContext: Option[dsp.DspClientContext]) + extends PipelineQuery + with HasPipelineCursor[UrtOrderedCursor] + with HasListId + with HomeAdsQuery { + override val product: Product = ListTweetsProduct + + override def withFeatureMap(features: FeatureMap): ListTweetsQuery = + copy(features = Some(features)) + + override val timelineRequestParams: Option[TimelineRequestParams] = + Some(TimelineRequestParams(homeTimelineType = Some(HomeTimelineType.HomeLatest))) +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_tweets/param/BUILD.bazel b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_tweets/param/BUILD.bazel new file mode 100644 index 0000000000..18867ad9e9 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_tweets/param/BUILD.bazel @@ -0,0 +1,12 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + strict_deps = True, + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/javax/inject:javax.inject", + "configapi/configapi-core/src/main/scala/com/twitter/timelines/configapi", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/param/decider", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/product", + ], +) diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_tweets/param/ListTweetsParam.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_tweets/param/ListTweetsParam.scala new file mode 100644 index 0000000000..827d29f10e --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_tweets/param/ListTweetsParam.scala @@ -0,0 +1,22 @@ +package com.twitter.home_mixer.product.list_tweets.param + +import com.twitter.timelines.configapi.FSBoundedParam +import com.twitter.timelines.configapi.FSParam + +object ListTweetsParam { + val SupportedClientFSName = "list_tweets_supported_client" + + object EnableAdsCandidatePipelineParam + extends FSParam[Boolean]( + name = "list_tweets_enable_ads", + default = false + ) + + object ServerMaxResultsParam + extends FSBoundedParam[Int]( + name = "list_tweets_server_max_results", + default = 100, + min = 1, + max = 500 + ) +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_tweets/param/ListTweetsParamConfig.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_tweets/param/ListTweetsParamConfig.scala new file mode 100644 index 0000000000..01575db925 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/list_tweets/param/ListTweetsParamConfig.scala @@ -0,0 +1,23 @@ +package com.twitter.home_mixer.product.list_tweets.param + +import com.twitter.home_mixer.param.decider.DeciderKey +import com.twitter.home_mixer.product.list_tweets.param.ListTweetsParam.EnableAdsCandidatePipelineParam +import com.twitter.home_mixer.product.list_tweets.param.ListTweetsParam.ServerMaxResultsParam +import com.twitter.home_mixer.product.list_tweets.param.ListTweetsParam.SupportedClientFSName +import com.twitter.product_mixer.core.product.ProductParamConfig +import com.twitter.servo.decider.DeciderKeyName +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class ListTweetsParamConfig @Inject() () extends ProductParamConfig { + override val enabledDeciderKey: DeciderKeyName = DeciderKey.EnableListTweetsProduct + override val supportedClientFSName: String = SupportedClientFSName + + override val booleanFSOverrides = + Seq(EnableAdsCandidatePipelineParam) + + override val boundedIntFSOverrides = Seq( + ServerMaxResultsParam + ) +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/BUILD.bazel b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/BUILD.bazel new file mode 100644 index 0000000000..e016b2a824 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/BUILD.bazel @@ -0,0 +1,57 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + strict_deps = True, + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/javax/inject:javax.inject", + "finatra/inject/inject-core/src/main/scala", + "finatra/inject/inject-core/src/main/scala/com/twitter/inject", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/candidate_pipeline", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/offline_aggregates", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/real_time_aggregates", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/filter", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/scorer", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/selector", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/side_effect", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/model", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/model/request", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/module", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/param", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/product/following/model", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/candidate_pipeline", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/feature_hydrator", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/marshaller", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/model", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/param", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/scorer", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/scoring_pipeline", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/side_effect", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/service", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/util", + "home-mixer/thrift/src/main/thrift:thrift-scala", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/feature_hydrator/query/async", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/feature_hydrator/query/impressed_tweets", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/filter", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/gate", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/premarshaller/urt", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/scorer/tweet_tlx", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/selector", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/side_effect", + "product-mixer/core/src/main/java/com/twitter/product_mixer/core/product/guice/scope", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/feature/featuremap/datarecord", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/configapi", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/decorator", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/request", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/common/presentation", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/marshalling/request", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/candidate", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/product", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/recommendation", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/scoring", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/product", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/product/guice", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/quality_factor", + ], +) diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/ScoredTweetsProductPipelineConfig.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/ScoredTweetsProductPipelineConfig.scala new file mode 100644 index 0000000000..9870030b80 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/ScoredTweetsProductPipelineConfig.scala @@ -0,0 +1,72 @@ +package com.twitter.home_mixer.product.scored_tweets + +import com.twitter.home_mixer.model.HomeFeatures.ServedTweetIdsFeature +import com.twitter.home_mixer.model.request.HomeMixerRequest +import com.twitter.home_mixer.model.request.ScoredTweetsProduct +import com.twitter.home_mixer.model.request.ScoredTweetsProductContext +import com.twitter.home_mixer.product.scored_tweets.model.ScoredTweetsQuery +import com.twitter.home_mixer.product.scored_tweets.param.ScoredTweetsParam.ServerMaxResultsParam +import com.twitter.home_mixer.product.scored_tweets.param.ScoredTweetsParamConfig +import com.twitter.home_mixer.service.HomeMixerAccessPolicy.DefaultHomeMixerAccessPolicy +import com.twitter.home_mixer.{thriftscala => t} +import com.twitter.product_mixer.core.feature.featuremap.FeatureMapBuilder +import com.twitter.product_mixer.core.functional_component.common.access_policy.AccessPolicy +import com.twitter.product_mixer.core.model.common.identifier.ComponentIdentifier +import com.twitter.product_mixer.core.model.common.identifier.ProductPipelineIdentifier +import com.twitter.product_mixer.core.model.marshalling.request.Product +import com.twitter.product_mixer.core.pipeline.PipelineConfig +import com.twitter.product_mixer.core.pipeline.pipeline_failure.BadRequest +import com.twitter.product_mixer.core.pipeline.pipeline_failure.PipelineFailure +import com.twitter.product_mixer.core.pipeline.product.ProductPipelineConfig +import com.twitter.product_mixer.core.product.ProductParamConfig +import com.twitter.timelines.configapi.Params +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class ScoredTweetsProductPipelineConfig @Inject() ( + scoredTweetsRecommendationPipelineConfig: ScoredTweetsRecommendationPipelineConfig, + scoredTweetsParamConfig: ScoredTweetsParamConfig) + extends ProductPipelineConfig[HomeMixerRequest, ScoredTweetsQuery, t.ScoredTweets] { + + override val identifier: ProductPipelineIdentifier = ProductPipelineIdentifier("ScoredTweets") + + override val product: Product = ScoredTweetsProduct + + override val paramConfig: ProductParamConfig = scoredTweetsParamConfig + + override def pipelineQueryTransformer( + request: HomeMixerRequest, + params: Params + ): ScoredTweetsQuery = { + val context = request.productContext match { + case Some(context: ScoredTweetsProductContext) => context + case _ => throw PipelineFailure(BadRequest, "ScoredTweetsProductContext not found") + } + + val featureMap = context.servedTweetIds.map { servedTweets => + FeatureMapBuilder() + .add(ServedTweetIdsFeature, servedTweets) + .build() + } + + ScoredTweetsQuery( + params = params, + clientContext = request.clientContext, + features = featureMap, + pipelineCursor = None, + requestedMaxResults = Some(params(ServerMaxResultsParam)), + debugOptions = request.debugParams.flatMap(_.debugOptions), + deviceContext = context.deviceContext, + seenTweetIds = context.seenTweetIds, + qualityFactorStatus = None + ) + } + + override val pipelines: Seq[PipelineConfig] = Seq(scoredTweetsRecommendationPipelineConfig) + + override def pipelineSelector(query: ScoredTweetsQuery): ComponentIdentifier = + scoredTweetsRecommendationPipelineConfig.identifier + + override val debugAccessPolicies: Set[AccessPolicy] = DefaultHomeMixerAccessPolicy +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/ScoredTweetsRecommendationPipelineConfig.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/ScoredTweetsRecommendationPipelineConfig.scala new file mode 100644 index 0000000000..8d64a29763 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/ScoredTweetsRecommendationPipelineConfig.scala @@ -0,0 +1,254 @@ +package com.twitter.home_mixer.product.scored_tweets + +import com.twitter.conversions.DurationOps._ +import com.twitter.home_mixer.functional_component.feature_hydrator.LastNonPollingTimeQueryFeatureHydrator +import com.twitter.home_mixer.functional_component.feature_hydrator.RealGraphInNetworkScoresQueryFeatureHydrator +import com.twitter.home_mixer.functional_component.feature_hydrator.RealGraphQueryFeatureHydrator +import com.twitter.home_mixer.functional_component.feature_hydrator.RealTimeInteractionGraphUserVertexQueryFeatureHydrator +import com.twitter.home_mixer.functional_component.feature_hydrator.RequestQueryFeatureHydrator +import com.twitter.home_mixer.functional_component.feature_hydrator.TweetImpressionsQueryFeatureHydrator +import com.twitter.home_mixer.functional_component.feature_hydrator.TwhinUserEngagementQueryFeatureHydrator +import com.twitter.home_mixer.functional_component.feature_hydrator.TwhinUserFollowQueryFeatureHydrator +import com.twitter.home_mixer.functional_component.feature_hydrator.UserLanguagesFeatureHydrator +import com.twitter.home_mixer.functional_component.feature_hydrator.UserStateQueryFeatureHydrator +import com.twitter.home_mixer.functional_component.feature_hydrator.offline_aggregates.PartAAggregateQueryFeatureHydrator +import com.twitter.home_mixer.functional_component.feature_hydrator.offline_aggregates.PartBAggregateQueryFeatureHydrator +import com.twitter.home_mixer.functional_component.feature_hydrator.real_time_aggregates.UserEngagementRealTimeAggregatesFeatureHydrator +import com.twitter.home_mixer.functional_component.filter.KeepBestOutOfNetworkTweetPerAuthorFilter +import com.twitter.home_mixer.functional_component.filter.OutOfNetworkCompetitorFilter +import com.twitter.home_mixer.functional_component.filter.OutOfNetworkCompetitorURLFilter +import com.twitter.home_mixer.functional_component.filter.PreviouslySeenTweetsFilter +import com.twitter.home_mixer.functional_component.filter.PreviouslyServedTweetsFilter +import com.twitter.home_mixer.functional_component.filter.RejectTweetFromViewerFilter +import com.twitter.home_mixer.functional_component.filter.RetweetDeduplicationFilter +import com.twitter.home_mixer.functional_component.side_effect.PublishClientSentImpressionsEventBusSideEffect +import com.twitter.home_mixer.functional_component.side_effect.PublishClientSentImpressionsManhattanSideEffect +import com.twitter.home_mixer.functional_component.side_effect.UpdateLastNonPollingTimeSideEffect +import com.twitter.home_mixer.model.HomeFeatures.ScoreFeature +import com.twitter.home_mixer.param.HomeMixerFlagName.TargetFetchLatency +import com.twitter.home_mixer.param.HomeMixerFlagName.TargetScoringLatency +import com.twitter.home_mixer.product.scored_tweets.candidate_pipeline.CachedScoredTweetsCandidatePipelineConfig +import com.twitter.home_mixer.product.scored_tweets.candidate_pipeline.ScoredTweetsCrMixerCandidatePipelineConfig +import com.twitter.home_mixer.product.scored_tweets.candidate_pipeline.ScoredTweetsFrsCandidatePipelineConfig +import com.twitter.home_mixer.product.scored_tweets.candidate_pipeline.ScoredTweetsInNetworkCandidatePipelineConfig +import com.twitter.home_mixer.product.scored_tweets.candidate_pipeline.ScoredTweetsUtegCandidatePipelineConfig +import com.twitter.home_mixer.product.scored_tweets.feature_hydrator.CachedScoredTweetsQueryFeatureHydrator +import com.twitter.home_mixer.product.scored_tweets.marshaller.ScoredTweetsResponseDomainMarshaller +import com.twitter.home_mixer.product.scored_tweets.marshaller.ScoredTweetsResponseTransportMarshaller +import com.twitter.home_mixer.product.scored_tweets.model.ScoredTweetsQuery +import com.twitter.home_mixer.product.scored_tweets.model.ScoredTweetsResponse +import com.twitter.home_mixer.product.scored_tweets.param.ScoredTweetsParam.ServerMaxResultsParam +import com.twitter.home_mixer.product.scored_tweets.scoring_pipeline.ScoredTweetsDiversityScoringPipelineConfig +import com.twitter.home_mixer.product.scored_tweets.scoring_pipeline.ScoredTweetsRescoreOONScoringPipelineConfig +import com.twitter.home_mixer.product.scored_tweets.scoring_pipeline.ScoredTweetsRescoreVerifiedAuthorScoringPipelineConfig +import com.twitter.home_mixer.product.scored_tweets.scoring_pipeline.ScoredTweetsScoringPipelineConfig +import com.twitter.home_mixer.product.scored_tweets.scoring_pipeline.ScoredTweetsWeightedScoresSumScoringPipelineConfig +import com.twitter.home_mixer.product.scored_tweets.side_effect.CachedScoredTweetsSideEffect +import com.twitter.home_mixer.product.scored_tweets.side_effect.ScribeServedCommonFeaturesAndCandidateFeaturesSideEffect +import com.twitter.home_mixer.{thriftscala => t} +import com.twitter.inject.annotations.Flag +import com.twitter.product_mixer.component_library.feature_hydrator.query.async.AsyncQueryFeatureHydrator +import com.twitter.product_mixer.component_library.feature_hydrator.query.impressed_tweets.ImpressedTweetsQueryFeatureHydrator +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.component_library.selector.DropDuplicateCandidates +import com.twitter.product_mixer.component_library.selector.DropMaxCandidates +import com.twitter.product_mixer.component_library.selector.IdAndClassDuplicationKey +import com.twitter.product_mixer.component_library.selector.InsertAppendResults +import com.twitter.product_mixer.component_library.selector.PickFirstCandidateMerger +import com.twitter.product_mixer.component_library.selector.UpdateSortCandidates +import com.twitter.product_mixer.component_library.selector.sorter.FeatureValueSorter +import com.twitter.product_mixer.core.functional_component.common.AllPipelines +import com.twitter.product_mixer.core.functional_component.feature_hydrator.QueryFeatureHydrator +import com.twitter.product_mixer.core.functional_component.filter.Filter +import com.twitter.product_mixer.core.functional_component.marshaller.TransportMarshaller +import com.twitter.product_mixer.core.functional_component.premarshaller.DomainMarshaller +import com.twitter.product_mixer.core.functional_component.selector.Selector +import com.twitter.product_mixer.core.functional_component.side_effect.PipelineResultSideEffect +import com.twitter.product_mixer.core.model.common.identifier.CandidatePipelineIdentifier +import com.twitter.product_mixer.core.model.common.identifier.ComponentIdentifier +import com.twitter.product_mixer.core.model.common.identifier.RecommendationPipelineIdentifier +import com.twitter.product_mixer.core.model.common.identifier.ScoringPipelineIdentifier +import com.twitter.product_mixer.core.pipeline.FailOpenPolicy +import com.twitter.product_mixer.core.pipeline.candidate.CandidatePipelineConfig +import com.twitter.product_mixer.core.pipeline.recommendation.RecommendationPipelineConfig +import com.twitter.product_mixer.core.pipeline.scoring.ScoringPipelineConfig +import com.twitter.product_mixer.core.quality_factor.BoundsWithDefault +import com.twitter.product_mixer.core.quality_factor.LinearLatencyQualityFactorConfig +import com.twitter.product_mixer.core.quality_factor.QualityFactorConfig +import com.twitter.util.Duration +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class ScoredTweetsRecommendationPipelineConfig @Inject() ( + scoredTweetsInNetworkCandidatePipelineConfig: ScoredTweetsInNetworkCandidatePipelineConfig, + scoredTweetsUtegCandidatePipelineConfig: ScoredTweetsUtegCandidatePipelineConfig, + scoredTweetsCrMixerCandidatePipelineConfig: ScoredTweetsCrMixerCandidatePipelineConfig, + scoredTweetsFrsCandidatePipelineConfig: ScoredTweetsFrsCandidatePipelineConfig, + cachedScoredTweetsCandidatePipelineConfig: CachedScoredTweetsCandidatePipelineConfig, + requestQueryFeatureHydrator: RequestQueryFeatureHydrator[ScoredTweetsQuery], + lastNonPollingTimeQueryFeatureHydrator: LastNonPollingTimeQueryFeatureHydrator, + realTimeInteractionGraphUserVertexQueryFeatureHydrator: RealTimeInteractionGraphUserVertexQueryFeatureHydrator, + userStateQueryFeatureHydrator: UserStateQueryFeatureHydrator, + userEngagementRealTimeAggregatesFeatureHydrator: UserEngagementRealTimeAggregatesFeatureHydrator, + twhinUserEngagementQueryFeatureHydrator: TwhinUserEngagementQueryFeatureHydrator, + twhinUserFollowQueryFeatureHydrator: TwhinUserFollowQueryFeatureHydrator, + cachedScoredTweetsQueryFeatureHydrator: CachedScoredTweetsQueryFeatureHydrator, + scoredTweetsScoringPipelineConfig: ScoredTweetsScoringPipelineConfig, + scoredTweetsWeightedScoresSumScoringPipelineConfig: ScoredTweetsWeightedScoresSumScoringPipelineConfig, + manhattanTweetImpressionsQueryFeatureHydrator: TweetImpressionsQueryFeatureHydrator[ + ScoredTweetsQuery + ], + memcacheTweetImpressionsQueryFeatureHydrator: ImpressedTweetsQueryFeatureHydrator, + publishClientSentImpressionsEventBusSideEffect: PublishClientSentImpressionsEventBusSideEffect, + publishClientSentImpressionsManhattanSideEffect: PublishClientSentImpressionsManhattanSideEffect, + realGraphInNetworkScoresQueryFeatureHydrator: RealGraphInNetworkScoresQueryFeatureHydrator, + realGraphQueryFeatureHydrator: RealGraphQueryFeatureHydrator, + userLanguagesFeatureHydrator: UserLanguagesFeatureHydrator, + partAAggregateQueryFeatureHydrator: PartAAggregateQueryFeatureHydrator, + partBAggregateQueryFeatureHydrator: PartBAggregateQueryFeatureHydrator, + cachedScoredTweetsSideEffect: CachedScoredTweetsSideEffect, + scribeServedCommonFeaturesAndCandidateFeaturesSideEffect: ScribeServedCommonFeaturesAndCandidateFeaturesSideEffect, + updateLastNonPollingTimeSideEffect: UpdateLastNonPollingTimeSideEffect[ + ScoredTweetsQuery, + ScoredTweetsResponse + ], + @Flag(TargetFetchLatency) targetFetchLatency: Duration, + @Flag(TargetScoringLatency) targetScoringLatency: Duration) + extends RecommendationPipelineConfig[ + ScoredTweetsQuery, + TweetCandidate, + ScoredTweetsResponse, + t.ScoredTweetsResponse + ] { + + override val identifier: RecommendationPipelineIdentifier = + RecommendationPipelineIdentifier("ScoredTweets") + + private val scoringStep = RecommendationPipelineConfig.scoringPipelinesStep + + override val fetchQueryFeatures: Seq[QueryFeatureHydrator[ScoredTweetsQuery]] = Seq( + requestQueryFeatureHydrator, + realGraphInNetworkScoresQueryFeatureHydrator, + cachedScoredTweetsQueryFeatureHydrator, + manhattanTweetImpressionsQueryFeatureHydrator, + memcacheTweetImpressionsQueryFeatureHydrator, + AsyncQueryFeatureHydrator(scoringStep, realGraphQueryFeatureHydrator), + AsyncQueryFeatureHydrator(scoringStep, lastNonPollingTimeQueryFeatureHydrator), + AsyncQueryFeatureHydrator(scoringStep, userStateQueryFeatureHydrator), + AsyncQueryFeatureHydrator(scoringStep, userLanguagesFeatureHydrator), + AsyncQueryFeatureHydrator(scoringStep, userEngagementRealTimeAggregatesFeatureHydrator), + AsyncQueryFeatureHydrator(scoringStep, realTimeInteractionGraphUserVertexQueryFeatureHydrator), + AsyncQueryFeatureHydrator(scoringStep, twhinUserFollowQueryFeatureHydrator), + AsyncQueryFeatureHydrator(scoringStep, twhinUserEngagementQueryFeatureHydrator), + AsyncQueryFeatureHydrator(scoringStep, partAAggregateQueryFeatureHydrator), + AsyncQueryFeatureHydrator(scoringStep, partBAggregateQueryFeatureHydrator), + ) + + override val candidatePipelines: Seq[ + CandidatePipelineConfig[ScoredTweetsQuery, _, _, TweetCandidate] + ] = Seq( + cachedScoredTweetsCandidatePipelineConfig, + scoredTweetsInNetworkCandidatePipelineConfig, + scoredTweetsUtegCandidatePipelineConfig, + scoredTweetsCrMixerCandidatePipelineConfig, + scoredTweetsFrsCandidatePipelineConfig + ) + + override val postCandidatePipelinesSelectors: Seq[Selector[ScoredTweetsQuery]] = Seq( + DropDuplicateCandidates( + pipelineScope = AllPipelines, + duplicationKey = IdAndClassDuplicationKey, + mergeStrategy = PickFirstCandidateMerger + ), + InsertAppendResults(AllPipelines) + ) + + override val globalFilters: Seq[Filter[ScoredTweetsQuery, TweetCandidate]] = Seq( + // sort these to have the "cheaper" filters run first + RejectTweetFromViewerFilter, + RetweetDeduplicationFilter, + PreviouslyServedTweetsFilter, + PreviouslySeenTweetsFilter, + OutOfNetworkCompetitorFilter + ) + + override val candidatePipelineFailOpenPolicies: Map[CandidatePipelineIdentifier, FailOpenPolicy] = + Map( + cachedScoredTweetsCandidatePipelineConfig.identifier -> FailOpenPolicy.Always, + scoredTweetsInNetworkCandidatePipelineConfig.identifier -> FailOpenPolicy.Always, + scoredTweetsUtegCandidatePipelineConfig.identifier -> FailOpenPolicy.Always, + scoredTweetsCrMixerCandidatePipelineConfig.identifier -> FailOpenPolicy.Always, + scoredTweetsFrsCandidatePipelineConfig.identifier -> FailOpenPolicy.Always + ) + + override val scoringPipelineFailOpenPolicies: Map[ScoringPipelineIdentifier, FailOpenPolicy] = + Map( + ScoredTweetsRescoreOONScoringPipelineConfig.identifier -> FailOpenPolicy.Always, + ScoredTweetsRescoreVerifiedAuthorScoringPipelineConfig.identifier -> FailOpenPolicy.Always, + ScoredTweetsDiversityScoringPipelineConfig.identifier -> FailOpenPolicy.Always + ) + + private val candidatePipelineQualityFactorConfig = LinearLatencyQualityFactorConfig( + qualityFactorBounds = BoundsWithDefault(minInclusive = 0.1, maxInclusive = 1.0, default = 0.4), + initialDelay = 60.seconds, + targetLatency = targetFetchLatency, + targetLatencyPercentile = 95.0, + delta = 0.00125 + ) + + private val scoringPipelineQualityFactorConfig = + candidatePipelineQualityFactorConfig.copy(targetLatency = targetScoringLatency) + + override val qualityFactorConfigs: Map[ComponentIdentifier, QualityFactorConfig] = Map( + // candidate pipelines + scoredTweetsInNetworkCandidatePipelineConfig.identifier -> candidatePipelineQualityFactorConfig, + scoredTweetsUtegCandidatePipelineConfig.identifier -> candidatePipelineQualityFactorConfig, + scoredTweetsCrMixerCandidatePipelineConfig.identifier -> candidatePipelineQualityFactorConfig, + scoredTweetsFrsCandidatePipelineConfig.identifier -> candidatePipelineQualityFactorConfig, + // scoring pipelines + scoredTweetsScoringPipelineConfig.identifier -> scoringPipelineQualityFactorConfig, + ) + + override val scoringPipelines: Seq[ScoringPipelineConfig[ScoredTweetsQuery, TweetCandidate]] = + Seq( + // scoring pipielines - run on non-cached candidates only since cached ones are already scored + scoredTweetsScoringPipelineConfig, + scoredTweetsWeightedScoresSumScoringPipelineConfig, + // re-scoring pipielines - run on all candidates since these are request specific + ScoredTweetsRescoreOONScoringPipelineConfig, + ScoredTweetsRescoreVerifiedAuthorScoringPipelineConfig, + ScoredTweetsDiversityScoringPipelineConfig + ) + + override val resultSelectors: Seq[Selector[ScoredTweetsQuery]] = Seq( + UpdateSortCandidates(AllPipelines, FeatureValueSorter.descending(ScoreFeature)), + DropMaxCandidates(AllPipelines, ServerMaxResultsParam), + InsertAppendResults(AllPipelines) + ) + + override val postSelectionFilters = Seq( + OutOfNetworkCompetitorURLFilter, + KeepBestOutOfNetworkTweetPerAuthorFilter, + ) + + override val resultSideEffects: Seq[ + PipelineResultSideEffect[ScoredTweetsQuery, ScoredTweetsResponse] + ] = Seq( + cachedScoredTweetsSideEffect, + scribeServedCommonFeaturesAndCandidateFeaturesSideEffect, + publishClientSentImpressionsEventBusSideEffect, + publishClientSentImpressionsManhattanSideEffect, + updateLastNonPollingTimeSideEffect + ) + + override val domainMarshaller: DomainMarshaller[ + ScoredTweetsQuery, + ScoredTweetsResponse + ] = ScoredTweetsResponseDomainMarshaller + + override val transportMarshaller: TransportMarshaller[ + ScoredTweetsResponse, + t.ScoredTweetsResponse + ] = ScoredTweetsResponseTransportMarshaller +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/candidate_pipeline/BUILD.bazel b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/candidate_pipeline/BUILD.bazel new file mode 100644 index 0000000000..c5cec9df1f --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/candidate_pipeline/BUILD.bazel @@ -0,0 +1,41 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + strict_deps = True, + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/javax/inject:javax.inject", + "cr-mixer/thrift/src/main/thrift:thrift-scala", + "finatra/inject/inject-core/src/main/scala", + "finatra/inject/inject-core/src/main/scala/com/twitter/inject", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/filter", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/gate", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/model", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/model/request", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/candidate_source", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/feature_hydrator", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/model", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/param", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/query_feature_hydrator", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/query_transformer", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/response_transformer", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/util", + "home-mixer/thrift/src/main/thrift:thrift-scala", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/candidate_source/cr_mixer", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/candidate_source/timeline_ranker", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/filter", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/gate", + "product-mixer/core/src/main/java/com/twitter/product_mixer/core/product/guice/scope", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/request", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/candidate", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/product", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/product/guice", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/quality_factor", + "src/thrift/com/twitter/timelineranker:thrift-scala", + ], + exports = [ + "cr-mixer/thrift/src/main/thrift:thrift-scala", + "src/thrift/com/twitter/timelineranker:thrift-scala", + ], +) diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/candidate_pipeline/CachedScoredTweetsCandidatePipelineConfig.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/candidate_pipeline/CachedScoredTweetsCandidatePipelineConfig.scala new file mode 100644 index 0000000000..95cc79aad3 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/candidate_pipeline/CachedScoredTweetsCandidatePipelineConfig.scala @@ -0,0 +1,53 @@ +package com.twitter.home_mixer.product.scored_tweets.candidate_pipeline + +import com.twitter.home_mixer.product.scored_tweets.candidate_pipeline.CachedScoredTweetsCandidatePipelineConfig._ +import com.twitter.home_mixer.product.scored_tweets.candidate_source.CachedScoredTweetsCandidateSource +import com.twitter.home_mixer.product.scored_tweets.model.ScoredTweetsQuery +import com.twitter.home_mixer.product.scored_tweets.response_transformer.CachedScoredTweetsResponseFeatureTransformer +import com.twitter.home_mixer.{thriftscala => hmt} +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.functional_component.candidate_source.BaseCandidateSource +import com.twitter.product_mixer.core.functional_component.transformer.CandidateFeatureTransformer +import com.twitter.product_mixer.core.functional_component.transformer.CandidatePipelineQueryTransformer +import com.twitter.product_mixer.core.functional_component.transformer.CandidatePipelineResultsTransformer +import com.twitter.product_mixer.core.model.common.identifier.CandidatePipelineIdentifier +import com.twitter.product_mixer.core.pipeline.candidate.CandidatePipelineConfig +import javax.inject.Inject +import javax.inject.Singleton + +/** + * Candidate Pipeline Config that fetches tweets from Scored Tweets Cache. + */ +@Singleton +class CachedScoredTweetsCandidatePipelineConfig @Inject() ( + cachedScoredTweetsCandidateSource: CachedScoredTweetsCandidateSource) + extends CandidatePipelineConfig[ + ScoredTweetsQuery, + ScoredTweetsQuery, + hmt.CachedScoredTweet, + TweetCandidate + ] { + + override val identifier: CandidatePipelineIdentifier = Identifier + + override val queryTransformer: CandidatePipelineQueryTransformer[ + ScoredTweetsQuery, + ScoredTweetsQuery + ] = identity + + override val candidateSource: BaseCandidateSource[ScoredTweetsQuery, hmt.CachedScoredTweet] = + cachedScoredTweetsCandidateSource + + override val featuresFromCandidateSourceTransformers: Seq[ + CandidateFeatureTransformer[hmt.CachedScoredTweet] + ] = Seq(CachedScoredTweetsResponseFeatureTransformer) + + override val resultTransformer: CandidatePipelineResultsTransformer[ + hmt.CachedScoredTweet, + TweetCandidate + ] = { sourceResult => TweetCandidate(id = sourceResult.tweetId) } +} + +object CachedScoredTweetsCandidatePipelineConfig { + val Identifier: CandidatePipelineIdentifier = CandidatePipelineIdentifier("CachedScoredTweets") +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/candidate_pipeline/ScoredTweetsCrMixerCandidatePipelineConfig.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/candidate_pipeline/ScoredTweetsCrMixerCandidatePipelineConfig.scala new file mode 100644 index 0000000000..1ec9353f8e --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/candidate_pipeline/ScoredTweetsCrMixerCandidatePipelineConfig.scala @@ -0,0 +1,98 @@ +package com.twitter.home_mixer.product.scored_tweets.candidate_pipeline + +import com.twitter.cr_mixer.{thriftscala => t} +import com.twitter.home_mixer.functional_component.feature_hydrator.TweetypieStaticEntitiesFeatureHydrator +import com.twitter.home_mixer.functional_component.filter.PredicateFeatureFilter +import com.twitter.home_mixer.functional_component.gate.MinCachedTweetsGate +import com.twitter.home_mixer.model.HomeFeatures.AuthorIdFeature +import com.twitter.home_mixer.product.scored_tweets.model.ScoredTweetsQuery +import com.twitter.home_mixer.product.scored_tweets.param.ScoredTweetsParam.CachedScoredTweets +import com.twitter.home_mixer.product.scored_tweets.param.ScoredTweetsParam.CrMixerSource +import com.twitter.home_mixer.product.scored_tweets.response_transformer.ScoredTweetsCrMixerResponseFeatureTransformer +import com.twitter.home_mixer.util.CachedScoredTweetsHelper +import com.twitter.product_mixer.component_library.candidate_source.cr_mixer.CrMixerTweetRecommendationsCandidateSource +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.functional_component.candidate_source.BaseCandidateSource +import com.twitter.product_mixer.core.functional_component.feature_hydrator.BaseCandidateFeatureHydrator +import com.twitter.product_mixer.core.functional_component.filter.Filter +import com.twitter.product_mixer.core.functional_component.gate.Gate +import com.twitter.product_mixer.core.functional_component.marshaller.request.ClientContextMarshaller +import com.twitter.product_mixer.core.functional_component.transformer.CandidateFeatureTransformer +import com.twitter.product_mixer.core.functional_component.transformer.CandidatePipelineQueryTransformer +import com.twitter.product_mixer.core.functional_component.transformer.CandidatePipelineResultsTransformer +import com.twitter.product_mixer.core.model.common.identifier.CandidatePipelineIdentifier +import com.twitter.product_mixer.core.model.common.identifier.FilterIdentifier +import com.twitter.product_mixer.core.pipeline.candidate.CandidatePipelineConfig +import com.twitter.timelines.configapi.decider.DeciderParam +import javax.inject.Inject +import javax.inject.Singleton + +/** + * Candidate Pipeline Config that fetches tweets from CrMixer. + */ +@Singleton +class ScoredTweetsCrMixerCandidatePipelineConfig @Inject() ( + crMixerTweetRecommendationsCandidateSource: CrMixerTweetRecommendationsCandidateSource, + tweetypieStaticEntitiesFeatureHydrator: TweetypieStaticEntitiesFeatureHydrator) + extends CandidatePipelineConfig[ + ScoredTweetsQuery, + t.CrMixerTweetRequest, + t.TweetRecommendation, + TweetCandidate + ] { + + override val identifier: CandidatePipelineIdentifier = + CandidatePipelineIdentifier("ScoredTweetsCrMixer") + + val HasAuthorFilterId = "HasAuthor" + + override val enabledDeciderParam: Option[DeciderParam[Boolean]] = + Some(CrMixerSource.EnableCandidatePipelineParam) + + override val gates: Seq[Gate[ScoredTweetsQuery]] = Seq( + MinCachedTweetsGate(identifier, CachedScoredTweets.MinCachedTweetsParam) + ) + + override val candidateSource: BaseCandidateSource[t.CrMixerTweetRequest, t.TweetRecommendation] = + crMixerTweetRecommendationsCandidateSource + + private val MaxTweetsToFetch = 500 + + override val queryTransformer: CandidatePipelineQueryTransformer[ + ScoredTweetsQuery, + t.CrMixerTweetRequest + ] = { query => + val maxCount = (query.getQualityFactorCurrentValue(identifier) * MaxTweetsToFetch).toInt + + val excludedTweetIds = query.features.map( + CachedScoredTweetsHelper.tweetImpressionsAndCachedScoredTweets(_, identifier)) + + t.CrMixerTweetRequest( + clientContext = ClientContextMarshaller(query.clientContext), + product = t.Product.Home, + productContext = + Some(t.ProductContext.HomeContext(t.HomeContext(maxResults = Some(maxCount)))), + excludedTweetIds = excludedTweetIds + ) + } + + override val preFilterFeatureHydrationPhase1: Seq[ + BaseCandidateFeatureHydrator[ScoredTweetsQuery, TweetCandidate, _] + ] = Seq(tweetypieStaticEntitiesFeatureHydrator) + + override val featuresFromCandidateSourceTransformers: Seq[ + CandidateFeatureTransformer[t.TweetRecommendation] + ] = Seq(ScoredTweetsCrMixerResponseFeatureTransformer) + + override val filters: Seq[Filter[ScoredTweetsQuery, TweetCandidate]] = Seq( + PredicateFeatureFilter.fromPredicate( + FilterIdentifier(HasAuthorFilterId), + shouldKeepCandidate = _.getOrElse(AuthorIdFeature, None).isDefined + ) + ) + + override val resultTransformer: CandidatePipelineResultsTransformer[ + t.TweetRecommendation, + TweetCandidate + ] = { sourceResult => TweetCandidate(id = sourceResult.tweetId) } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/candidate_pipeline/ScoredTweetsFrsCandidatePipelineConfig.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/candidate_pipeline/ScoredTweetsFrsCandidatePipelineConfig.scala new file mode 100644 index 0000000000..018eaa6886 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/candidate_pipeline/ScoredTweetsFrsCandidatePipelineConfig.scala @@ -0,0 +1,67 @@ +package com.twitter.home_mixer.product.scored_tweets.candidate_pipeline + +import com.twitter.home_mixer.functional_component.gate.MinCachedTweetsGate +import com.twitter.home_mixer.product.scored_tweets.model.ScoredTweetsQuery +import com.twitter.home_mixer.product.scored_tweets.param.ScoredTweetsParam.CachedScoredTweets +import com.twitter.home_mixer.product.scored_tweets.query_feature_hydrator.FrsSeedUsersQueryFeatureHydrator +import com.twitter.home_mixer.product.scored_tweets.query_transformer.TimelineRankerFrsQueryTransformer +import com.twitter.home_mixer.product.scored_tweets.response_transformer.ScoredTweetsFrsResponseFeatureTransformer +import com.twitter.product_mixer.component_library.candidate_source.timeline_ranker.TimelineRankerRecapCandidateSource +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.functional_component.candidate_source.BaseCandidateSource +import com.twitter.product_mixer.core.functional_component.feature_hydrator.BaseQueryFeatureHydrator +import com.twitter.product_mixer.core.functional_component.gate.Gate +import com.twitter.product_mixer.core.functional_component.transformer.CandidateFeatureTransformer +import com.twitter.product_mixer.core.functional_component.transformer.CandidatePipelineQueryTransformer +import com.twitter.product_mixer.core.functional_component.transformer.CandidatePipelineResultsTransformer +import com.twitter.product_mixer.core.model.common.identifier.CandidatePipelineIdentifier +import com.twitter.product_mixer.core.pipeline.candidate.CandidatePipelineConfig +import com.twitter.timelineranker.{thriftscala => tlr} +import javax.inject.Inject +import javax.inject.Singleton + +/** + * Candidate Pipeline Config that takes user recommendations from Follow Recommendation Service (FRS) + * and makes a TimelineRanker->Earlybird query for tweet candidates from those users. + * Additionally, the candidate pipeline hydrates followedByUserIds so that followed-by social proof + * can be used. + */ +@Singleton +class ScoredTweetsFrsCandidatePipelineConfig @Inject() ( + timelineRankerRecapCandidateSource: TimelineRankerRecapCandidateSource, + frsSeedUsersQueryFeatureHydrator: FrsSeedUsersQueryFeatureHydrator) + extends CandidatePipelineConfig[ + ScoredTweetsQuery, + tlr.RecapQuery, + tlr.CandidateTweet, + TweetCandidate + ] { + + override val identifier: CandidatePipelineIdentifier = + CandidatePipelineIdentifier("ScoredTweetsFrs") + + override val gates: Seq[Gate[ScoredTweetsQuery]] = Seq( + MinCachedTweetsGate(identifier, CachedScoredTweets.MinCachedTweetsParam) + ) + + override val queryFeatureHydration: Seq[ + BaseQueryFeatureHydrator[ScoredTweetsQuery, _] + ] = Seq(frsSeedUsersQueryFeatureHydrator) + + override val candidateSource: BaseCandidateSource[tlr.RecapQuery, tlr.CandidateTweet] = + timelineRankerRecapCandidateSource + + override val queryTransformer: CandidatePipelineQueryTransformer[ + ScoredTweetsQuery, + tlr.RecapQuery + ] = TimelineRankerFrsQueryTransformer(identifier) + + override val featuresFromCandidateSourceTransformers: Seq[ + CandidateFeatureTransformer[tlr.CandidateTweet] + ] = Seq(ScoredTweetsFrsResponseFeatureTransformer) + + override val resultTransformer: CandidatePipelineResultsTransformer[ + tlr.CandidateTweet, + TweetCandidate + ] = { candidate => TweetCandidate(candidate.tweet.get.id) } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/candidate_pipeline/ScoredTweetsInNetworkCandidatePipelineConfig.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/candidate_pipeline/ScoredTweetsInNetworkCandidatePipelineConfig.scala new file mode 100644 index 0000000000..28118238d1 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/candidate_pipeline/ScoredTweetsInNetworkCandidatePipelineConfig.scala @@ -0,0 +1,82 @@ +package com.twitter.home_mixer.product.scored_tweets.candidate_pipeline + +import com.twitter.home_mixer.functional_component.feature_hydrator.ReplyFeatureHydrator +import com.twitter.home_mixer.functional_component.feature_hydrator.RetweetSourceTweetFeatureHydrator +import com.twitter.home_mixer.functional_component.filter.RetweetSourceTweetRemovingFilter +import com.twitter.home_mixer.functional_component.gate.MinCachedTweetsGate +import com.twitter.home_mixer.product.scored_tweets.model.ScoredTweetsQuery +import com.twitter.home_mixer.product.scored_tweets.param.ScoredTweetsParam.CachedScoredTweets +import com.twitter.home_mixer.product.scored_tweets.param.ScoredTweetsParam.InNetworkSource +import com.twitter.home_mixer.product.scored_tweets.query_transformer.TimelineRankerInNetworkQueryTransformer +import com.twitter.home_mixer.product.scored_tweets.response_transformer.ScoredTweetsInNetworkResponseFeatureTransformer +import com.twitter.product_mixer.component_library.candidate_source.timeline_ranker.TimelineRankerInNetworkCandidateSource +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.functional_component.candidate_source.BaseCandidateSource +import com.twitter.product_mixer.core.functional_component.feature_hydrator.BaseCandidateFeatureHydrator +import com.twitter.product_mixer.core.functional_component.filter.Filter +import com.twitter.product_mixer.core.functional_component.gate.Gate +import com.twitter.product_mixer.core.functional_component.transformer.CandidateFeatureTransformer +import com.twitter.product_mixer.core.functional_component.transformer.CandidatePipelineQueryTransformer +import com.twitter.product_mixer.core.functional_component.transformer.CandidatePipelineResultsTransformer +import com.twitter.product_mixer.core.model.common.identifier.CandidatePipelineIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.product_mixer.core.pipeline.candidate.CandidatePipelineConfig +import com.twitter.timelineranker.{thriftscala => t} +import com.twitter.timelines.configapi.decider.DeciderParam + +import javax.inject.Inject +import javax.inject.Singleton + +/** + * Candidate Pipeline Config to fetch in-network tweets from Timeline Ranker's Recycled source + */ +@Singleton +class ScoredTweetsInNetworkCandidatePipelineConfig @Inject() ( + timelineRankerInNetworkCandidateSource: TimelineRankerInNetworkCandidateSource, + replyFeatureHydrator: ReplyFeatureHydrator) + extends CandidatePipelineConfig[ + ScoredTweetsQuery, + t.RecapQuery, + t.CandidateTweet, + TweetCandidate + ] { + + override val identifier: CandidatePipelineIdentifier = + CandidatePipelineIdentifier("ScoredTweetsInNetwork") + + override val enabledDeciderParam: Option[DeciderParam[Boolean]] = + Some(InNetworkSource.EnableCandidatePipelineParam) + + override val gates: Seq[Gate[ScoredTweetsQuery]] = Seq( + MinCachedTweetsGate(identifier, CachedScoredTweets.MinCachedTweetsParam) + ) + + override val candidateSource: BaseCandidateSource[t.RecapQuery, t.CandidateTweet] = + timelineRankerInNetworkCandidateSource + + override val queryTransformer: CandidatePipelineQueryTransformer[ + ScoredTweetsQuery, + t.RecapQuery + ] = TimelineRankerInNetworkQueryTransformer(identifier) + + override val preFilterFeatureHydrationPhase1: Seq[ + BaseCandidateFeatureHydrator[PipelineQuery, TweetCandidate, _] + ] = Seq(RetweetSourceTweetFeatureHydrator) + + override def filters: Seq[Filter[ScoredTweetsQuery, TweetCandidate]] = Seq( + RetweetSourceTweetRemovingFilter + ) + + override val postFilterFeatureHydration: Seq[ + BaseCandidateFeatureHydrator[PipelineQuery, TweetCandidate, _] + ] = Seq(replyFeatureHydrator) + + override val featuresFromCandidateSourceTransformers: Seq[ + CandidateFeatureTransformer[t.CandidateTweet] + ] = Seq(ScoredTweetsInNetworkResponseFeatureTransformer) + + override val resultTransformer: CandidatePipelineResultsTransformer[ + t.CandidateTweet, + TweetCandidate + ] = { sourceResult => TweetCandidate(id = sourceResult.tweet.get.id) } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/candidate_pipeline/ScoredTweetsUtegCandidatePipelineConfig.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/candidate_pipeline/ScoredTweetsUtegCandidatePipelineConfig.scala new file mode 100644 index 0000000000..b58ca784f0 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/candidate_pipeline/ScoredTweetsUtegCandidatePipelineConfig.scala @@ -0,0 +1,63 @@ +package com.twitter.home_mixer.product.scored_tweets.candidate_pipeline + +import com.twitter.home_mixer.functional_component.gate.MinCachedTweetsGate +import com.twitter.home_mixer.product.scored_tweets.model.ScoredTweetsQuery +import com.twitter.home_mixer.product.scored_tweets.param.ScoredTweetsParam.CachedScoredTweets +import com.twitter.home_mixer.product.scored_tweets.param.ScoredTweetsParam.UtegSource +import com.twitter.home_mixer.product.scored_tweets.query_transformer.TimelineRankerUtegQueryTransformer +import com.twitter.home_mixer.product.scored_tweets.response_transformer.ScoredTweetsUtegResponseFeatureTransformer +import com.twitter.product_mixer.component_library.candidate_source.timeline_ranker.TimelineRankerUtegCandidateSource +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.functional_component.candidate_source.BaseCandidateSource +import com.twitter.product_mixer.core.functional_component.gate.Gate +import com.twitter.product_mixer.core.functional_component.transformer.CandidateFeatureTransformer +import com.twitter.product_mixer.core.functional_component.transformer.CandidatePipelineQueryTransformer +import com.twitter.product_mixer.core.functional_component.transformer.CandidatePipelineResultsTransformer +import com.twitter.product_mixer.core.model.common.identifier.CandidatePipelineIdentifier +import com.twitter.product_mixer.core.pipeline.candidate.CandidatePipelineConfig +import com.twitter.timelineranker.{thriftscala => t} +import com.twitter.timelines.configapi.decider.DeciderParam + +import javax.inject.Inject +import javax.inject.Singleton + +/** + * Candidate Pipeline Config that fetches tweets from the Timeline Ranker UTEG Candidate Source + */ +@Singleton +class ScoredTweetsUtegCandidatePipelineConfig @Inject() ( + timelineRankerUtegCandidateSource: TimelineRankerUtegCandidateSource) + extends CandidatePipelineConfig[ + ScoredTweetsQuery, + t.UtegLikedByTweetsQuery, + t.CandidateTweet, + TweetCandidate + ] { + + override val identifier: CandidatePipelineIdentifier = + CandidatePipelineIdentifier("ScoredTweetsUteg") + + override val enabledDeciderParam: Option[DeciderParam[Boolean]] = + Some(UtegSource.EnableCandidatePipelineParam) + + override val gates: Seq[Gate[ScoredTweetsQuery]] = Seq( + MinCachedTweetsGate(identifier, CachedScoredTweets.MinCachedTweetsParam) + ) + + override val candidateSource: BaseCandidateSource[t.UtegLikedByTweetsQuery, t.CandidateTweet] = + timelineRankerUtegCandidateSource + + override val queryTransformer: CandidatePipelineQueryTransformer[ + ScoredTweetsQuery, + t.UtegLikedByTweetsQuery + ] = TimelineRankerUtegQueryTransformer(identifier) + + override val featuresFromCandidateSourceTransformers: Seq[ + CandidateFeatureTransformer[t.CandidateTweet] + ] = Seq(ScoredTweetsUtegResponseFeatureTransformer) + + override val resultTransformer: CandidatePipelineResultsTransformer[ + t.CandidateTweet, + TweetCandidate + ] = { sourceResult => TweetCandidate(id = sourceResult.tweet.get.id) } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/candidate_source/BUILD.bazel b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/candidate_source/BUILD.bazel new file mode 100644 index 0000000000..58fe8621d3 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/candidate_source/BUILD.bazel @@ -0,0 +1,12 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + strict_deps = True, + tags = ["bazel-compatible"], + dependencies = [ + "home-mixer/server/src/main/scala/com/twitter/home_mixer/model", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/util", + "home-mixer/thrift/src/main/thrift:thrift-scala", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/product", + ], +) diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/candidate_source/CachedScoredTweetsCandidateSource.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/candidate_source/CachedScoredTweetsCandidateSource.scala new file mode 100644 index 0000000000..15a522efa6 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/candidate_source/CachedScoredTweetsCandidateSource.scala @@ -0,0 +1,24 @@ +package com.twitter.home_mixer.product.scored_tweets.candidate_source + +import com.twitter.home_mixer.util.CachedScoredTweetsHelper +import com.twitter.home_mixer.{thriftscala => hmt} +import com.twitter.product_mixer.core.functional_component.candidate_source.CandidateSource +import com.twitter.product_mixer.core.model.common.identifier.CandidateSourceIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.stitch.Stitch + +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class CachedScoredTweetsCandidateSource @Inject() () + extends CandidateSource[PipelineQuery, hmt.CachedScoredTweet] { + + override val identifier: CandidateSourceIdentifier = + CandidateSourceIdentifier("CachedScoredTweets") + + override def apply(request: PipelineQuery): Stitch[Seq[hmt.CachedScoredTweet]] = { + Stitch.value( + request.features.map(CachedScoredTweetsHelper.unseenCachedScoredTweets).getOrElse(Seq.empty)) + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/feature_hydrator/BUILD.bazel b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/feature_hydrator/BUILD.bazel new file mode 100644 index 0000000000..504826eed6 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/feature_hydrator/BUILD.bazel @@ -0,0 +1,29 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + strict_deps = True, + tags = ["bazel-compatible"], + dependencies = [ + "home-mixer/server/src/main/scala/com/twitter/home_mixer/model", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/model/request", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/param", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/model", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/param", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/query_transformer", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/util", + "home-mixer/thrift/src/main/thrift:thrift-scala", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/feature_hydrator/query/impressed_tweets", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/pipeline/candidate/flexible_injection_pipeline/transformer", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/feature_hydrator", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/candidate", + "servo/repo/src/main/scala", + "servo/util/src/main/scala", + "src/thrift/com/twitter/timelineranker:thrift-scala", + "stitch/stitch-core", + "timelineranker/common/src/main/scala/com/twitter/timelineranker/model", + "timelines/src/main/scala/com/twitter/timelines/common/model", + "timelines/src/main/scala/com/twitter/timelines/earlybird/common/utils", + "timelines/src/main/scala/com/twitter/timelines/model/candidate", + "timelines/src/main/scala/com/twitter/timelines/model/types", + ], +) diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/feature_hydrator/CachedScoredTweetsQueryFeatureHydrator.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/feature_hydrator/CachedScoredTweetsQueryFeatureHydrator.scala new file mode 100644 index 0000000000..7801a9645a --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/feature_hydrator/CachedScoredTweetsQueryFeatureHydrator.scala @@ -0,0 +1,51 @@ +package com.twitter.home_mixer.product.scored_tweets.feature_hydrator + +import com.twitter.home_mixer.model.HomeFeatures._ +import com.twitter.home_mixer.product.scored_tweets.param.ScoredTweetsParam.CachedScoredTweets +import com.twitter.home_mixer.{thriftscala => hmt} +import com.twitter.product_mixer.core.feature.Feature +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.feature.featuremap.FeatureMapBuilder +import com.twitter.product_mixer.core.functional_component.feature_hydrator.QueryFeatureHydrator +import com.twitter.product_mixer.core.model.common.identifier.FeatureHydratorIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.servo.cache.TtlCache +import com.twitter.stitch.Stitch +import com.twitter.timelines.model.UserId +import com.twitter.util.Return +import com.twitter.util.Throw +import com.twitter.util.Time + +import javax.inject.Inject +import javax.inject.Singleton + +/** + * Fetch scored Tweets from cache and exclude the seen ones + */ +@Singleton +case class CachedScoredTweetsQueryFeatureHydrator @Inject() ( + scoredTweetsCache: TtlCache[UserId, hmt.CachedScoredTweets]) + extends QueryFeatureHydrator[PipelineQuery] { + + override val identifier: FeatureHydratorIdentifier = + FeatureHydratorIdentifier("CachedScoredTweets") + + override val features: Set[Feature[_, _]] = Set(CachedScoredTweetsFeature) + + override def hydrate(query: PipelineQuery): Stitch[FeatureMap] = { + val userId = query.getRequiredUserId + val tweetScoreTtl = query.params(CachedScoredTweets.TTLParam) + + Stitch.callFuture(scoredTweetsCache.get(Seq(userId))).map { keyValueResult => + keyValueResult(userId) match { + case Return(cachedCandidatesOpt) => + val cachedScoredTweets = cachedCandidatesOpt.map(_.tweets).getOrElse(Seq.empty) + val nonExpiredTweets = cachedScoredTweets.filter { tweet => + tweet.lastScoredTimestampMs.exists(Time.fromMilliseconds(_).untilNow < tweetScoreTtl) + } + FeatureMapBuilder().add(CachedScoredTweetsFeature, nonExpiredTweets).build() + case Throw(exception) => throw exception + } + } + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/marshaller/BUILD.bazel b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/marshaller/BUILD.bazel new file mode 100644 index 0000000000..db158c23f3 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/marshaller/BUILD.bazel @@ -0,0 +1,19 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + strict_deps = True, + tags = ["bazel-compatible"], + dependencies = [ + "home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/model", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/model/request", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/model", + "home-mixer/thrift/src/main/thrift:thrift-scala", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/candidate", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/scorer/tweet_tlx", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/feature/featuremap", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/premarshaller", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/common", + ], +) diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/marshaller/ScoredTweetsResponseDomainMarshaller.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/marshaller/ScoredTweetsResponseDomainMarshaller.scala new file mode 100644 index 0000000000..3ba4b201be --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/marshaller/ScoredTweetsResponseDomainMarshaller.scala @@ -0,0 +1,64 @@ +package com.twitter.home_mixer.product.scored_tweets.marshaller + +import com.twitter.home_mixer.model.HomeFeatures._ +import com.twitter.home_mixer.product.scored_tweets.model.ScoredTweet +import com.twitter.home_mixer.product.scored_tweets.model.ScoredTweetsQuery +import com.twitter.home_mixer.product.scored_tweets.model.ScoredTweetsResponse +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.functional_component.marshaller.response.urt.metadata.TopicContextFunctionalityTypeMarshaller +import com.twitter.product_mixer.core.functional_component.premarshaller.DomainMarshaller +import com.twitter.product_mixer.core.model.common.identifier.DomainMarshallerIdentifier +import com.twitter.product_mixer.core.model.common.presentation.CandidateWithDetails +import com.twitter.product_mixer.core.model.common.presentation.ItemCandidateWithDetails +import com.twitter.product_mixer.core.model.common.presentation.ModuleCandidateWithDetails + +/** + * Creates a domain model of the Scored Tweets product response from the set of candidates selected + */ +object ScoredTweetsResponseDomainMarshaller + extends DomainMarshaller[ScoredTweetsQuery, ScoredTweetsResponse] { + + override val identifier: DomainMarshallerIdentifier = + DomainMarshallerIdentifier("ScoredTweetsResponse") + + override def apply( + query: ScoredTweetsQuery, + selections: Seq[CandidateWithDetails] + ): ScoredTweetsResponse = ScoredTweetsResponse( + scoredTweets = selections.collect { + case ItemCandidateWithDetails(candidate: TweetCandidate, _, features) => + Seq(mkScoredTweet(candidate.id, features)) + case ModuleCandidateWithDetails(candidates, _, _) => + candidates.map { candidate => mkScoredTweet(candidate.candidateIdLong, candidate.features) } + }.flatten + ) + + private def mkScoredTweet(tweetId: Long, features: FeatureMap): ScoredTweet = { + val topicFunctionalityType = features + .getOrElse(TopicContextFunctionalityTypeFeature, None) + .map(TopicContextFunctionalityTypeMarshaller(_)) + + ScoredTweet( + tweetId = tweetId, + authorId = features.get(AuthorIdFeature).get, + score = features.get(ScoreFeature), + suggestType = features.get(SuggestTypeFeature).get, + sourceTweetId = features.getOrElse(SourceTweetIdFeature, None), + sourceUserId = features.getOrElse(SourceUserIdFeature, None), + quotedTweetId = features.getOrElse(QuotedTweetIdFeature, None), + quotedUserId = features.getOrElse(QuotedUserIdFeature, None), + inReplyToTweetId = features.getOrElse(InReplyToTweetIdFeature, None), + inReplyToUserId = features.getOrElse(InReplyToUserIdFeature, None), + directedAtUserId = features.getOrElse(DirectedAtUserIdFeature, None), + inNetwork = Some(features.getOrElse(InNetworkFeature, false)), + favoritedByUserIds = Some(features.getOrElse(FavoritedByUserIdsFeature, Seq.empty)), + followedByUserIds = Some(features.getOrElse(FollowedByUserIdsFeature, Seq.empty)), + topicId = features.getOrElse(TopicIdSocialContextFeature, None), + topicFunctionalityType = topicFunctionalityType, + ancestors = Some(features.getOrElse(AncestorsFeature, Seq.empty)), + isReadFromCache = Some(features.getOrElse(IsReadFromCacheFeature, false)), + streamToKafka = Some(features.getOrElse(StreamToKafkaFeature, false)) + ) + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/marshaller/ScoredTweetsResponseTransportMarshaller.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/marshaller/ScoredTweetsResponseTransportMarshaller.scala new file mode 100644 index 0000000000..355f076fe0 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/marshaller/ScoredTweetsResponseTransportMarshaller.scala @@ -0,0 +1,43 @@ +package com.twitter.home_mixer.product.scored_tweets.marshaller + +import com.twitter.home_mixer.product.scored_tweets.model.ScoredTweetsResponse +import com.twitter.home_mixer.{thriftscala => t} +import com.twitter.product_mixer.core.functional_component.marshaller.TransportMarshaller +import com.twitter.product_mixer.core.model.common.identifier.TransportMarshallerIdentifier + +/** + * Marshall the domain model into our transport (Thrift) model. + */ +object ScoredTweetsResponseTransportMarshaller + extends TransportMarshaller[ScoredTweetsResponse, t.ScoredTweetsResponse] { + + override val identifier: TransportMarshallerIdentifier = + TransportMarshallerIdentifier("ScoredTweetsResponse") + + override def apply(input: ScoredTweetsResponse): t.ScoredTweetsResponse = { + val scoredTweets = input.scoredTweets.map { tweet => + t.ScoredTweet( + tweetId = tweet.tweetId, + authorId = tweet.authorId, + score = tweet.score, + suggestType = Some(tweet.suggestType), + sourceTweetId = tweet.sourceTweetId, + sourceUserId = tweet.sourceUserId, + quotedTweetId = tweet.quotedTweetId, + quotedUserId = tweet.quotedUserId, + inReplyToTweetId = tweet.inReplyToTweetId, + inReplyToUserId = tweet.inReplyToUserId, + directedAtUserId = tweet.directedAtUserId, + inNetwork = tweet.inNetwork, + favoritedByUserIds = tweet.favoritedByUserIds, + followedByUserIds = tweet.followedByUserIds, + topicId = tweet.topicId, + topicFunctionalityType = tweet.topicFunctionalityType, + ancestors = tweet.ancestors, + isReadFromCache = tweet.isReadFromCache, + streamToKafka = tweet.streamToKafka + ) + } + t.ScoredTweetsResponse(scoredTweets) + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/model/BUILD.bazel b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/model/BUILD.bazel new file mode 100644 index 0000000000..8fd8325cb0 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/model/BUILD.bazel @@ -0,0 +1,23 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + strict_deps = True, + tags = ["bazel-compatible"], + dependencies = [ + "home-mixer/server/src/main/scala/com/twitter/home_mixer/model", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/model/request", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/cursor", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/query/ads", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/pipeline/candidate/flexible_injection_pipeline", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/quality_factor", + "src/thrift/com/twitter/timelinescorer/common/scoredtweetcandidate:thrift-scala", + "src/thrift/com/twitter/timelineservice/server/internal:thrift-scala", + ], + exports = [ + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/cursor", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/query/ads", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/pipeline/candidate/flexible_injection_pipeline", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline", + ], +) diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/model/ScoredTweetsQuery.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/model/ScoredTweetsQuery.scala new file mode 100644 index 0000000000..a2eb3a4663 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/model/ScoredTweetsQuery.scala @@ -0,0 +1,39 @@ +package com.twitter.home_mixer.product.scored_tweets.model + +import com.twitter.home_mixer.model.request.DeviceContext +import com.twitter.home_mixer.model.request.HasDeviceContext +import com.twitter.home_mixer.model.request.HasSeenTweetIds +import com.twitter.home_mixer.model.request.ScoredTweetsProduct +import com.twitter.product_mixer.component_library.model.cursor.UrtOrderedCursor +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.model.marshalling.request._ +import com.twitter.product_mixer.core.pipeline.HasPipelineCursor +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.product_mixer.core.quality_factor.HasQualityFactorStatus +import com.twitter.product_mixer.core.quality_factor.QualityFactorStatus +import com.twitter.timelines.configapi.Params + +case class ScoredTweetsQuery( + override val params: Params, + override val clientContext: ClientContext, + override val pipelineCursor: Option[UrtOrderedCursor], + override val requestedMaxResults: Option[Int], + override val debugOptions: Option[DebugOptions], + override val features: Option[FeatureMap], + override val deviceContext: Option[DeviceContext], + override val seenTweetIds: Option[Seq[Long]], + override val qualityFactorStatus: Option[QualityFactorStatus]) + extends PipelineQuery + with HasPipelineCursor[UrtOrderedCursor] + with HasDeviceContext + with HasSeenTweetIds + with HasQualityFactorStatus { + override val product: Product = ScoredTweetsProduct + + override def withFeatureMap(features: FeatureMap): ScoredTweetsQuery = + copy(features = Some(features)) + + override def withQualityFactorStatus( + qualityFactorStatus: QualityFactorStatus + ): ScoredTweetsQuery = copy(qualityFactorStatus = Some(qualityFactorStatus)) +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/model/ScoredTweetsResponse.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/model/ScoredTweetsResponse.scala new file mode 100644 index 0000000000..81cdc22113 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/model/ScoredTweetsResponse.scala @@ -0,0 +1,29 @@ +package com.twitter.home_mixer.product.scored_tweets.model + +import com.twitter.product_mixer.core.model.marshalling.HasMarshalling +import com.twitter.timelineservice.suggests.{thriftscala => st} +import com.twitter.tweetconvosvc.tweet_ancestor.{thriftscala => ta} +import com.twitter.timelines.render.{thriftscala => urt} + +case class ScoredTweet( + tweetId: Long, + authorId: Long, + score: Option[Double], + suggestType: st.SuggestType, + sourceTweetId: Option[Long], + sourceUserId: Option[Long], + quotedTweetId: Option[Long], + quotedUserId: Option[Long], + inReplyToTweetId: Option[Long], + inReplyToUserId: Option[Long], + directedAtUserId: Option[Long], + inNetwork: Option[Boolean], + favoritedByUserIds: Option[Seq[Long]], + followedByUserIds: Option[Seq[Long]], + topicId: Option[Long], + topicFunctionalityType: Option[urt.TopicContextFunctionalityType], + ancestors: Option[Seq[ta.TweetAncestor]], + isReadFromCache: Option[Boolean], + streamToKafka: Option[Boolean]) + +case class ScoredTweetsResponse(scoredTweets: Seq[ScoredTweet]) extends HasMarshalling diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/param/BUILD.bazel b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/param/BUILD.bazel new file mode 100644 index 0000000000..d402ecc186 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/param/BUILD.bazel @@ -0,0 +1,15 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + strict_deps = True, + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/javax/inject:javax.inject", + "configapi/configapi-core/src/main/scala/com/twitter/timelines/configapi", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/param/decider", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/decorator/urt", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/product", + "timelinemixer/common/src/main/scala/com/twitter/timelinemixer/clients/timelineranker", + "util/util-core/src/main/scala/com/twitter/conversions", + ], +) diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/param/ScoredTweetsParam.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/param/ScoredTweetsParam.scala new file mode 100644 index 0000000000..261be593a0 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/param/ScoredTweetsParam.scala @@ -0,0 +1,176 @@ +package com.twitter.home_mixer.product.scored_tweets.param + +import com.twitter.conversions.DurationOps._ +import com.twitter.home_mixer.param.decider.DeciderKey +import com.twitter.timelines.configapi.DurationConversion +import com.twitter.timelines.configapi.FSBoundedParam +import com.twitter.timelines.configapi.FSParam +import com.twitter.timelines.configapi.HasDurationConversion +import com.twitter.timelines.configapi.decider.BooleanDeciderParam +import com.twitter.util.Duration + +object ScoredTweetsParam { + val SupportedClientFSName = "scored_tweets_supported_client" + + object CrMixerSource { + object EnableCandidatePipelineParam + extends BooleanDeciderParam(DeciderKey.EnableScoredTweetsCrMixerCandidatePipeline) + } + + object FrsTweetSource { + object EnableCandidatePipelineParam + extends BooleanDeciderParam(DeciderKey.EnableScoredTweetsFrsCandidatePipeline) + } + + object InNetworkSource { + object EnableCandidatePipelineParam + extends BooleanDeciderParam(DeciderKey.EnableScoredTweetsInNetworkCandidatePipeline) + } + + object QualityFactor { + object MaxTweetsToScoreParam + extends FSBoundedParam[Int]( + name = "scored_tweets_quality_factor_max_tweets_to_score", + default = 1100, + min = 0, + max = 10000 + ) + + object CrMixerMaxTweetsToScoreParam + extends FSBoundedParam[Int]( + name = "scored_tweets_quality_factor_cr_mixer_max_tweets_to_score", + default = 500, + min = 0, + max = 10000 + ) + } + object ServerMaxResultsParam + extends FSBoundedParam[Int]( + name = "scored_tweets_server_max_results", + default = 120, + min = 1, + max = 500 + ) + object UtegSource { + object EnableCandidatePipelineParam + extends BooleanDeciderParam(DeciderKey.EnableScoredTweetsUtegCandidatePipeline) + } + + object CachedScoredTweets { + object TTLParam + extends FSBoundedParam[Duration]( + name = "scored_tweets_cached_scored_tweets_ttl_minutes", + default = 3.minutes, + min = 0.minute, + max = 60.minutes + ) + with HasDurationConversion { + override val durationConversion: DurationConversion = DurationConversion.FromMinutes + } + + object MinCachedTweetsParam + extends FSBoundedParam[Int]( + name = "scored_tweets_cached_scored_tweets_min_cached_tweets", + default = 30, + min = 0, + max = 1000 + ) + } + + object Scoring { + object HomeModelParam + extends FSParam[String](name = "scored_tweets_home_model", default = "Home") + + object ModelWeights { + + object FavParam + extends FSBoundedParam[Double]( + name = "scored_tweets_model_weight_fav", + default = 1.0, + min = 0.0, + max = 100.0 + ) + + object RetweetParam + extends FSBoundedParam[Double]( + name = "scored_tweets_model_weight_retweet", + default = 1.0, + min = 0.0, + max = 100.0 + ) + + object ReplyParam + extends FSBoundedParam[Double]( + name = "scored_tweets_model_weight_reply", + default = 1.0, + min = 0.0, + max = 100.0 + ) + + object GoodProfileClickParam + extends FSBoundedParam[Double]( + name = "scored_tweets_model_weight_good_profile_click", + default = 1.0, + min = 0.0, + max = 1000000.0 + ) + + object VideoPlayback50Param + extends FSBoundedParam[Double]( + name = "scored_tweets_model_weight_video_playback50", + default = 1.0, + min = 0.0, + max = 100.0 + ) + + object ReplyEngagedByAuthorParam + extends FSBoundedParam[Double]( + name = "scored_tweets_model_weight_reply_engaged_by_author", + default = 1.0, + min = 0.0, + max = 200.0 + ) + + object GoodClickParam + extends FSBoundedParam[Double]( + name = "scored_tweets_model_weight_good_click", + default = 1.0, + min = 0.0, + max = 1000000.0 + ) + + object GoodClickV2Param + extends FSBoundedParam[Double]( + name = "scored_tweets_model_weight_good_click_v2", + default = 1.0, + min = 0.0, + max = 1000000.0 + ) + + object NegativeFeedbackV2Param + extends FSBoundedParam[Double]( + name = "scored_tweets_model_weight_negative_feedback_v2", + default = 1.0, + min = -1000.0, + max = 0.0 + ) + + object ReportParam + extends FSBoundedParam[Double]( + name = "scored_tweets_model_weight_report", + default = 1.0, + min = -20000.0, + max = 0.0 + ) + } + } + + object EnableSimClustersSimilarityFeatureHydrationDeciderParam + extends BooleanDeciderParam(decider = DeciderKey.EnableSimClustersSimilarityFeatureHydration) + + object CompetitorSetParam + extends FSParam[Set[Long]](name = "scored_tweets_competitor_list", default = Set.empty) + + object CompetitorURLSeqParam + extends FSParam[Seq[String]](name = "scored_tweets_competitor_url_list", default = Seq.empty) +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/param/ScoredTweetsParamConfig.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/param/ScoredTweetsParamConfig.scala new file mode 100644 index 0000000000..ae82db20fd --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/param/ScoredTweetsParamConfig.scala @@ -0,0 +1,59 @@ +package com.twitter.home_mixer.product.scored_tweets.param + +import com.twitter.home_mixer.param.decider.DeciderKey +import com.twitter.home_mixer.product.scored_tweets.param.ScoredTweetsParam._ +import com.twitter.product_mixer.core.product.ProductParamConfig +import com.twitter.servo.decider.DeciderKeyName + +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class ScoredTweetsParamConfig @Inject() () extends ProductParamConfig { + override val enabledDeciderKey: DeciderKeyName = DeciderKey.EnableScoredTweetsProduct + override val supportedClientFSName: String = SupportedClientFSName + + override val booleanDeciderOverrides = Seq( + CrMixerSource.EnableCandidatePipelineParam, + FrsTweetSource.EnableCandidatePipelineParam, + InNetworkSource.EnableCandidatePipelineParam, + UtegSource.EnableCandidatePipelineParam, + ScoredTweetsParam.EnableSimClustersSimilarityFeatureHydrationDeciderParam + ) + + override val boundedIntFSOverrides = Seq( + CachedScoredTweets.MinCachedTweetsParam, + QualityFactor.CrMixerMaxTweetsToScoreParam, + QualityFactor.MaxTweetsToScoreParam, + ServerMaxResultsParam + ) + + override val boundedDurationFSOverrides = Seq( + CachedScoredTweets.TTLParam + ) + + override val stringFSOverrides = Seq( + Scoring.HomeModelParam + ) + + override val boundedDoubleFSOverrides = Seq( + Scoring.ModelWeights.FavParam, + Scoring.ModelWeights.ReplyParam, + Scoring.ModelWeights.RetweetParam, + Scoring.ModelWeights.GoodClickParam, + Scoring.ModelWeights.GoodClickV2Param, + Scoring.ModelWeights.GoodProfileClickParam, + Scoring.ModelWeights.ReplyEngagedByAuthorParam, + Scoring.ModelWeights.VideoPlayback50Param, + Scoring.ModelWeights.ReportParam, + Scoring.ModelWeights.NegativeFeedbackV2Param, + ) + + override val longSetFSOverrides = Seq( + CompetitorSetParam, + ) + + override val stringSeqFSOverrides = Seq( + CompetitorURLSeqParam + ) +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/query_feature_hydrator/BUILD.bazel b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/query_feature_hydrator/BUILD.bazel new file mode 100644 index 0000000000..154f840077 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/query_feature_hydrator/BUILD.bazel @@ -0,0 +1,18 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + strict_deps = True, + dependencies = [ + "follow-recommendations-service/thrift/src/main/thrift:thrift-scala", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/model", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/model/request", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/model", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/candidate_source/recommendations", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/pipeline/candidate/flexible_injection_pipeline/transformer", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/candidate_source/strato", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/product", + "src/thrift/com/twitter/timelineranker:thrift-scala", + "timelinemixer/common/src/main/scala/com/twitter/timelinemixer/clients/timelineranker", + ], +) diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/query_feature_hydrator/FrsSeedUsersQueryFeatureHydrator.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/query_feature_hydrator/FrsSeedUsersQueryFeatureHydrator.scala new file mode 100644 index 0000000000..194146dac8 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/query_feature_hydrator/FrsSeedUsersQueryFeatureHydrator.scala @@ -0,0 +1,64 @@ +package com.twitter.home_mixer.product.scored_tweets.query_feature_hydrator + +import com.twitter.follow_recommendations.{thriftscala => frs} +import com.twitter.home_mixer.product.scored_tweets.model.ScoredTweetsQuery +import com.twitter.product_mixer.component_library.candidate_source.recommendations.UserFollowRecommendationsCandidateSource +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.feature.Feature +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.feature.featuremap.FeatureMapBuilder +import com.twitter.product_mixer.core.functional_component.candidate_source.strato.StratoKeyView +import com.twitter.product_mixer.core.functional_component.feature_hydrator.QueryFeatureHydrator +import com.twitter.product_mixer.core.model.common.identifier.FeatureHydratorIdentifier +import com.twitter.stitch.Stitch +import javax.inject.Inject +import javax.inject.Singleton + +object FrsSeedUserIdsFeature extends Feature[TweetCandidate, Option[Seq[Long]]] +object FrsUserToFollowedByUserIdsFeature extends Feature[TweetCandidate, Map[Long, Seq[Long]]] + +@Singleton +case class FrsSeedUsersQueryFeatureHydrator @Inject() ( + userFollowRecommendationsCandidateSource: UserFollowRecommendationsCandidateSource) + extends QueryFeatureHydrator[ScoredTweetsQuery] { + + private val maxUsersToFetch = 100 + + override val identifier: FeatureHydratorIdentifier = FeatureHydratorIdentifier("FrsSeedUsers") + + override def features: Set[Feature[_, _]] = Set( + FrsSeedUserIdsFeature, + FrsUserToFollowedByUserIdsFeature + ) + + override def hydrate(query: ScoredTweetsQuery): Stitch[FeatureMap] = { + val frsRequest = frs.RecommendationRequest( + clientContext = frs.ClientContext(query.getOptionalUserId), + displayLocation = frs.DisplayLocation.HomeTimelineTweetRecs, + maxResults = Some(maxUsersToFetch) + ) + + userFollowRecommendationsCandidateSource(StratoKeyView(frsRequest, Unit)) + .map { userRecommendations: Seq[frs.UserRecommendation] => + val seedUserIds = userRecommendations.map(_.userId) + val seedUserIdsSet = seedUserIds.toSet + + val userToFollowedByUserIds: Map[Long, Seq[Long]] = userRecommendations.flatMap { + userRecommendation => + if (seedUserIdsSet.contains(userRecommendation.userId)) { + val followProof = + userRecommendation.reason.flatMap(_.accountProof).flatMap(_.followProof) + val followedByUserIds = followProof.map(_.userIds).getOrElse(Seq.empty) + Some(userRecommendation.userId -> followedByUserIds) + } else { + None + } + }.toMap + + FeatureMapBuilder() + .add(FrsSeedUserIdsFeature, Some(seedUserIds)) + .add(FrsUserToFollowedByUserIdsFeature, userToFollowedByUserIds) + .build() + } + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/query_transformer/BUILD.bazel b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/query_transformer/BUILD.bazel new file mode 100644 index 0000000000..a7db4784bb --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/query_transformer/BUILD.bazel @@ -0,0 +1,24 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + strict_deps = True, + dependencies = [ + "home-mixer/server/src/main/scala/com/twitter/home_mixer/model", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/model/request", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/query_feature_hydrator", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/util", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/transformer", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/product", + "snowflake/src/main/scala/com/twitter/snowflake/id", + "src/thrift/com/twitter/timelineranker:thrift-scala", + "timelinemixer/common/src/main/scala/com/twitter/timelinemixer/clients/timelineranker", + "timelineranker/common/src/main/scala/com/twitter/timelineranker/model", + "timelines:util", + "timelines/src/main/scala/com/twitter/timelines/common/model", + "timelines/src/main/scala/com/twitter/timelines/earlybird/common/options", + "timelines/src/main/scala/com/twitter/timelines/earlybird/common/utils", + "timelines/src/main/scala/com/twitter/timelines/model/candidate", + "timelineservice/common:model", + ], +) diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/query_transformer/TimelineRankerFrsQueryTransformer.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/query_transformer/TimelineRankerFrsQueryTransformer.scala new file mode 100644 index 0000000000..e141d1ee7f --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/query_transformer/TimelineRankerFrsQueryTransformer.scala @@ -0,0 +1,43 @@ +package com.twitter.home_mixer.product.scored_tweets.query_transformer + +import com.twitter.conversions.DurationOps._ +import com.twitter.home_mixer.model.request.HasDeviceContext +import com.twitter.home_mixer.product.scored_tweets.query_feature_hydrator.FrsSeedUserIdsFeature +import com.twitter.home_mixer.product.scored_tweets.query_transformer.TimelineRankerFrsQueryTransformer._ +import com.twitter.product_mixer.core.functional_component.transformer.CandidatePipelineQueryTransformer +import com.twitter.product_mixer.core.model.common.identifier.CandidatePipelineIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.product_mixer.core.quality_factor.HasQualityFactorStatus +import com.twitter.timelineranker.{thriftscala => t} +import com.twitter.timelines.common.model.TweetKindOption +import com.twitter.timelines.model.candidate.CandidateTweetSourceId +import com.twitter.util.Duration + +object TimelineRankerFrsQueryTransformer { + private val SinceDuration = 24.hours + private val MaxTweetsToFetch = 100 + + private val tweetKindOptions: TweetKindOption.ValueSet = + TweetKindOption(includeOriginalTweetsAndQuotes = true) +} + +case class TimelineRankerFrsQueryTransformer[ + Query <: PipelineQuery with HasQualityFactorStatus with HasDeviceContext +]( + override val candidatePipelineIdentifier: CandidatePipelineIdentifier, + override val maxTweetsToFetch: Int = MaxTweetsToFetch, + override val sinceDuration: Duration = SinceDuration) + extends CandidatePipelineQueryTransformer[Query, t.RecapQuery] + with TimelineRankerQueryTransformer[Query] { + + override val candidateTweetSourceId = CandidateTweetSourceId.FrsTweet + override val skipVeryRecentTweets = false + override val options = tweetKindOptions + + override def seedAuthorIds(query: Query): Option[Seq[Long]] = { + query.features.flatMap(_.getOrElse(FrsSeedUserIdsFeature, None)) + } + + override def transform(input: Query): t.RecapQuery = + buildTimelineRankerQuery(input).toThriftRecapQuery +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/query_transformer/TimelineRankerInNetworkQueryTransformer.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/query_transformer/TimelineRankerInNetworkQueryTransformer.scala new file mode 100644 index 0000000000..bec8f74f28 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/query_transformer/TimelineRankerInNetworkQueryTransformer.scala @@ -0,0 +1,42 @@ +package com.twitter.home_mixer.product.scored_tweets.query_transformer + +import com.twitter.conversions.DurationOps._ +import com.twitter.home_mixer.model.request.HasDeviceContext +import com.twitter.home_mixer.product.scored_tweets.query_transformer.TimelineRankerInNetworkQueryTransformer._ +import com.twitter.product_mixer.core.functional_component.transformer.CandidatePipelineQueryTransformer +import com.twitter.product_mixer.core.model.common.identifier.CandidatePipelineIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.product_mixer.core.quality_factor.HasQualityFactorStatus +import com.twitter.timelineranker.{thriftscala => t} +import com.twitter.timelines.common.model.TweetKindOption +import com.twitter.timelines.model.candidate.CandidateTweetSourceId +import com.twitter.util.Duration + +object TimelineRankerInNetworkQueryTransformer { + private val SinceDuration = 24.hours + private val MaxTweetsToFetch = 500 + + private val tweetKindOptions: TweetKindOption.ValueSet = TweetKindOption( + includeReplies = true, + includeRetweets = true, + includeOriginalTweetsAndQuotes = true, + includeExtendedReplies = true + ) +} + +case class TimelineRankerInNetworkQueryTransformer[ + Query <: PipelineQuery with HasQualityFactorStatus with HasDeviceContext +]( + override val candidatePipelineIdentifier: CandidatePipelineIdentifier, + override val maxTweetsToFetch: Int = MaxTweetsToFetch, + override val sinceDuration: Duration = SinceDuration) + extends CandidatePipelineQueryTransformer[Query, t.RecapQuery] + with TimelineRankerQueryTransformer[Query] { + + override val candidateTweetSourceId = CandidateTweetSourceId.RecycledTweet + override val skipVeryRecentTweets = false + override val options = tweetKindOptions + + override def transform(input: Query): t.RecapQuery = + buildTimelineRankerQuery(input).toThriftRecapQuery +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/query_transformer/TimelineRankerQueryTransformer.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/query_transformer/TimelineRankerQueryTransformer.scala new file mode 100644 index 0000000000..d5b29974b2 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/query_transformer/TimelineRankerQueryTransformer.scala @@ -0,0 +1,109 @@ +package com.twitter.home_mixer.product.scored_tweets.query_transformer + +import com.twitter.home_mixer.model.HomeFeatures.RealGraphInNetworkScoresFeature +import com.twitter.home_mixer.model.request.HasDeviceContext +import com.twitter.home_mixer.product.scored_tweets.query_transformer.TimelineRankerQueryTransformer._ +import com.twitter.home_mixer.util.CachedScoredTweetsHelper +import com.twitter.product_mixer.core.model.common.identifier.CandidatePipelineIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.product_mixer.core.quality_factor.HasQualityFactorStatus +import com.twitter.timelinemixer.clients.timelineranker.EarlybirdScoringModels +import com.twitter.timelinemixer.clients.timelineranker.EarlybirdScoringModelsId +import com.twitter.timelineranker.{model => tlr} +import com.twitter.timelines.common.model.TweetKindOption +import com.twitter.timelines.earlybird.common.options.EarlybirdOptions +import com.twitter.timelines.earlybird.common.options.EarlybirdScoringModelConfig +import com.twitter.timelines.earlybird.common.utils.SearchOperator +import com.twitter.timelines.model.UserId +import com.twitter.timelines.model.candidate.CandidateTweetSourceId +import com.twitter.timelines.util.SnowflakeSortIndexHelper +import com.twitter.util.Duration +import com.twitter.util.Time + +object TimelineRankerQueryTransformer { + + /** + * Specifies the maximum number of excluded tweet ids to include in the search index query. + * Earlybird's named multi term disjunction map feature supports up to 1500 tweet ids. + */ + private val EarlybirdMaxExcludedTweets = 1500 + + /** + * Maximum number of query hits each earlybird shard is allowed to accumulate before + * early-terminating the query and reducing the hits to MaxNumEarlybirdResults. + */ + private val EarlybirdMaxHits = 1000 + + /** + * Maximum number of results TLR should retrieve from each earlybird shard. + */ + private val EarlybirdMaxResults = 200 +} + +trait TimelineRankerQueryTransformer[ + Query <: PipelineQuery with HasQualityFactorStatus with HasDeviceContext] { + def maxTweetsToFetch: Int + def sinceDuration: Duration + def options: TweetKindOption.ValueSet = TweetKindOption.Default + def candidateTweetSourceId: CandidateTweetSourceId.Value + def skipVeryRecentTweets: Boolean + def utegLikedByTweetsOptions(query: Query): Option[tlr.UtegLikedByTweetsOptions] = None + def seedAuthorIds(query: Query): Option[Seq[Long]] = None + def candidatePipelineIdentifier: CandidatePipelineIdentifier + def earlybirdModels: Seq[EarlybirdScoringModelConfig] = + EarlybirdScoringModels.fromEnum(EarlybirdScoringModelsId.UnifiedEngagementProd) + def tensorflowModel: Option[String] = None + + def buildTimelineRankerQuery(query: Query): tlr.RecapQuery = { + val sinceTime: Time = sinceDuration.ago + val untilTime: Time = Time.now + + val fromTweetIdExclusive = SnowflakeSortIndexHelper.timestampToFakeId(sinceTime) + val toTweetIdExclusive = SnowflakeSortIndexHelper.timestampToFakeId(untilTime) + val range = tlr.TweetIdRange(Some(fromTweetIdExclusive), Some(toTweetIdExclusive)) + + val excludedTweetIds = query.features.map { featureMap => + CachedScoredTweetsHelper.tweetImpressionsAndCachedScoredTweetsInRange( + featureMap, + candidatePipelineIdentifier, + EarlybirdMaxExcludedTweets, + sinceTime, + untilTime) + } + + val maxCount = + (query.getQualityFactorCurrentValue(candidatePipelineIdentifier) * maxTweetsToFetch).toInt + + val authorScoreMap = query.features + .map(_.getOrElse(RealGraphInNetworkScoresFeature, Map.empty[UserId, Double])) + .getOrElse(Map.empty) + + val deviceContext = + query.deviceContext.map(_.toTimelineServiceDeviceContext(query.clientContext)) + + val earlyBirdOptions = EarlybirdOptions( + maxNumHitsPerShard = EarlybirdMaxHits, + maxNumResultsPerShard = EarlybirdMaxResults, + models = earlybirdModels, + authorScoreMap = authorScoreMap, + skipVeryRecentTweets = skipVeryRecentTweets, + tensorflowModel = tensorflowModel + ) + + tlr.RecapQuery( + userId = query.getRequiredUserId, + maxCount = Some(maxCount), + range = Some(range), + options = options, + searchOperator = SearchOperator.Exclude, + earlybirdOptions = Some(earlyBirdOptions), + deviceContext = deviceContext, + authorIds = seedAuthorIds(query), + excludedTweetIds = excludedTweetIds, + utegLikedByTweetsOptions = utegLikedByTweetsOptions(query), + searchClientSubId = None, + candidateTweetSourceId = Some(candidateTweetSourceId), + hydratesContentFeatures = Some(false) + ) + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/query_transformer/TimelineRankerUtegQueryTransformer.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/query_transformer/TimelineRankerUtegQueryTransformer.scala new file mode 100644 index 0000000000..03a63acf90 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/query_transformer/TimelineRankerUtegQueryTransformer.scala @@ -0,0 +1,59 @@ +package com.twitter.home_mixer.product.scored_tweets.query_transformer + +import com.twitter.conversions.DurationOps._ +import com.twitter.home_mixer.model.HomeFeatures.RealGraphInNetworkScoresFeature +import com.twitter.home_mixer.model.request.HasDeviceContext +import com.twitter.home_mixer.product.scored_tweets.query_transformer.TimelineRankerUtegQueryTransformer._ +import com.twitter.product_mixer.core.functional_component.transformer.CandidatePipelineQueryTransformer +import com.twitter.product_mixer.core.model.common.identifier.CandidatePipelineIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.product_mixer.core.quality_factor.HasQualityFactorStatus +import com.twitter.timelinemixer.clients.timelineranker.EarlybirdScoringModels +import com.twitter.timelinemixer.clients.timelineranker.EarlybirdScoringModelsId +import com.twitter.timelineranker.{model => tlr} +import com.twitter.timelineranker.{thriftscala => t} +import com.twitter.timelines.common.model.TweetKindOption +import com.twitter.timelines.model.UserId +import com.twitter.timelines.model.candidate.CandidateTweetSourceId +import com.twitter.util.Duration + +object TimelineRankerUtegQueryTransformer { + private val SinceDuration = 24.hours + private val MaxTweetsToFetch = 500 + private val MaxUtegCandidates = 800 + + private val TensorflowModel = "timelines_rectweet_replica" + + private val tweetKindOptions = TweetKindOption(includeReplies = true) + + def utegEarlybirdModels = + EarlybirdScoringModels.fromEnum(EarlybirdScoringModelsId.UnifiedEngagementRectweet) +} + +case class TimelineRankerUtegQueryTransformer[ + Query <: PipelineQuery with HasQualityFactorStatus with HasDeviceContext +]( + override val candidatePipelineIdentifier: CandidatePipelineIdentifier, + override val maxTweetsToFetch: Int = MaxTweetsToFetch, + override val sinceDuration: Duration = SinceDuration) + extends CandidatePipelineQueryTransformer[Query, t.UtegLikedByTweetsQuery] + with TimelineRankerQueryTransformer[Query] { + + override val candidateTweetSourceId = CandidateTweetSourceId.RecommendedTweet + override val skipVeryRecentTweets = true + override val earlybirdModels = utegEarlybirdModels + override val tensorflowModel = Some(TensorflowModel) + + override def utegLikedByTweetsOptions(input: Query): Option[tlr.UtegLikedByTweetsOptions] = Some( + tlr.UtegLikedByTweetsOptions( + utegCount = MaxUtegCandidates, + isInNetwork = false, + weightedFollowings = input.features + .map(_.getOrElse(RealGraphInNetworkScoresFeature, Map.empty[UserId, Double])) + .getOrElse(Map.empty) + ) + ) + + override def transform(input: Query): t.UtegLikedByTweetsQuery = + buildTimelineRankerQuery(input).toThriftUtegLikedByTweetsQuery +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/response_transformer/BUILD.bazel b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/response_transformer/BUILD.bazel new file mode 100644 index 0000000000..32f150c38c --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/response_transformer/BUILD.bazel @@ -0,0 +1,18 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + strict_deps = True, + dependencies = [ + "cr-mixer/thrift/src/main/thrift:thrift-scala", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/marshaller/timelines", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/model", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/util/tweetypie/content", + "home-mixer/thrift/src/main/thrift:thrift-scala", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/transformer", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/product", + "src/thrift/com/twitter/timelineranker:thrift-scala", + "topic-social-proof/server/src/main/thrift:thrift-scala", + ], +) diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/response_transformer/CachedScoredTweetsResponseFeatureTransformer.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/response_transformer/CachedScoredTweetsResponseFeatureTransformer.scala new file mode 100644 index 0000000000..9822a53316 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/response_transformer/CachedScoredTweetsResponseFeatureTransformer.scala @@ -0,0 +1,94 @@ +package com.twitter.home_mixer.product.scored_tweets.response_transformer + +import com.twitter.home_mixer.marshaller.timelines.TopicContextFunctionalityTypeUnmarshaller +import com.twitter.home_mixer.model.HomeFeatures.AncestorsFeature +import com.twitter.home_mixer.model.HomeFeatures.AuthorIdFeature +import com.twitter.home_mixer.model.HomeFeatures.AuthorIsBlueVerifiedFeature +import com.twitter.home_mixer.model.HomeFeatures.CachedCandidatePipelineIdentifierFeature +import com.twitter.home_mixer.model.HomeFeatures.DirectedAtUserIdFeature +import com.twitter.home_mixer.model.HomeFeatures.FavoritedByUserIdsFeature +import com.twitter.home_mixer.model.HomeFeatures.FollowedByUserIdsFeature +import com.twitter.home_mixer.model.HomeFeatures.InNetworkFeature +import com.twitter.home_mixer.model.HomeFeatures.InReplyToTweetIdFeature +import com.twitter.home_mixer.model.HomeFeatures.InReplyToUserIdFeature +import com.twitter.home_mixer.model.HomeFeatures.IsReadFromCacheFeature +import com.twitter.home_mixer.model.HomeFeatures.IsRetweetFeature +import com.twitter.home_mixer.model.HomeFeatures.LastScoredTimestampMsFeature +import com.twitter.home_mixer.model.HomeFeatures.QuotedTweetIdFeature +import com.twitter.home_mixer.model.HomeFeatures.QuotedUserIdFeature +import com.twitter.home_mixer.model.HomeFeatures.ScoreFeature +import com.twitter.home_mixer.model.HomeFeatures.SourceTweetIdFeature +import com.twitter.home_mixer.model.HomeFeatures.SourceUserIdFeature +import com.twitter.home_mixer.model.HomeFeatures.SuggestTypeFeature +import com.twitter.home_mixer.model.HomeFeatures.TopicContextFunctionalityTypeFeature +import com.twitter.home_mixer.model.HomeFeatures.TopicIdSocialContextFeature +import com.twitter.home_mixer.model.HomeFeatures.TweetUrlsFeature +import com.twitter.home_mixer.model.HomeFeatures.WeightedModelScoreFeature +import com.twitter.home_mixer.{thriftscala => hmt} +import com.twitter.product_mixer.core.feature.Feature +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.feature.featuremap.FeatureMapBuilder +import com.twitter.product_mixer.core.functional_component.transformer.CandidateFeatureTransformer +import com.twitter.product_mixer.core.model.common.identifier.TransformerIdentifier + +object CachedScoredTweetsResponseFeatureTransformer + extends CandidateFeatureTransformer[hmt.CachedScoredTweet] { + + override val identifier: TransformerIdentifier = + TransformerIdentifier("CachedScoredTweetsResponse") + + override val features: Set[Feature[_, _]] = Set( + AncestorsFeature, + AuthorIdFeature, + AuthorIsBlueVerifiedFeature, + CachedCandidatePipelineIdentifierFeature, + DirectedAtUserIdFeature, + FavoritedByUserIdsFeature, + FollowedByUserIdsFeature, + InNetworkFeature, + InReplyToTweetIdFeature, + InReplyToUserIdFeature, + IsReadFromCacheFeature, + IsRetweetFeature, + LastScoredTimestampMsFeature, + QuotedTweetIdFeature, + QuotedUserIdFeature, + ScoreFeature, + SourceTweetIdFeature, + SourceUserIdFeature, + SuggestTypeFeature, + TopicContextFunctionalityTypeFeature, + TopicIdSocialContextFeature, + TweetUrlsFeature, + WeightedModelScoreFeature + ) + + override def transform(candidate: hmt.CachedScoredTweet): FeatureMap = + FeatureMapBuilder() + .add(AncestorsFeature, candidate.ancestors.getOrElse(Seq.empty)) + .add(AuthorIdFeature, candidate.userId) + .add(AuthorIsBlueVerifiedFeature, candidate.authorIsBlueVerified.getOrElse(false)) + .add(CachedCandidatePipelineIdentifierFeature, candidate.candidatePipelineIdentifier) + .add(DirectedAtUserIdFeature, candidate.directedAtUserId) + .add(FavoritedByUserIdsFeature, candidate.favoritedByUserIds.getOrElse(Seq.empty)) + .add(FollowedByUserIdsFeature, candidate.followedByUserIds.getOrElse(Seq.empty)) + .add(InNetworkFeature, candidate.isInNetwork.getOrElse(false)) + .add(InReplyToTweetIdFeature, candidate.inReplyToTweetId) + .add(InReplyToUserIdFeature, candidate.inReplyToUserId) + .add(IsReadFromCacheFeature, true) + .add(IsRetweetFeature, candidate.isRetweet.getOrElse(false)) + .add(LastScoredTimestampMsFeature, candidate.lastScoredTimestampMs) + .add(QuotedTweetIdFeature, candidate.quotedTweetId) + .add(QuotedUserIdFeature, candidate.quotedUserId) + .add(ScoreFeature, candidate.score) + .add(SourceTweetIdFeature, candidate.sourceTweetId) + .add(SourceUserIdFeature, candidate.sourceUserId) + .add(SuggestTypeFeature, candidate.suggestType) + .add( + TopicContextFunctionalityTypeFeature, + candidate.topicFunctionalityType.map(TopicContextFunctionalityTypeUnmarshaller(_))) + .add(TopicIdSocialContextFeature, candidate.topicId) + .add(TweetUrlsFeature, candidate.urlsList.getOrElse(Seq.empty)) + .add(WeightedModelScoreFeature, candidate.score) + .build() +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/response_transformer/ScoredTweetsCrMixerResponseFeatureTransformer.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/response_transformer/ScoredTweetsCrMixerResponseFeatureTransformer.scala new file mode 100644 index 0000000000..96aef36b62 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/response_transformer/ScoredTweetsCrMixerResponseFeatureTransformer.scala @@ -0,0 +1,65 @@ +package com.twitter.home_mixer.product.scored_tweets.response_transformer + +import com.twitter.cr_mixer.{thriftscala => crm} +import com.twitter.home_mixer.model.HomeFeatures.AuthorIdFeature +import com.twitter.home_mixer.model.HomeFeatures.CandidateSourceIdFeature +import com.twitter.home_mixer.model.HomeFeatures.FromInNetworkSourceFeature +import com.twitter.home_mixer.model.HomeFeatures.IsRandomTweetFeature +import com.twitter.home_mixer.model.HomeFeatures.StreamToKafkaFeature +import com.twitter.home_mixer.model.HomeFeatures.SuggestTypeFeature +import com.twitter.home_mixer.model.HomeFeatures.TSPMetricTagFeature +import com.twitter.product_mixer.core.feature.Feature +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.feature.featuremap.FeatureMapBuilder +import com.twitter.product_mixer.core.functional_component.transformer.CandidateFeatureTransformer +import com.twitter.product_mixer.core.model.common.identifier.TransformerIdentifier +import com.twitter.timelineservice.suggests.logging.candidate_tweet_source_id.{thriftscala => cts} +import com.twitter.timelineservice.suggests.{thriftscala => st} +import com.twitter.tsp.{thriftscala => tsp} + +object ScoredTweetsCrMixerResponseFeatureTransformer + extends CandidateFeatureTransformer[crm.TweetRecommendation] { + + override val identifier: TransformerIdentifier = + TransformerIdentifier("ScoredTweetsCrMixerResponse") + + override val features: Set[Feature[_, _]] = Set( + AuthorIdFeature, + CandidateSourceIdFeature, + FromInNetworkSourceFeature, + IsRandomTweetFeature, + StreamToKafkaFeature, + SuggestTypeFeature, + TSPMetricTagFeature + ) + + override def transform(candidate: crm.TweetRecommendation): FeatureMap = { + val crMixerMetricTags = candidate.metricTags.getOrElse(Seq.empty) + val tspMetricTag = crMixerMetricTags + .map(CrMixerMetricTagToTspMetricTag) + .filter(_.nonEmpty).map(_.get).toSet + + FeatureMapBuilder() + .add(AuthorIdFeature, candidate.authorId) + .add(CandidateSourceIdFeature, Some(cts.CandidateTweetSourceId.Simcluster)) + .add(FromInNetworkSourceFeature, false) + .add(IsRandomTweetFeature, false) + .add(StreamToKafkaFeature, true) + .add(SuggestTypeFeature, Some(st.SuggestType.ScTweet)) + .add(TSPMetricTagFeature, tspMetricTag) + .build() + } + + private def CrMixerMetricTagToTspMetricTag( + crMixerMetricTag: crm.MetricTag + ): Option[tsp.MetricTag] = crMixerMetricTag match { + case crm.MetricTag.TweetFavorite => Some(tsp.MetricTag.TweetFavorite) + case crm.MetricTag.Retweet => Some(tsp.MetricTag.Retweet) + case crm.MetricTag.UserFollow => Some(tsp.MetricTag.UserFollow) + case crm.MetricTag.PushOpenOrNtabClick => Some(tsp.MetricTag.PushOpenOrNtabClick) + case crm.MetricTag.UserInterestedIn => Some(tsp.MetricTag.UserInterestedIn) + case crm.MetricTag.HomeTweetClick => Some(tsp.MetricTag.HomeTweetClick) + case crm.MetricTag.HomeVideoView => Some(tsp.MetricTag.HomeVideoView) + case _ => None + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/response_transformer/ScoredTweetsFrsResponseFeatureTransformer.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/response_transformer/ScoredTweetsFrsResponseFeatureTransformer.scala new file mode 100644 index 0000000000..077dccf24b --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/response_transformer/ScoredTweetsFrsResponseFeatureTransformer.scala @@ -0,0 +1,31 @@ +package com.twitter.home_mixer.product.scored_tweets.response_transformer + +import com.twitter.home_mixer.model.HomeFeatures.CandidateSourceIdFeature +import com.twitter.home_mixer.model.HomeFeatures.SuggestTypeFeature +import com.twitter.product_mixer.core.feature.Feature +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.feature.featuremap.FeatureMapBuilder +import com.twitter.product_mixer.core.functional_component.transformer.CandidateFeatureTransformer +import com.twitter.product_mixer.core.model.common.identifier.TransformerIdentifier +import com.twitter.timelineranker.{thriftscala => tlr} +import com.twitter.timelineservice.suggests.logging.candidate_tweet_source_id.{thriftscala => cts} +import com.twitter.timelineservice.suggests.{thriftscala => st} + +object ScoredTweetsFrsResponseFeatureTransformer + extends CandidateFeatureTransformer[tlr.CandidateTweet] { + + override val identifier: TransformerIdentifier = TransformerIdentifier("ScoredTweetsFrsResponse") + + override val features: Set[Feature[_, _]] = TimelineRankerResponseTransformer.features + + override def transform(candidate: tlr.CandidateTweet): FeatureMap = { + val baseFeatures = TimelineRankerResponseTransformer.transform(candidate) + + val features = FeatureMapBuilder() + .add(CandidateSourceIdFeature, Some(cts.CandidateTweetSourceId.FrsTweet)) + .add(SuggestTypeFeature, Some(st.SuggestType.FrsTweet)) + .build() + + baseFeatures ++ features + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/response_transformer/ScoredTweetsInNetworkResponseFeatureTransformer.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/response_transformer/ScoredTweetsInNetworkResponseFeatureTransformer.scala new file mode 100644 index 0000000000..ada45d3e18 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/response_transformer/ScoredTweetsInNetworkResponseFeatureTransformer.scala @@ -0,0 +1,34 @@ +package com.twitter.home_mixer.product.scored_tweets.response_transformer + +import com.twitter.home_mixer.model.HomeFeatures.CandidateSourceIdFeature +import com.twitter.home_mixer.model.HomeFeatures.FromInNetworkSourceFeature +import com.twitter.home_mixer.model.HomeFeatures.SuggestTypeFeature +import com.twitter.product_mixer.core.feature.Feature +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.feature.featuremap.FeatureMapBuilder +import com.twitter.product_mixer.core.functional_component.transformer.CandidateFeatureTransformer +import com.twitter.product_mixer.core.model.common.identifier.TransformerIdentifier +import com.twitter.timelineranker.{thriftscala => tlr} +import com.twitter.timelineservice.suggests.logging.candidate_tweet_source_id.{thriftscala => cts} +import com.twitter.timelineservice.suggests.{thriftscala => st} + +object ScoredTweetsInNetworkResponseFeatureTransformer + extends CandidateFeatureTransformer[tlr.CandidateTweet] { + + override val identifier: TransformerIdentifier = + TransformerIdentifier("ScoredTweetsInNetworkResponse") + + override val features: Set[Feature[_, _]] = TimelineRankerResponseTransformer.features + + override def transform(candidate: tlr.CandidateTweet): FeatureMap = { + val baseFeatures = TimelineRankerResponseTransformer.transform(candidate) + + val features = FeatureMapBuilder() + .add(CandidateSourceIdFeature, Some(cts.CandidateTweetSourceId.RecycledTweet)) + .add(FromInNetworkSourceFeature, true) + .add(SuggestTypeFeature, Some(st.SuggestType.RecycledTweetInline)) + .build() + + baseFeatures ++ features + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/response_transformer/ScoredTweetsUtegResponseFeatureTransformer.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/response_transformer/ScoredTweetsUtegResponseFeatureTransformer.scala new file mode 100644 index 0000000000..e7bd61b2b3 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/response_transformer/ScoredTweetsUtegResponseFeatureTransformer.scala @@ -0,0 +1,31 @@ +package com.twitter.home_mixer.product.scored_tweets.response_transformer + +import com.twitter.home_mixer.model.HomeFeatures.CandidateSourceIdFeature +import com.twitter.home_mixer.model.HomeFeatures.SuggestTypeFeature +import com.twitter.product_mixer.core.feature.Feature +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.feature.featuremap.FeatureMapBuilder +import com.twitter.product_mixer.core.functional_component.transformer.CandidateFeatureTransformer +import com.twitter.product_mixer.core.model.common.identifier.TransformerIdentifier +import com.twitter.timelineranker.{thriftscala => tlr} +import com.twitter.timelineservice.suggests.logging.candidate_tweet_source_id.{thriftscala => cts} +import com.twitter.timelineservice.suggests.{thriftscala => st} + +object ScoredTweetsUtegResponseFeatureTransformer + extends CandidateFeatureTransformer[tlr.CandidateTweet] { + + override val identifier: TransformerIdentifier = TransformerIdentifier("ScoredTweetsUtegResponse") + + override val features: Set[Feature[_, _]] = TimelineRankerResponseTransformer.features + + override def transform(candidate: tlr.CandidateTweet): FeatureMap = { + val baseFeatures = TimelineRankerResponseTransformer.transform(candidate) + + val features = FeatureMapBuilder() + .add(CandidateSourceIdFeature, Some(cts.CandidateTweetSourceId.RecommendedTweet)) + .add(SuggestTypeFeature, Some(st.SuggestType.ActivityTweet)) + .build() + + baseFeatures ++ features + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/response_transformer/TimelineRankerResponseTransformer.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/response_transformer/TimelineRankerResponseTransformer.scala new file mode 100644 index 0000000000..e431ef6f4c --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/response_transformer/TimelineRankerResponseTransformer.scala @@ -0,0 +1,91 @@ +package com.twitter.home_mixer.product.scored_tweets.response_transformer + +import com.twitter.home_mixer.model.HomeFeatures.AuthorIdFeature +import com.twitter.home_mixer.model.HomeFeatures.CandidateSourceIdFeature +import com.twitter.home_mixer.model.HomeFeatures.DirectedAtUserIdFeature +import com.twitter.home_mixer.model.HomeFeatures.EarlybirdFeature +import com.twitter.home_mixer.model.HomeFeatures.EarlybirdScoreFeature +import com.twitter.home_mixer.model.HomeFeatures.FromInNetworkSourceFeature +import com.twitter.home_mixer.model.HomeFeatures.HasImageFeature +import com.twitter.home_mixer.model.HomeFeatures.HasVideoFeature +import com.twitter.home_mixer.model.HomeFeatures.InReplyToTweetIdFeature +import com.twitter.home_mixer.model.HomeFeatures.InReplyToUserIdFeature +import com.twitter.home_mixer.model.HomeFeatures.IsRandomTweetFeature +import com.twitter.home_mixer.model.HomeFeatures.IsRetweetFeature +import com.twitter.home_mixer.model.HomeFeatures.MentionScreenNameFeature +import com.twitter.home_mixer.model.HomeFeatures.MentionUserIdFeature +import com.twitter.home_mixer.model.HomeFeatures.QuotedTweetIdFeature +import com.twitter.home_mixer.model.HomeFeatures.QuotedUserIdFeature +import com.twitter.home_mixer.model.HomeFeatures.SemanticAnnotationFeature +import com.twitter.home_mixer.model.HomeFeatures.SourceTweetIdFeature +import com.twitter.home_mixer.model.HomeFeatures.SourceUserIdFeature +import com.twitter.home_mixer.model.HomeFeatures.StreamToKafkaFeature +import com.twitter.home_mixer.model.HomeFeatures.SuggestTypeFeature +import com.twitter.home_mixer.model.HomeFeatures.TweetUrlsFeature +import com.twitter.home_mixer.util.tweetypie.content.TweetMediaFeaturesExtractor +import com.twitter.product_mixer.core.feature.Feature +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.feature.featuremap.FeatureMapBuilder +import com.twitter.timelineranker.{thriftscala => tlr} + +object TimelineRankerResponseTransformer { + + val features: Set[Feature[_, _]] = Set( + AuthorIdFeature, + CandidateSourceIdFeature, + DirectedAtUserIdFeature, + EarlybirdFeature, + EarlybirdScoreFeature, + FromInNetworkSourceFeature, + HasImageFeature, + HasVideoFeature, + InReplyToTweetIdFeature, + InReplyToUserIdFeature, + IsRandomTweetFeature, + IsRetweetFeature, + MentionScreenNameFeature, + MentionUserIdFeature, + SemanticAnnotationFeature, + StreamToKafkaFeature, + QuotedTweetIdFeature, + QuotedUserIdFeature, + SourceTweetIdFeature, + SourceUserIdFeature, + SuggestTypeFeature, + TweetUrlsFeature + ) + + def transform(candidate: tlr.CandidateTweet): FeatureMap = { + val tweet = candidate.tweet + val quotedTweet = tweet.flatMap(_.quotedTweet) + val mentions = tweet.flatMap(_.mentions).getOrElse(Seq.empty) + val coreData = tweet.flatMap(_.coreData) + val share = coreData.flatMap(_.share) + val reply = coreData.flatMap(_.reply) + val semanticAnnotations = + tweet.flatMap(_.escherbirdEntityAnnotations.map(_.entityAnnotations)).getOrElse(Seq.empty) + + FeatureMapBuilder() + .add(AuthorIdFeature, coreData.map(_.userId)) + .add(DirectedAtUserIdFeature, coreData.flatMap(_.directedAtUser.map(_.userId))) + .add(EarlybirdFeature, candidate.features) + .add(EarlybirdScoreFeature, candidate.features.map(_.earlybirdScore)) + .add(FromInNetworkSourceFeature, false) + .add(HasImageFeature, tweet.exists(TweetMediaFeaturesExtractor.hasImage)) + .add(HasVideoFeature, tweet.exists(TweetMediaFeaturesExtractor.hasVideo)) + .add(InReplyToTweetIdFeature, reply.flatMap(_.inReplyToStatusId)) + .add(InReplyToUserIdFeature, reply.map(_.inReplyToUserId)) + .add(IsRandomTweetFeature, candidate.features.exists(_.isRandomTweet.getOrElse(false))) + .add(IsRetweetFeature, share.isDefined) + .add(MentionScreenNameFeature, mentions.map(_.screenName)) + .add(MentionUserIdFeature, mentions.flatMap(_.userId)) + .add(SemanticAnnotationFeature, semanticAnnotations) + .add(StreamToKafkaFeature, true) + .add(QuotedTweetIdFeature, quotedTweet.map(_.tweetId)) + .add(QuotedUserIdFeature, quotedTweet.map(_.userId)) + .add(SourceTweetIdFeature, share.map(_.sourceStatusId)) + .add(SourceUserIdFeature, share.map(_.sourceUserId)) + .add(TweetUrlsFeature, candidate.features.flatMap(_.urlsList).getOrElse(Seq.empty)) + .build() + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/scorer/BUILD.bazel b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/scorer/BUILD.bazel new file mode 100644 index 0000000000..6fcaddf0f8 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/scorer/BUILD.bazel @@ -0,0 +1,22 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + strict_deps = True, + dependencies = [ + "3rdparty/jvm/io/opil:tensorflow-serving-client", + "cortex-deepbird/thrift/src/main/thrift:thrift-java", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/model", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/model/request", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/module", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/model", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/param", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/scorer/common", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/feature/featuremap/datarecord", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/product", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/util", + "src/scala/com/twitter/timelines/prediction/features/recap", + "src/thrift/com/twitter/timelinescorer:thrift-scala", + ], +) diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/scorer/DiversityDiscountProvider.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/scorer/DiversityDiscountProvider.scala new file mode 100644 index 0000000000..aa292e89ab --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/scorer/DiversityDiscountProvider.scala @@ -0,0 +1,30 @@ +package com.twitter.home_mixer.product.scored_tweets.scorer + +import com.twitter.home_mixer.model.HomeFeatures.AuthorIdFeature +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.model.common.CandidateWithFeatures + +trait DiversityDiscountProvider { + + def entityId(candidate: CandidateWithFeatures[TweetCandidate]): Option[Long] + + /** + * Compute the discounted score for the position + * @param score the previous score for the candidate + * @param position zero-based position for the candidate for the given entity + * @return the discounted score for the candidate + */ + def discount(score: Double, position: Int): Double +} + +object AuthorDiversityDiscountProvider extends DiversityDiscountProvider { + private val Decay = 0.5 + private val Floor = 0.25 + + override def entityId(candidate: CandidateWithFeatures[TweetCandidate]): Option[Long] = + candidate.features.getOrElse(AuthorIdFeature, None) + + // Provides an exponential decay based discount by position (with a floor) + override def discount(score: Double, position: Int): Double = + score * ((1 - Floor) * Math.pow(Decay, position) + Floor) +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/scorer/DiversityScorer.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/scorer/DiversityScorer.scala new file mode 100644 index 0000000000..758467717e --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/scorer/DiversityScorer.scala @@ -0,0 +1,57 @@ +package com.twitter.home_mixer.product.scored_tweets.scorer + +import com.twitter.home_mixer.model.HomeFeatures.ScoreFeature +import com.twitter.home_mixer.product.scored_tweets.model.ScoredTweetsQuery +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.feature.Feature +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.feature.featuremap.FeatureMapBuilder +import com.twitter.product_mixer.core.functional_component.scorer.Scorer +import com.twitter.product_mixer.core.model.common.CandidateWithFeatures +import com.twitter.product_mixer.core.model.common.identifier.ScorerIdentifier +import com.twitter.stitch.Stitch + +/** + * Discounts scores of each consecutive tweet (ordered by score desc) from the + * same entity (e.g. author, engager, topic) based on the discount factor provided + */ + +case class DiversityScorer(diversityDiscountProvider: DiversityDiscountProvider) + extends Scorer[ScoredTweetsQuery, TweetCandidate] { + + override val identifier: ScorerIdentifier = ScorerIdentifier("Diversity") + + override val features: Set[Feature[_, _]] = Set(ScoreFeature) + + override def apply( + query: ScoredTweetsQuery, + candidates: Seq[CandidateWithFeatures[TweetCandidate]] + ): Stitch[Seq[FeatureMap]] = { + val candidateIdScoreMap = candidates + .groupBy(diversityDiscountProvider.entityId) + .flatMap { + case (entityIdOpt, entityCandidates) => + val candidateScores = entityCandidates + .map { candidate => + val score = candidate.features.getOrElse(ScoreFeature, None).getOrElse(0.0) + (candidate.candidate.id, score) + }.sortBy(_._2)(Ordering.Double.reverse) + + if (entityIdOpt.isDefined) { + candidateScores.zipWithIndex.map { + case ((candidateId, score), index) => + candidateId -> diversityDiscountProvider.discount(score, index) + } + } else candidateScores + } + + Stitch.value { + candidates.map { candidate => + val score = candidateIdScoreMap.getOrElse(candidate.candidate.id, 0.0) + FeatureMapBuilder() + .add(ScoreFeature, Some(score)) + .build() + } + } + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/scorer/HomeNaviModelDataRecordScorer.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/scorer/HomeNaviModelDataRecordScorer.scala new file mode 100644 index 0000000000..23bf07da6d --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/scorer/HomeNaviModelDataRecordScorer.scala @@ -0,0 +1,233 @@ +package com.twitter.home_mixer.product.scored_tweets.scorer + +import com.twitter.dal.personal_data.{thriftjava => pd} +import com.twitter.finagle.stats.Stat +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.home_mixer.product.scored_tweets.param.ScoredTweetsParam.Scoring.ModelWeights +import com.twitter.ml.api.DataRecord +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.feature.Feature +import com.twitter.product_mixer.core.feature.FeatureWithDefaultOnFailure +import com.twitter.product_mixer.core.feature.datarecord.BaseDataRecordFeature +import com.twitter.product_mixer.core.feature.datarecord.DataRecordInAFeature +import com.twitter.product_mixer.core.feature.datarecord.DataRecordOptionalFeature +import com.twitter.product_mixer.core.feature.datarecord.DoubleDataRecordCompatible +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.feature.featuremap.datarecord.AllFeatures +import com.twitter.product_mixer.core.feature.featuremap.datarecord.DataRecordConverter +import com.twitter.product_mixer.core.feature.featuremap.datarecord.DataRecordExtractor +import com.twitter.product_mixer.core.feature.featuremap.datarecord.FeaturesScope +import com.twitter.product_mixer.core.functional_component.scorer.Scorer +import com.twitter.product_mixer.core.model.common.CandidateWithFeatures +import com.twitter.product_mixer.core.model.common.UniversalNoun +import com.twitter.product_mixer.core.model.common.identifier.ScorerIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.product_mixer.core.pipeline.pipeline_failure.IllegalStateFailure +import com.twitter.product_mixer.core.pipeline.pipeline_failure.PipelineFailure +import com.twitter.product_mixer.core.util.OffloadFuturePools +import com.twitter.stitch.Stitch +import com.twitter.timelines.clients.predictionservice.PredictionServiceGRPCClient +import com.twitter.timelines.configapi.FSBoundedParam +import com.twitter.timelines.prediction.features.recap.RecapFeatures +import com.twitter.util.Future +import com.twitter.util.Return + +object CommonFeaturesDataRecordFeature + extends DataRecordInAFeature[PipelineQuery] + with FeatureWithDefaultOnFailure[PipelineQuery, DataRecord] { + override def defaultValue: DataRecord = new DataRecord() +} + +object CandidateFeaturesDataRecordFeature + extends DataRecordInAFeature[TweetCandidate] + with FeatureWithDefaultOnFailure[TweetCandidate, DataRecord] { + override def defaultValue: DataRecord = new DataRecord() +} + +case class HomeNaviModelDataRecordScorer[ + Query <: PipelineQuery, + Candidate <: UniversalNoun[Any], + CandidateFeatures <: BaseDataRecordFeature[Candidate, _], + ResultFeatures <: BaseDataRecordFeature[Candidate, _] +]( + override val identifier: ScorerIdentifier, + modelClient: PredictionServiceGRPCClient, + candidateFeatures: FeaturesScope[CandidateFeatures], + resultFeatures: Set[ResultFeatures], + statsReceiver: StatsReceiver) + extends Scorer[Query, Candidate] { + + require(resultFeatures.nonEmpty, "Result features cannot be empty") + + override val features: Set[Feature[_, _]] = + resultFeatures.asInstanceOf[ + Set[Feature[_, _]]] + CommonFeaturesDataRecordFeature + CandidateFeaturesDataRecordFeature + + private val queryDataRecordAdapter = new DataRecordConverter(AllFeatures()) + private val candidatesDataRecordAdapter = new DataRecordConverter(candidateFeatures) + private val resultDataRecordExtractor = new DataRecordExtractor(resultFeatures) + + private val scopedStatsReceiver = statsReceiver.scope(getClass.getSimpleName) + private val failuresStat = scopedStatsReceiver.stat("failures") + private val responsesStat = scopedStatsReceiver.stat("responses") + private val invalidResponsesSizeCounter = scopedStatsReceiver.counter("invalidResponsesSize") + private val candidatesDataRecordAdapterLatencyStat = + scopedStatsReceiver.scope("candidatesDataRecordAdapter").stat("latency_ms") + + private val DataRecordConstructionParallelism = 32 + + override def apply( + query: Query, + candidates: Seq[CandidateWithFeatures[Candidate]] + ): Stitch[Seq[FeatureMap]] = { + val commonRecord = query.features.map(queryDataRecordAdapter.toDataRecord) + val candidateRecords: Future[Seq[DataRecord]] = + Stat.time(candidatesDataRecordAdapterLatencyStat) { + OffloadFuturePools.parallelize[FeatureMap, DataRecord]( + candidates.map(_.features), + candidatesDataRecordAdapter.toDataRecord(_), + DataRecordConstructionParallelism, + new DataRecord + ) + } + + Stitch.callFuture { + candidateRecords.flatMap { records => + val predictionResponses = + modelClient.getPredictions( + records = records, + commonFeatures = commonRecord, + modelId = Some("Home") + ) + + predictionResponses.map { responses => + failuresStat.add(responses.count(_.isThrow)) + responsesStat.add(responses.size) + + if (responses.size == candidates.size) { + val predictedScoreFeatureMaps = responses.map { + case Return(dataRecord) => + resultDataRecordExtractor.fromDataRecord(dataRecord) + case _ => + resultDataRecordExtractor.fromDataRecord(new DataRecord()) + } + + // add Data Record to feature map, which will be used for logging in later stage + predictedScoreFeatureMaps.zip(records).map { + case (predictedScoreFeatureMap, candidateRecord) => + predictedScoreFeatureMap + + (key = CandidateFeaturesDataRecordFeature, value = candidateRecord) + + (key = CommonFeaturesDataRecordFeature, value = + commonRecord.getOrElse(new DataRecord())) + } + } else { + invalidResponsesSizeCounter.incr() + throw PipelineFailure(IllegalStateFailure, "Result Size mismatched candidates size") + } + } + } + } + } +} + +/** + * Features for results returned by Navi user-tweet prediction models. + */ +object HomeNaviModelDataRecordScorer { + val RequestBatchSize = 32 + + sealed trait PredictedScoreFeature + extends DataRecordOptionalFeature[TweetCandidate, Double] + with DoubleDataRecordCompatible { + def statName: String + + def modelWeightParam: FSBoundedParam[Double] + } + + object PredictedFavoriteScoreFeature extends PredictedScoreFeature { + override val featureName: String = RecapFeatures.PREDICTED_IS_FAVORITED.getFeatureName + override val personalDataTypes: Set[pd.PersonalDataType] = Set.empty + override val statName = "fav" + override val modelWeightParam = ModelWeights.FavParam + } + + object PredictedReplyScoreFeature extends PredictedScoreFeature { + override val featureName: String = RecapFeatures.PREDICTED_IS_REPLIED.getFeatureName + override val personalDataTypes: Set[pd.PersonalDataType] = Set.empty + override val statName = "reply" + override val modelWeightParam = ModelWeights.ReplyParam + } + + object PredictedRetweetScoreFeature extends PredictedScoreFeature { + override val featureName: String = RecapFeatures.PREDICTED_IS_RETWEETED.getFeatureName + override val personalDataTypes: Set[pd.PersonalDataType] = Set.empty + override val statName = "retweet" + override val modelWeightParam = ModelWeights.RetweetParam + } + + object PredictedReplyEngagedByAuthorScoreFeature extends PredictedScoreFeature { + override val featureName: String = + RecapFeatures.PREDICTED_IS_REPLIED_REPLY_ENGAGED_BY_AUTHOR.getFeatureName + override val personalDataTypes: Set[pd.PersonalDataType] = Set.empty + override val statName = "reply_engaged_by_author" + override val modelWeightParam = ModelWeights.ReplyEngagedByAuthorParam + } + + object PredictedGoodClickConvoDescFavoritedOrRepliedScoreFeature extends PredictedScoreFeature { + override val featureName: String = RecapFeatures.PREDICTED_IS_GOOD_CLICKED_V1.getFeatureName + override val personalDataTypes: Set[pd.PersonalDataType] = Set.empty + override val statName = "good_click_convo_desc_favorited_or_replied" + override val modelWeightParam = ModelWeights.GoodClickParam + } + + object PredictedGoodClickConvoDescUamGt2ScoreFeature extends PredictedScoreFeature { + override val featureName: String = RecapFeatures.PREDICTED_IS_GOOD_CLICKED_V2.getFeatureName + override val personalDataTypes: Set[pd.PersonalDataType] = Set.empty + override val statName = "good_click_convo_desc_uam_gt_2" + override val modelWeightParam = ModelWeights.GoodClickV2Param + } + + object PredictedNegativeFeedbackV2ScoreFeature extends PredictedScoreFeature { + override val featureName: String = + RecapFeatures.PREDICTED_IS_NEGATIVE_FEEDBACK_V2.getFeatureName + override val personalDataTypes: Set[pd.PersonalDataType] = Set.empty + override val statName = "negative_feedback_v2" + override val modelWeightParam = ModelWeights.NegativeFeedbackV2Param + } + + object PredictedGoodProfileClickScoreFeature extends PredictedScoreFeature { + override val featureName: String = + RecapFeatures.PREDICTED_IS_PROFILE_CLICKED_AND_PROFILE_ENGAGED.getFeatureName + override val personalDataTypes: Set[pd.PersonalDataType] = Set.empty + override val statName = "good_profile_click" + override val modelWeightParam = ModelWeights.GoodProfileClickParam + } + + object PredictedReportedScoreFeature extends PredictedScoreFeature { + override val featureName: String = + RecapFeatures.PREDICTED_IS_REPORT_TWEET_CLICKED.getFeatureName + override val personalDataTypes: Set[pd.PersonalDataType] = Set.empty + override val statName = "reported" + override val modelWeightParam = ModelWeights.ReportParam + } + + object PredictedVideoPlayback50ScoreFeature extends PredictedScoreFeature { + override val featureName: String = RecapFeatures.PREDICTED_IS_VIDEO_PLAYBACK_50.getFeatureName + override val personalDataTypes: Set[pd.PersonalDataType] = Set.empty + override val statName = "video_playback_50" + override val modelWeightParam = ModelWeights.VideoPlayback50Param + } + + val PredictedScoreFeatures: Seq[PredictedScoreFeature] = Seq( + PredictedFavoriteScoreFeature, + PredictedReplyScoreFeature, + PredictedRetweetScoreFeature, + PredictedReplyEngagedByAuthorScoreFeature, + PredictedGoodClickConvoDescFavoritedOrRepliedScoreFeature, + PredictedGoodClickConvoDescUamGt2ScoreFeature, + PredictedNegativeFeedbackV2ScoreFeature, + PredictedGoodProfileClickScoreFeature, + PredictedReportedScoreFeature, + PredictedVideoPlayback50ScoreFeature, + ) +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/scorer/WeightedScoresSumScorer.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/scorer/WeightedScoresSumScorer.scala new file mode 100644 index 0000000000..9f937448f5 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/scorer/WeightedScoresSumScorer.scala @@ -0,0 +1,91 @@ +package com.twitter.home_mixer.product.scored_tweets.scorer + +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.home_mixer.model.HomeFeatures.ScoreFeature +import com.twitter.home_mixer.model.HomeFeatures.WeightedModelScoreFeature +import com.twitter.home_mixer.product.scored_tweets.model.ScoredTweetsQuery +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.feature.Feature +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.feature.featuremap.FeatureMapBuilder +import com.twitter.product_mixer.core.functional_component.scorer.Scorer +import com.twitter.product_mixer.core.model.common.CandidateWithFeatures +import com.twitter.product_mixer.core.model.common.identifier.ScorerIdentifier +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.stitch.Stitch +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class WeightedScoresSumScorer @Inject() (statsReceiver: StatsReceiver) + extends Scorer[ScoredTweetsQuery, TweetCandidate] { + + override val identifier: ScorerIdentifier = ScorerIdentifier("WeightedScoresSum") + + override val features: Set[Feature[_, _]] = Set(WeightedModelScoreFeature, ScoreFeature) + + private val StatsReadabilityMultiplier = 1000 + private val Epsilon = 0.001 + private val PredictedScoreStatName = f"predicted_score_${StatsReadabilityMultiplier}x" + private val MissingScoreStatName = "missing_score" + + private val scopedStatsProvider = statsReceiver.scope(getClass.getSimpleName) + private val scoreStat = scopedStatsProvider.stat(f"score_${StatsReadabilityMultiplier}x") + + override def apply( + query: ScoredTweetsQuery, + candidates: Seq[CandidateWithFeatures[TweetCandidate]] + ): Stitch[Seq[FeatureMap]] = { + val features = candidates.map { candidate => + val score = weightedModelScore(query, candidate.features) + scoreStat.add((score * StatsReadabilityMultiplier).toFloat) + FeatureMapBuilder() + .add(WeightedModelScoreFeature, Some(score)) + .add(ScoreFeature, Some(score)) + .build() + } + + Stitch.value(features) + } + + /** + * (1) compute weighted sum of predicted scores of all engagements + * (2) convert negative score to positive score if needed + */ + private def weightedModelScore( + query: PipelineQuery, + features: FeatureMap + ): Double = { + val weightedScoreAndModelWeightSeq: Seq[(Double, Double)] = + HomeNaviModelDataRecordScorer.PredictedScoreFeatures.map { scoreFeature => + val predictedScoreOpt = features.getOrElse(scoreFeature, None) + + predictedScoreOpt match { + case Some(predictedScore) => + scopedStatsProvider + .stat(scoreFeature.statName, PredictedScoreStatName) + .add((predictedScore * StatsReadabilityMultiplier).toFloat) + case None => + scopedStatsProvider.counter(scoreFeature.statName, MissingScoreStatName).incr() + } + + val weight = query.params(scoreFeature.modelWeightParam) + (predictedScoreOpt.getOrElse(0.0) * weight, weight) + } + + val (weightedScores, modelWeights) = weightedScoreAndModelWeightSeq.unzip + val combinedScoreSum = weightedScores.sum + + val positiveModelWeightsSum = modelWeights.filter(_ > 0.0).sum + val negativeModelWeightsSum = modelWeights.filter(_ < 0).sum.abs + val modelWeightsSum = positiveModelWeightsSum + negativeModelWeightsSum + + val weightedScoresSum = + if (modelWeightsSum == 0) combinedScoreSum.max(0.0) + else if (combinedScoreSum < 0) + (combinedScoreSum + negativeModelWeightsSum) / modelWeightsSum * Epsilon + else combinedScoreSum + Epsilon + + weightedScoresSum + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/scoring_pipeline/BUILD.bazel b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/scoring_pipeline/BUILD.bazel new file mode 100644 index 0000000000..fcd1bbda88 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/scoring_pipeline/BUILD.bazel @@ -0,0 +1,33 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + strict_deps = True, + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/javax/inject:javax.inject", + "finatra/inject/inject-core/src/main/scala", + "finatra/inject/inject-core/src/main/scala/com/twitter/inject", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/offline_aggregates", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/real_time_aggregates", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/scorer", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/model", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/model/request", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/module", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/candidate_pipeline", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/model", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/param", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/scorer", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/util", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/gate", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/scorer/common", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/scorer/tweet_tlx", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/selector", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/feature", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/feature/featuremap/datarecord", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/candidate", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/scoring", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/product", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/product/guice", + ], +) diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/scoring_pipeline/ScoredTweetsDiversityScoringPipelineConfig.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/scoring_pipeline/ScoredTweetsDiversityScoringPipelineConfig.scala new file mode 100644 index 0000000000..3a202f5941 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/scoring_pipeline/ScoredTweetsDiversityScoringPipelineConfig.scala @@ -0,0 +1,25 @@ +package com.twitter.home_mixer.product.scored_tweets.scoring_pipeline + +import com.twitter.home_mixer.product.scored_tweets.model.ScoredTweetsQuery +import com.twitter.home_mixer.product.scored_tweets.scorer.AuthorDiversityDiscountProvider +import com.twitter.home_mixer.product.scored_tweets.scorer.DiversityScorer +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.component_library.selector.InsertAppendResults +import com.twitter.product_mixer.core.functional_component.common.AllPipelines +import com.twitter.product_mixer.core.functional_component.scorer.Scorer +import com.twitter.product_mixer.core.functional_component.selector.Selector +import com.twitter.product_mixer.core.model.common.identifier.ScoringPipelineIdentifier +import com.twitter.product_mixer.core.pipeline.scoring.ScoringPipelineConfig + +object ScoredTweetsDiversityScoringPipelineConfig + extends ScoringPipelineConfig[ScoredTweetsQuery, TweetCandidate] { + + override val identifier: ScoringPipelineIdentifier = + ScoringPipelineIdentifier("ScoredTweetsDiversity") + + override val selectors: Seq[Selector[ScoredTweetsQuery]] = + Seq(InsertAppendResults(AllPipelines)) + + override val scorers: Seq[Scorer[ScoredTweetsQuery, TweetCandidate]] = + Seq(DiversityScorer(AuthorDiversityDiscountProvider)) +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/scoring_pipeline/ScoredTweetsRescoreOONScoringPipelineConfig.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/scoring_pipeline/ScoredTweetsRescoreOONScoringPipelineConfig.scala new file mode 100644 index 0000000000..71d53d2ec3 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/scoring_pipeline/ScoredTweetsRescoreOONScoringPipelineConfig.scala @@ -0,0 +1,23 @@ +package com.twitter.home_mixer.product.scored_tweets.scoring_pipeline + +import com.twitter.home_mixer.functional_component.scorer.OONTweetScalingScorer +import com.twitter.home_mixer.product.scored_tweets.model.ScoredTweetsQuery +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.component_library.selector.InsertAppendResults +import com.twitter.product_mixer.core.functional_component.common.AllPipelines +import com.twitter.product_mixer.core.functional_component.scorer.Scorer +import com.twitter.product_mixer.core.functional_component.selector.Selector +import com.twitter.product_mixer.core.model.common.identifier.ScoringPipelineIdentifier +import com.twitter.product_mixer.core.pipeline.scoring.ScoringPipelineConfig + +object ScoredTweetsRescoreOONScoringPipelineConfig + extends ScoringPipelineConfig[ScoredTweetsQuery, TweetCandidate] { + + override val identifier: ScoringPipelineIdentifier = + ScoringPipelineIdentifier("ScoredTweetsRescoreOON") + + override val selectors: Seq[Selector[ScoredTweetsQuery]] = + Seq(InsertAppendResults(AllPipelines)) + + override val scorers: Seq[Scorer[ScoredTweetsQuery, TweetCandidate]] = Seq(OONTweetScalingScorer) +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/scoring_pipeline/ScoredTweetsRescoreVerifiedAuthorScoringPipelineConfig.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/scoring_pipeline/ScoredTweetsRescoreVerifiedAuthorScoringPipelineConfig.scala new file mode 100644 index 0000000000..d07e5c965e --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/scoring_pipeline/ScoredTweetsRescoreVerifiedAuthorScoringPipelineConfig.scala @@ -0,0 +1,24 @@ +package com.twitter.home_mixer.product.scored_tweets.scoring_pipeline + +import com.twitter.home_mixer.functional_component.scorer.VerifiedAuthorScalingScorer +import com.twitter.home_mixer.product.scored_tweets.model.ScoredTweetsQuery +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.component_library.selector.InsertAppendResults +import com.twitter.product_mixer.core.functional_component.common.AllPipelines +import com.twitter.product_mixer.core.functional_component.scorer.Scorer +import com.twitter.product_mixer.core.functional_component.selector.Selector +import com.twitter.product_mixer.core.model.common.identifier.ScoringPipelineIdentifier +import com.twitter.product_mixer.core.pipeline.scoring.ScoringPipelineConfig + +object ScoredTweetsRescoreVerifiedAuthorScoringPipelineConfig + extends ScoringPipelineConfig[ScoredTweetsQuery, TweetCandidate] { + + override val identifier: ScoringPipelineIdentifier = + ScoringPipelineIdentifier("ScoredTweetsRescoreVerifiedAuthor") + + override val selectors: Seq[Selector[ScoredTweetsQuery]] = + Seq(InsertAppendResults(AllPipelines)) + + override val scorers: Seq[Scorer[ScoredTweetsQuery, TweetCandidate]] = + Seq(VerifiedAuthorScalingScorer) +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/scoring_pipeline/ScoredTweetsScoringPipelineConfig.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/scoring_pipeline/ScoredTweetsScoringPipelineConfig.scala new file mode 100644 index 0000000000..6d2b35c19b --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/scoring_pipeline/ScoredTweetsScoringPipelineConfig.scala @@ -0,0 +1,174 @@ +package com.twitter.home_mixer.product.scored_tweets.scoring_pipeline + +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.home_mixer.functional_component.feature_hydrator._ +import com.twitter.home_mixer.functional_component.feature_hydrator.offline_aggregates.Phase1EdgeAggregateFeatureHydrator +import com.twitter.home_mixer.functional_component.feature_hydrator.offline_aggregates.Phase2EdgeAggregateFeatureHydrator +import com.twitter.home_mixer.functional_component.feature_hydrator.real_time_aggregates._ +import com.twitter.home_mixer.model.HomeFeatures.EarlybirdScoreFeature +import com.twitter.home_mixer.product.scored_tweets.candidate_pipeline.CachedScoredTweetsCandidatePipelineConfig +import com.twitter.home_mixer.product.scored_tweets.candidate_pipeline.ScoredTweetsCrMixerCandidatePipelineConfig +import com.twitter.home_mixer.product.scored_tweets.candidate_pipeline.ScoredTweetsFrsCandidatePipelineConfig +import com.twitter.home_mixer.product.scored_tweets.candidate_pipeline.ScoredTweetsInNetworkCandidatePipelineConfig +import com.twitter.home_mixer.product.scored_tweets.candidate_pipeline.ScoredTweetsUtegCandidatePipelineConfig +import com.twitter.home_mixer.product.scored_tweets.model.ScoredTweetsQuery +import com.twitter.home_mixer.product.scored_tweets.param.ScoredTweetsParam.QualityFactor +import com.twitter.home_mixer.product.scored_tweets.scorer.HomeNaviModelDataRecordScorer +import com.twitter.product_mixer.component_library.gate.NonEmptyCandidatesGate +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.component_library.selector.DropMaxCandidates +import com.twitter.product_mixer.component_library.selector.InsertAppendResults +import com.twitter.product_mixer.component_library.selector.UpdateSortCandidates +import com.twitter.product_mixer.core.feature.featuremap.datarecord.AllFeatures +import com.twitter.product_mixer.core.functional_component.common.AllExceptPipelines +import com.twitter.product_mixer.core.functional_component.common.SpecificPipelines +import com.twitter.product_mixer.core.functional_component.feature_hydrator.BaseCandidateFeatureHydrator +import com.twitter.product_mixer.core.functional_component.gate.BaseGate +import com.twitter.product_mixer.core.functional_component.scorer.Scorer +import com.twitter.product_mixer.core.functional_component.selector.Selector +import com.twitter.product_mixer.core.model.common.identifier.ScorerIdentifier +import com.twitter.product_mixer.core.model.common.identifier.ScoringPipelineIdentifier +import com.twitter.product_mixer.core.model.common.presentation.CandidateWithDetails +import com.twitter.product_mixer.core.model.common.presentation.ItemCandidateWithDetails +import com.twitter.product_mixer.core.pipeline.pipeline_failure.PipelineFailure +import com.twitter.product_mixer.core.pipeline.pipeline_failure.UnexpectedCandidateResult +import com.twitter.product_mixer.core.pipeline.scoring.ScoringPipelineConfig +import com.twitter.timelines.clients.predictionservice.PredictionGRPCService +import com.twitter.timelines.clients.predictionservice.PredictionServiceGRPCClient +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class ScoredTweetsScoringPipelineConfig @Inject() ( + scoredTweetsInNetworkCandidatePipelineConfig: ScoredTweetsInNetworkCandidatePipelineConfig, + scoredTweetsUtegCandidatePipelineConfig: ScoredTweetsUtegCandidatePipelineConfig, + scoredTweetsCrMixerCandidatePipelineConfig: ScoredTweetsCrMixerCandidatePipelineConfig, + scoredTweetsFrsCandidatePipelineConfig: ScoredTweetsFrsCandidatePipelineConfig, + predictionGRPCService: PredictionGRPCService, + ancestorFeatureHydrator: AncestorFeatureHydrator, + authorFeatureHydrator: AuthorFeatureHydrator, + earlybirdFeatureHydrator: EarlybirdFeatureHydrator, + metricCenterUserCountingFeatureHydrator: MetricCenterUserCountingFeatureHydrator, + tweetypieContentFeatureHydrator: TweetypieContentFeatureHydrator, + gizmoduckAuthorSafetyFeatureHydrator: GizmoduckAuthorSafetyFeatureHydrator, + graphTwoHopFeatureHydrator: GraphTwoHopFeatureHydrator, + socialGraphServiceFeatureHydrator: SocialGraphServiceFeatureHydrator, + twhinAuthorFollow20220101FeatureHydrator: TwhinAuthorFollow20220101FeatureHydrator, + userFollowedTopicIdsFeatureHydrator: UserFollowedTopicIdsFeatureHydrator, + utegFeatureHydrator: UtegFeatureHydrator, + realGraphViewerAuthorFeatureHydrator: RealGraphViewerAuthorFeatureHydrator, + realGraphViewerRelatedUsersFeatureHydrator: RealGraphViewerRelatedUsersFeatureHydrator, + realTimeInteractionGraphEdgeFeatureHydrator: RealTimeInteractionGraphEdgeFeatureHydrator, + engagementsReceivedByAuthorRealTimeAggregateFeatureHydrator: EngagementsReceivedByAuthorRealTimeAggregateFeatureHydrator, + topicCountryEngagementRealTimeAggregateFeatureHydrator: TopicCountryEngagementRealTimeAggregateFeatureHydrator, + topicEngagementRealTimeAggregateFeatureHydrator: TopicEngagementRealTimeAggregateFeatureHydrator, + tspInferredTopicFeatureHydrator: TSPInferredTopicFeatureHydrator, + tweetCountryEngagementRealTimeAggregateFeatureHydrator: TweetCountryEngagementRealTimeAggregateFeatureHydrator, + tweetEngagementRealTimeAggregateFeatureHydrator: TweetEngagementRealTimeAggregateFeatureHydrator, + twitterListEngagementRealTimeAggregateFeatureHydrator: TwitterListEngagementRealTimeAggregateFeatureHydrator, + userAuthorEngagementRealTimeAggregateFeatureHydrator: UserAuthorEngagementRealTimeAggregateFeatureHydrator, + simClustersEngagementSimilarityFeatureHydrator: SimClustersEngagementSimilarityFeatureHydrator, + phase1EdgeAggregateFeatureHydrator: Phase1EdgeAggregateFeatureHydrator, + phase2EdgeAggregateFeatureHydrator: Phase2EdgeAggregateFeatureHydrator, + statsReceiver: StatsReceiver) + extends ScoringPipelineConfig[ScoredTweetsQuery, TweetCandidate] { + + override val identifier: ScoringPipelineIdentifier = ScoringPipelineIdentifier("ScoredTweets") + + private val nonCachedScoringPipelineScope = AllExceptPipelines( + pipelinesToExclude = Set(CachedScoredTweetsCandidatePipelineConfig.Identifier) + ) + + override val gates: Seq[BaseGate[ScoredTweetsQuery]] = Seq( + NonEmptyCandidatesGate(nonCachedScoringPipelineScope) + ) + + private val earlybirdScorePipelineScope = Set( + scoredTweetsInNetworkCandidatePipelineConfig.identifier, + scoredTweetsUtegCandidatePipelineConfig.identifier, + scoredTweetsFrsCandidatePipelineConfig.identifier + ) + + private val earlybirdScoreOrdering: Ordering[CandidateWithDetails] = + Ordering.by[CandidateWithDetails, Double] { + case ItemCandidateWithDetails(_, _, features) => + -features.getOrElse(EarlybirdScoreFeature, None).getOrElse(0.0) + case _ => throw PipelineFailure(UnexpectedCandidateResult, "Invalid candidate type") + } + + override val selectors: Seq[Selector[ScoredTweetsQuery]] = Seq( + UpdateSortCandidates(SpecificPipelines(earlybirdScorePipelineScope), earlybirdScoreOrdering), + new DropMaxCandidates( + pipelineScope = SpecificPipelines(earlybirdScorePipelineScope), + maxSelector = (query, _, _) => + (query.getQualityFactorCurrentValue(identifier) * + query.params(QualityFactor.MaxTweetsToScoreParam)).toInt + ), + new DropMaxCandidates( + pipelineScope = SpecificPipelines(scoredTweetsCrMixerCandidatePipelineConfig.identifier), + maxSelector = (query, _, _) => + (query.getQualityFactorCurrentValue(identifier) * + query.params(QualityFactor.CrMixerMaxTweetsToScoreParam)).toInt + ), + // Select candidates for Heavy Ranker Feature Hydration and Scoring + InsertAppendResults(nonCachedScoringPipelineScope) + ) + + override val preScoringFeatureHydrationPhase1: Seq[ + BaseCandidateFeatureHydrator[ScoredTweetsQuery, TweetCandidate, _] + ] = Seq( + ancestorFeatureHydrator, + authorFeatureHydrator, + earlybirdFeatureHydrator, + gizmoduckAuthorSafetyFeatureHydrator, + graphTwoHopFeatureHydrator, + metricCenterUserCountingFeatureHydrator, + socialGraphServiceFeatureHydrator, + TweetMetaDataFeatureHydrator, + tweetypieContentFeatureHydrator, + twhinAuthorFollow20220101FeatureHydrator, + userFollowedTopicIdsFeatureHydrator, + utegFeatureHydrator, + realTimeInteractionGraphEdgeFeatureHydrator, + realGraphViewerAuthorFeatureHydrator, + // real time aggregates + engagementsReceivedByAuthorRealTimeAggregateFeatureHydrator, + simClustersEngagementSimilarityFeatureHydrator, + tspInferredTopicFeatureHydrator, + tweetCountryEngagementRealTimeAggregateFeatureHydrator, + tweetEngagementRealTimeAggregateFeatureHydrator, + twitterListEngagementRealTimeAggregateFeatureHydrator, + userAuthorEngagementRealTimeAggregateFeatureHydrator, + // offline aggregates + phase1EdgeAggregateFeatureHydrator + ) + + override val preScoringFeatureHydrationPhase2: Seq[ + BaseCandidateFeatureHydrator[ScoredTweetsQuery, TweetCandidate, _] + ] = Seq( + realGraphViewerRelatedUsersFeatureHydrator, + TimeFeaturesHydrator, + topicCountryEngagementRealTimeAggregateFeatureHydrator, + topicEngagementRealTimeAggregateFeatureHydrator, + phase2EdgeAggregateFeatureHydrator + ) + + private val homeNaviModelDataRecordScorer: Scorer[ScoredTweetsQuery, TweetCandidate] = { + val modelClient = new PredictionServiceGRPCClient( + service = predictionGRPCService, + statsReceiver = statsReceiver, + requestBatchSize = HomeNaviModelDataRecordScorer.RequestBatchSize, + useCompact = false + ) + HomeNaviModelDataRecordScorer( + identifier = ScorerIdentifier("HomeNaviModel"), + modelClient = modelClient, + candidateFeatures = AllFeatures(), + resultFeatures = HomeNaviModelDataRecordScorer.PredictedScoreFeatures.toSet, + statsReceiver = statsReceiver + ) + } + + override val scorers: Seq[Scorer[ScoredTweetsQuery, TweetCandidate]] = + Seq(homeNaviModelDataRecordScorer) +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/scoring_pipeline/ScoredTweetsWeightedScoresSumScoringPipelineConfig.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/scoring_pipeline/ScoredTweetsWeightedScoresSumScoringPipelineConfig.scala new file mode 100644 index 0000000000..459bc4d44e --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/scoring_pipeline/ScoredTweetsWeightedScoresSumScoringPipelineConfig.scala @@ -0,0 +1,34 @@ +package com.twitter.home_mixer.product.scored_tweets.scoring_pipeline + +import com.twitter.home_mixer.product.scored_tweets.candidate_pipeline.CachedScoredTweetsCandidatePipelineConfig +import com.twitter.home_mixer.product.scored_tweets.model.ScoredTweetsQuery +import com.twitter.home_mixer.product.scored_tweets.scorer.WeightedScoresSumScorer +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.component_library.selector.InsertAppendResults +import com.twitter.product_mixer.core.functional_component.common.AllExceptPipelines +import com.twitter.product_mixer.core.functional_component.scorer.Scorer +import com.twitter.product_mixer.core.functional_component.selector.Selector +import com.twitter.product_mixer.core.model.common.identifier.ScoringPipelineIdentifier +import com.twitter.product_mixer.core.pipeline.scoring.ScoringPipelineConfig +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class ScoredTweetsWeightedScoresSumScoringPipelineConfig @Inject() ( + weightedScoresSumScorer: WeightedScoresSumScorer) + extends ScoringPipelineConfig[ScoredTweetsQuery, TweetCandidate] { + + override val identifier: ScoringPipelineIdentifier = + ScoringPipelineIdentifier("ScoredTweetsWeightedScoresSum") + + override val selectors: Seq[Selector[ScoredTweetsQuery]] = Seq( + InsertAppendResults( + AllExceptPipelines(pipelinesToExclude = + Set(CachedScoredTweetsCandidatePipelineConfig.Identifier)) + ) + ) + + override val scorers: Seq[Scorer[ScoredTweetsQuery, TweetCandidate]] = Seq( + weightedScoresSumScorer + ) +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/side_effect/BUILD.bazel b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/side_effect/BUILD.bazel new file mode 100644 index 0000000000..f6c404ec65 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/side_effect/BUILD.bazel @@ -0,0 +1,35 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + strict_deps = True, + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/javax/inject:javax.inject", + "finagle/finagle-mysql/src/main/scala", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/functional_component/feature_hydrator/adapters/non_ml_features", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/model", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/model/request", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/param", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/model", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/param", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/scorer", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/service", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/util", + "home-mixer/thrift/src/main/thrift:thrift-scala", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/scorer/tweet_tlx", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/side_effect", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/feature/featuremap/datarecord", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/functional_component/marshaller/response/urt", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/product", + "servo/repo/src/main/scala", + "servo/util/src/main/scala", + "src/scala/com/twitter/timelines/prediction/common/adapters", + "src/thrift/com/twitter/timelines/suggests/common:data_record_metadata-scala", + "src/thrift/com/twitter/timelines/suggests/common:poly_data_record-java", + "timelines/ml:pldr-client", + "timelines/ml:pldr-conversion", + "timelines/ml/cont_train/common/domain/src/main/scala/com/twitter/timelines/ml/cont_train/common/domain/non_scalding", + "timelines/src/main/scala/com/twitter/timelines/util/stats", + ], +) diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/side_effect/CachedScoredTweetsSideEffect.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/side_effect/CachedScoredTweetsSideEffect.scala new file mode 100644 index 0000000000..abe076c54a --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/side_effect/CachedScoredTweetsSideEffect.scala @@ -0,0 +1,123 @@ +package com.twitter.home_mixer.product.scored_tweets.side_effect + +import com.twitter.home_mixer.model.HomeFeatures.AncestorsFeature +import com.twitter.home_mixer.model.HomeFeatures.AuthorIdFeature +import com.twitter.home_mixer.model.HomeFeatures.AuthorIsBlueVerifiedFeature +import com.twitter.home_mixer.model.HomeFeatures.CachedCandidatePipelineIdentifierFeature +import com.twitter.home_mixer.model.HomeFeatures.DirectedAtUserIdFeature +import com.twitter.home_mixer.model.HomeFeatures.FavoritedByUserIdsFeature +import com.twitter.home_mixer.model.HomeFeatures.FollowedByUserIdsFeature +import com.twitter.home_mixer.model.HomeFeatures.InNetworkFeature +import com.twitter.home_mixer.model.HomeFeatures.InReplyToTweetIdFeature +import com.twitter.home_mixer.model.HomeFeatures.InReplyToUserIdFeature +import com.twitter.home_mixer.model.HomeFeatures.IsRetweetFeature +import com.twitter.home_mixer.model.HomeFeatures.LastScoredTimestampMsFeature +import com.twitter.home_mixer.model.HomeFeatures.QuotedTweetIdFeature +import com.twitter.home_mixer.model.HomeFeatures.QuotedUserIdFeature +import com.twitter.home_mixer.model.HomeFeatures.ScoreFeature +import com.twitter.home_mixer.model.HomeFeatures.SourceTweetIdFeature +import com.twitter.home_mixer.model.HomeFeatures.SourceUserIdFeature +import com.twitter.home_mixer.model.HomeFeatures.SuggestTypeFeature +import com.twitter.home_mixer.model.HomeFeatures.TopicContextFunctionalityTypeFeature +import com.twitter.home_mixer.model.HomeFeatures.TopicIdSocialContextFeature +import com.twitter.home_mixer.model.HomeFeatures.TweetUrlsFeature +import com.twitter.home_mixer.model.HomeFeatures.WeightedModelScoreFeature +import com.twitter.home_mixer.product.scored_tweets.model.ScoredTweetsResponse +import com.twitter.home_mixer.product.scored_tweets.param.ScoredTweetsParam.CachedScoredTweets +import com.twitter.home_mixer.service.HomeMixerAlertConfig +import com.twitter.home_mixer.{thriftscala => hmt} +import com.twitter.product_mixer.core.functional_component.marshaller.response.urt.metadata.TopicContextFunctionalityTypeMarshaller +import com.twitter.product_mixer.core.functional_component.side_effect.PipelineResultSideEffect +import com.twitter.product_mixer.core.model.common.identifier.SideEffectIdentifier +import com.twitter.product_mixer.core.model.common.presentation.CandidateWithDetails +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.servo.cache.TtlCache +import com.twitter.stitch.Stitch +import com.twitter.util.Time + +import javax.inject.Inject +import javax.inject.Singleton + +@Singleton +class CachedScoredTweetsSideEffect @Inject() ( + scoredTweetsCache: TtlCache[Long, hmt.CachedScoredTweets]) + extends PipelineResultSideEffect[PipelineQuery, ScoredTweetsResponse] { + + override val identifier: SideEffectIdentifier = SideEffectIdentifier("CachedScoredTweets") + + private val MaxTweetsToCache = 1000 + + def buildCachedScoredTweets( + candidates: Seq[CandidateWithDetails] + ): hmt.CachedScoredTweets = { + val tweets = candidates.map { candidate => + val favoritedByUserIds = candidate.features.getOrElse(FavoritedByUserIdsFeature, Seq.empty) + val followedByUserIds = candidate.features.getOrElse(FollowedByUserIdsFeature, Seq.empty) + val ancestors = candidate.features.getOrElse(AncestorsFeature, Seq.empty) + val urlsList = candidate.features.getOrElse(TweetUrlsFeature, Seq.empty) + + hmt.CachedScoredTweet( + tweetId = candidate.candidateIdLong, + // Cache the model score instead of the final score because rescoring is per-request + score = candidate.features.getOrElse(WeightedModelScoreFeature, None), + lastScoredTimestampMs = Some( + candidate.features + .getOrElse(LastScoredTimestampMsFeature, None) + .getOrElse(Time.now.inMilliseconds)), + candidatePipelineIdentifier = Some( + candidate.features + .getOrElse(CachedCandidatePipelineIdentifierFeature, None) + .getOrElse(candidate.source.name)), + userId = candidate.features.getOrElse(AuthorIdFeature, None), + sourceTweetId = candidate.features.getOrElse(SourceTweetIdFeature, None), + sourceUserId = candidate.features.getOrElse(SourceUserIdFeature, None), + isRetweet = Some(candidate.features.getOrElse(IsRetweetFeature, false)), + isInNetwork = Some(candidate.features.getOrElse(InNetworkFeature, false)), + suggestType = candidate.features.getOrElse(SuggestTypeFeature, None), + quotedTweetId = candidate.features.getOrElse(QuotedTweetIdFeature, None), + quotedUserId = candidate.features.getOrElse(QuotedUserIdFeature, None), + inReplyToTweetId = candidate.features.getOrElse(InReplyToTweetIdFeature, None), + inReplyToUserId = candidate.features.getOrElse(InReplyToUserIdFeature, None), + directedAtUserId = candidate.features.getOrElse(DirectedAtUserIdFeature, None), + favoritedByUserIds = if (favoritedByUserIds.nonEmpty) Some(favoritedByUserIds) else None, + followedByUserIds = if (followedByUserIds.nonEmpty) Some(followedByUserIds) else None, + topicId = candidate.features.getOrElse(TopicIdSocialContextFeature, None), + topicFunctionalityType = candidate.features + .getOrElse(TopicContextFunctionalityTypeFeature, None).map( + TopicContextFunctionalityTypeMarshaller(_)), + ancestors = if (ancestors.nonEmpty) Some(ancestors) else None, + urlsList = if (urlsList.nonEmpty) Some(urlsList) else None, + authorIsBlueVerified = + Some(candidate.features.getOrElse(AuthorIsBlueVerifiedFeature, false)) + ) + } + + hmt.CachedScoredTweets(tweets = tweets) + } + + final override def apply( + inputs: PipelineResultSideEffect.Inputs[PipelineQuery, ScoredTweetsResponse] + ): Stitch[Unit] = { + val candidates = (inputs.selectedCandidates ++ inputs.remainingCandidates).filter { candidate => + val score = candidate.features.getOrElse(ScoreFeature, None) + score.exists(_ > 0.0) + } + + val truncatedCandidates = + if (candidates.size > MaxTweetsToCache) + candidates + .sortBy(-_.features.getOrElse(ScoreFeature, None).getOrElse(0.0)) + .take(MaxTweetsToCache) + else candidates + + if (truncatedCandidates.nonEmpty) { + val ttl = inputs.query.params(CachedScoredTweets.TTLParam) + val scoredTweets = buildCachedScoredTweets(truncatedCandidates) + Stitch.callFuture(scoredTweetsCache.set(inputs.query.getRequiredUserId, scoredTweets, ttl)) + } else Stitch.Unit + } + + override val alerts = Seq( + HomeMixerAlertConfig.BusinessHours.defaultSuccessRateAlert(99.4) + ) +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/side_effect/ScribeServedCommonFeaturesAndCandidateFeaturesSideEffect.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/side_effect/ScribeServedCommonFeaturesAndCandidateFeaturesSideEffect.scala new file mode 100644 index 0000000000..2599832ccd --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/product/scored_tweets/side_effect/ScribeServedCommonFeaturesAndCandidateFeaturesSideEffect.scala @@ -0,0 +1,213 @@ +package com.twitter.home_mixer.product.scored_tweets.side_effect + +import com.twitter.conversions.DurationOps._ +import com.twitter.finagle.mysql.Client +import com.twitter.finagle.mysql.Transactions +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.finagle.util.DefaultTimer +import com.twitter.home_mixer.functional_component.feature_hydrator.adapters.non_ml_features.NonMLCandidateFeatures +import com.twitter.home_mixer.functional_component.feature_hydrator.adapters.non_ml_features.NonMLCandidateFeaturesAdapter +import com.twitter.home_mixer.functional_component.feature_hydrator.adapters.non_ml_features.NonMLCommonFeatures +import com.twitter.home_mixer.functional_component.feature_hydrator.adapters.non_ml_features.NonMLCommonFeaturesAdapter +import com.twitter.home_mixer.model.HomeFeatures.ServedRequestIdFeature +import com.twitter.home_mixer.model.HomeFeatures.SourceTweetIdFeature +import com.twitter.home_mixer.param.HomeMixerFlagName.DataRecordMetadataStoreConfigsYmlFlag +import com.twitter.home_mixer.param.HomeMixerInjectionNames.CandidateFeaturesScribeEventPublisher +import com.twitter.home_mixer.param.HomeMixerInjectionNames.CommonFeaturesScribeEventPublisher +import com.twitter.home_mixer.param.HomeMixerInjectionNames.MinimumFeaturesScribeEventPublisher +import com.twitter.home_mixer.product.scored_tweets.model.ScoredTweetsQuery +import com.twitter.home_mixer.product.scored_tweets.model.ScoredTweetsResponse +import com.twitter.home_mixer.product.scored_tweets.scorer.CandidateFeaturesDataRecordFeature +import com.twitter.home_mixer.product.scored_tweets.scorer.CommonFeaturesDataRecordFeature +import com.twitter.home_mixer.product.scored_tweets.scorer.HomeNaviModelDataRecordScorer.PredictedScoreFeatures +import com.twitter.home_mixer.util.CandidatesUtil.getOriginalAuthorId +import com.twitter.inject.annotations.Flag +import com.twitter.logpipeline.client.common.EventPublisher +import com.twitter.ml.api.DataRecordMerger +import com.twitter.product_mixer.core.feature.featuremap.datarecord.DataRecordConverter +import com.twitter.product_mixer.core.feature.featuremap.datarecord.SpecificFeatures +import com.twitter.product_mixer.core.functional_component.side_effect.PipelineResultSideEffect +import com.twitter.product_mixer.core.model.common.identifier.SideEffectIdentifier +import com.twitter.stitch.Stitch +import com.twitter.timelines.ml.cont_train.common.domain.non_scalding.CandidateAndCommonFeaturesStreamingUtils +import com.twitter.timelines.ml.pldr.client.MysqlClientUtils +import com.twitter.timelines.ml.pldr.client.VersionedMetadataCacheClient +import com.twitter.timelines.ml.pldr.conversion.VersionIdAndFeatures +import com.twitter.timelines.suggests.common.data_record_metadata.{thriftscala => drmd} +import com.twitter.timelines.suggests.common.poly_data_record.{thriftjava => pldr} +import com.twitter.timelines.util.stats.OptionObserver +import com.twitter.util.Time +import com.twitter.util.Try +import com.twitter.util.logging.Logging +import javax.inject.Inject +import javax.inject.Named +import javax.inject.Singleton +import scala.collection.JavaConverters._ + +/** + * (1) Scribe common features sent to prediction service + some other features as PLDR format into logs + * (2) Scribe candidate features sent to prediction service + some other features as PLDR format into another logs + */ +@Singleton +class ScribeServedCommonFeaturesAndCandidateFeaturesSideEffect @Inject() ( + @Flag(DataRecordMetadataStoreConfigsYmlFlag) dataRecordMetadataStoreConfigsYml: String, + @Named(CommonFeaturesScribeEventPublisher) commonFeaturesScribeEventPublisher: EventPublisher[ + pldr.PolyDataRecord + ], + @Named(CandidateFeaturesScribeEventPublisher) candidateFeaturesScribeEventPublisher: EventPublisher[ + pldr.PolyDataRecord + ], + @Named(MinimumFeaturesScribeEventPublisher) minimumFeaturesScribeEventPublisher: EventPublisher[ + pldr.PolyDataRecord + ], + statsReceiver: StatsReceiver, +) extends PipelineResultSideEffect[ScoredTweetsQuery, ScoredTweetsResponse] + with Logging { + + override val identifier: SideEffectIdentifier = SideEffectIdentifier( + "ScribeServedCommonFeaturesAndCandidateFeatures") + + private val drMerger = new DataRecordMerger + private val postScoringCandidateFeatures = SpecificFeatures(PredictedScoreFeatures.toSet) + private val postScoringCandidateFeaturesDataRecordAdapter = new DataRecordConverter( + postScoringCandidateFeatures) + + private val scopedStatsReceiver = statsReceiver.scope(getClass.getSimpleName) + private val metadataFetchFailedCounter = scopedStatsReceiver.counter("metadataFetchFailed") + private val commonFeaturesScribeCounter = scopedStatsReceiver.counter("commonFeaturesScribe") + private val commonFeaturesPLDROptionObserver = OptionObserver( + scopedStatsReceiver.scope("commonFeaturesPLDR")) + private val candidateFeaturesScribeCounter = + scopedStatsReceiver.counter("candidateFeaturesScribe") + private val candidateFeaturesPLDROptionObserver = OptionObserver( + scopedStatsReceiver.scope("candidateFeaturesPLDR")) + private val minimumFeaturesPLDROptionObserver = OptionObserver( + scopedStatsReceiver.scope("minimumFeaturesPLDR")) + private val minimumFeaturesScribeCounter = + scopedStatsReceiver.counter("minimumFeaturesScribe") + + lazy private val dataRecordMetadataStoreClient: Option[Client with Transactions] = + Try { + MysqlClientUtils.mysqlClientProvider( + MysqlClientUtils.parseConfigFromYaml(dataRecordMetadataStoreConfigsYml)) + }.onFailure { e => info(s"Error building MySQL client: $e") }.toOption + + lazy private val versionedMetadataCacheClientOpt: Option[ + VersionedMetadataCacheClient[Map[drmd.FeaturesCategory, Option[VersionIdAndFeatures]]] + ] = + dataRecordMetadataStoreClient.map { mysqlClient => + new VersionedMetadataCacheClient[Map[drmd.FeaturesCategory, Option[VersionIdAndFeatures]]]( + maximumSize = 1, + expireDurationOpt = None, + mysqlClient = mysqlClient, + transform = CandidateAndCommonFeaturesStreamingUtils.metadataTransformer, + statsReceiver = statsReceiver + ) + } + + versionedMetadataCacheClientOpt.foreach { versionedMetadataCacheClient => + versionedMetadataCacheClient + .metadataFetchTimerTask( + CandidateAndCommonFeaturesStreamingUtils.metadataFetchKey, + metadataFetchTimer = DefaultTimer, + metadataFetchInterval = 90.seconds, + metadataFetchFailedCounter = metadataFetchFailedCounter + ) + } + + override def apply( + inputs: PipelineResultSideEffect.Inputs[ScoredTweetsQuery, ScoredTweetsResponse] + ): Stitch[Unit] = { + Stitch.value { + val servedTimestamp: Long = Time.now.inMilliseconds + val nonMLCommonFeatures = NonMLCommonFeatures( + userId = inputs.query.getRequiredUserId, + predictionRequestId = + inputs.query.features.flatMap(_.getOrElse(ServedRequestIdFeature, None)), + servedTimestamp = servedTimestamp + ) + val nonMLCommonFeaturesDataRecord = + NonMLCommonFeaturesAdapter.adaptToDataRecords(nonMLCommonFeatures).asScala.head + + /** + * Steps of scribing common features + * (1) fetch common features as data record + * (2) extract additional feature as data record, e.g. predictionRequestId which is used as join key in downstream jobs + * (3) merge two data records above and convert the merged data record to pldr + * (4) publish pldr + */ + val commonFeaturesDataRecordOpt = + inputs.selectedCandidates.headOption.map(_.features.get(CommonFeaturesDataRecordFeature)) + val commonFeaturesPLDROpt = commonFeaturesDataRecordOpt.flatMap { commonFeaturesDataRecord => + drMerger.merge(commonFeaturesDataRecord, nonMLCommonFeaturesDataRecord) + + CandidateAndCommonFeaturesStreamingUtils.commonFeaturesToPolyDataRecord( + versionedMetadataCacheClientOpt = versionedMetadataCacheClientOpt, + commonFeatures = commonFeaturesDataRecord, + valueFormat = pldr.PolyDataRecord._Fields.LITE_COMPACT_DATA_RECORD + ) + } + + commonFeaturesPLDROptionObserver(commonFeaturesPLDROpt).foreach { pldr => + commonFeaturesScribeEventPublisher.publish(pldr) + commonFeaturesScribeCounter.incr() + } + + /** + * steps of scribing candidate features + * (1) fetch candidate features as data record + * (2) extract additional features (mostly non ML features including predicted scores, predictionRequestId, userId, tweetId) + * (3) merge data records and convert the merged data record into pldr + * (4) publish pldr + */ + inputs.selectedCandidates.foreach { candidate => + val candidateFeaturesDataRecord = candidate.features.get(CandidateFeaturesDataRecordFeature) + + /** + * extract predicted scores as data record and merge it into original data record + */ + val postScoringCandidateFeaturesDataRecord = + postScoringCandidateFeaturesDataRecordAdapter.toDataRecord(candidate.features) + drMerger.merge(candidateFeaturesDataRecord, postScoringCandidateFeaturesDataRecord) + + /** + * extract non ML common features as data record and merge it into original data record + */ + drMerger.merge(candidateFeaturesDataRecord, nonMLCommonFeaturesDataRecord) + + /** + * extract non ML candidate features as data record and merge it into original data record + */ + val nonMLCandidateFeatures = NonMLCandidateFeatures( + tweetId = candidate.candidateIdLong, + sourceTweetId = candidate.features.getOrElse(SourceTweetIdFeature, None), + originalAuthorId = getOriginalAuthorId(candidate.features) + ) + val nonMLCandidateFeaturesDataRecord = + NonMLCandidateFeaturesAdapter.adaptToDataRecords(nonMLCandidateFeatures).asScala.head + drMerger.merge(candidateFeaturesDataRecord, nonMLCandidateFeaturesDataRecord) + + val candidateFeaturesPLDROpt = + CandidateAndCommonFeaturesStreamingUtils.candidateFeaturesToPolyDataRecord( + versionedMetadataCacheClientOpt = versionedMetadataCacheClientOpt, + candidateFeatures = candidateFeaturesDataRecord, + valueFormat = pldr.PolyDataRecord._Fields.LITE_COMPACT_DATA_RECORD + ) + + candidateFeaturesPLDROptionObserver(candidateFeaturesPLDROpt).foreach { pldr => + candidateFeaturesScribeEventPublisher.publish(pldr) + candidateFeaturesScribeCounter.incr() + } + + // scribe minimum features which are used to join labels from client events. + val minimumFeaturesPLDROpt = candidateFeaturesPLDROpt + .map(CandidateAndCommonFeaturesStreamingUtils.extractMinimumFeaturesFromPldr) + .map(pldr.PolyDataRecord.dataRecord) + minimumFeaturesPLDROptionObserver(minimumFeaturesPLDROpt).foreach { pldr => + minimumFeaturesScribeEventPublisher.publish(pldr) + minimumFeaturesScribeCounter.incr() + } + } + } + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/service/BUILD.bazel b/home-mixer/server/src/main/scala/com/twitter/home_mixer/service/BUILD.bazel new file mode 100644 index 0000000000..c0211ff723 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/service/BUILD.bazel @@ -0,0 +1,15 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + strict_deps = True, + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/javax/inject:javax.inject", + "configapi/configapi-core", + "home-mixer/thrift/src/main/thrift:thrift-scala", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/pipeline/product", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/product", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/product/registry", + "stitch/stitch-core", + ], +) diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/service/HomeMixerAccessPolicy.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/service/HomeMixerAccessPolicy.scala new file mode 100644 index 0000000000..853f4b56a6 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/service/HomeMixerAccessPolicy.scala @@ -0,0 +1,13 @@ +package com.twitter.home_mixer.service + +import com.twitter.product_mixer.core.functional_component.common.access_policy.AccessPolicy +import com.twitter.product_mixer.core.functional_component.common.access_policy.AllowedLdapGroups + +object HomeMixerAccessPolicy { + + /** + * Access policies can be configured on a product-by-product basis but you may also want products + * to have a common policy. + */ + val DefaultHomeMixerAccessPolicy: Set[AccessPolicy] = Set(AllowedLdapGroups(Set.empty[String])) +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/service/HomeMixerAlertConfig.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/service/HomeMixerAlertConfig.scala new file mode 100644 index 0000000000..93c361516c --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/service/HomeMixerAlertConfig.scala @@ -0,0 +1,65 @@ +package com.twitter.home_mixer.service + +import com.twitter.conversions.DurationOps._ +import com.twitter.product_mixer.core.functional_component.common.alert.Destination +import com.twitter.product_mixer.core.functional_component.common.alert.EmptyResponseRateAlert +import com.twitter.product_mixer.core.functional_component.common.alert.LatencyAlert +import com.twitter.product_mixer.core.functional_component.common.alert.NotificationGroup +import com.twitter.product_mixer.core.functional_component.common.alert.P99 +import com.twitter.product_mixer.core.functional_component.common.alert.Percentile +import com.twitter.product_mixer.core.functional_component.common.alert.SuccessRateAlert +import com.twitter.product_mixer.core.functional_component.common.alert.predicate.TriggerIfAbove +import com.twitter.product_mixer.core.functional_component.common.alert.predicate.TriggerIfBelow +import com.twitter.product_mixer.core.functional_component.common.alert.predicate.TriggerIfLatencyAbove +import com.twitter.util.Duration + +/** + * Notifications (email, pagerduty, etc) can be specific per-alert but it is common for multiple + * products to share notification configuration. + */ +object HomeMixerAlertConfig { + val DefaultNotificationGroup: NotificationGroup = NotificationGroup( + warn = Destination(emails = Seq("")), + critical = Destination(emails = Seq("")) + ) + + object BusinessHours { + val DefaultNotificationGroup: NotificationGroup = NotificationGroup( + warn = Destination(emails = Seq("")), + critical = Destination(emails = Seq("")) + ) + + def defaultEmptyResponseRateAlert(warnThreshold: Double = 50, criticalThreshold: Double = 80) = + EmptyResponseRateAlert( + notificationGroup = DefaultNotificationGroup, + warnPredicate = TriggerIfAbove(warnThreshold), + criticalPredicate = TriggerIfAbove(criticalThreshold) + ) + + def defaultSuccessRateAlert( + threshold: Double = 99.5, + warnDatapointsPastThreshold: Int = 20, + criticalDatapointsPastThreshold: Int = 30, + duration: Int = 30 + ) = SuccessRateAlert( + notificationGroup = DefaultNotificationGroup, + warnPredicate = TriggerIfBelow(threshold, warnDatapointsPastThreshold, duration), + criticalPredicate = TriggerIfBelow(threshold, criticalDatapointsPastThreshold, duration), + ) + + def defaultLatencyAlert( + latencyThreshold: Duration = 200.millis, + warningDatapointsPastThreshold: Int = 15, + criticalDatapointsPastThreshold: Int = 30, + duration: Int = 30, + percentile: Percentile = P99 + ): LatencyAlert = LatencyAlert( + notificationGroup = DefaultNotificationGroup, + percentile = percentile, + warnPredicate = + TriggerIfLatencyAbove(latencyThreshold, warningDatapointsPastThreshold, duration), + criticalPredicate = + TriggerIfLatencyAbove(latencyThreshold, criticalDatapointsPastThreshold, duration) + ) + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/service/ScoredTweetsService.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/service/ScoredTweetsService.scala new file mode 100644 index 0000000000..158e5ee450 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/service/ScoredTweetsService.scala @@ -0,0 +1,24 @@ +package com.twitter.home_mixer.service + +import com.twitter.home_mixer.{thriftscala => t} +import com.twitter.product_mixer.core.model.marshalling.request.Request +import com.twitter.product_mixer.core.pipeline.product.ProductPipelineRequest +import com.twitter.product_mixer.core.product.registry.ProductPipelineRegistry +import com.twitter.stitch.Stitch +import com.twitter.timelines.configapi.Params +import javax.inject.Inject +import javax.inject.Singleton +import scala.reflect.runtime.universe._ + +@Singleton +class ScoredTweetsService @Inject() (productPipelineRegistry: ProductPipelineRegistry) { + + def getScoredTweetsResponse[RequestType <: Request]( + request: RequestType, + params: Params + )( + implicit requestTypeTag: TypeTag[RequestType] + ): Stitch[t.ScoredTweetsResponse] = productPipelineRegistry + .getProductPipeline[RequestType, t.ScoredTweetsResponse](request.product) + .process(ProductPipelineRequest(request, params)) +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/store/BUILD.bazel b/home-mixer/server/src/main/scala/com/twitter/home_mixer/store/BUILD.bazel new file mode 100644 index 0000000000..89535dbd32 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/store/BUILD.bazel @@ -0,0 +1,17 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + strict_deps = True, + tags = ["bazel-compatible"], + dependencies = [ + "3rdparty/jvm/com/twitter/bijection:scrooge", + "3rdparty/jvm/com/twitter/storehaus:core", + "home-mixer/server/src/main/scala/com/twitter/home_mixer/util", + "src/thrift/com/twitter/service/metastore/gen:thrift-java", + "src/thrift/com/twitter/service/metastore/gen:thrift-scala", + "src/thrift/com/twitter/wtf/candidate:wtf-candidate-scala", + "stitch/stitch-core", + "storage/clients/manhattan/client/src/main/scala", + "timelinemixer/server/src/main/scala/com/twitter/timelinemixer/injection/repository/uss", + ], +) diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/store/RealGraphInNetworkScoresStore.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/store/RealGraphInNetworkScoresStore.scala new file mode 100644 index 0000000000..ce0b182be2 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/store/RealGraphInNetworkScoresStore.scala @@ -0,0 +1,34 @@ +package com.twitter.home_mixer.store + +import com.twitter.bijection.Injection +import com.twitter.home_mixer.store.ManhattanRealGraphKVDescriptor._ +import com.twitter.stitch.Stitch +import com.twitter.storage.client.manhattan.bijections.Bijections +import com.twitter.storage.client.manhattan.bijections.Bijections.BinaryScalaInjection +import com.twitter.storage.client.manhattan.kv.ManhattanKVEndpoint +import com.twitter.storage.client.manhattan.kv.impl.ReadOnlyKeyDescriptor +import com.twitter.storage.client.manhattan.kv.impl.ValueDescriptor +import com.twitter.storehaus.ReadableStore +import com.twitter.util.Future +import com.twitter.wtf.candidate.{thriftscala => wtf} + +object ManhattanRealGraphKVDescriptor { + implicit val byteArray2Buf = Bijections.BytesBijection + + val realGraphDatasetName = "real_graph_scores_in" + val keyInjection = Injection.connect[Long, Array[Byte]].andThen(Bijections.BytesInjection) + val keyDesc = ReadOnlyKeyDescriptor(keyInjection) + val valueDesc = ValueDescriptor(BinaryScalaInjection(wtf.CandidateSeq)) + val realGraphDatasetKey = keyDesc.withDataset(realGraphDatasetName) +} + +/** + * Hydrates real graph in network scores for a viewer + */ +class RealGraphInNetworkScoresStore(manhattanKVEndpoint: ManhattanKVEndpoint) + extends ReadableStore[Long, Seq[wtf.Candidate]] { + + override def get(viewerId: Long): Future[Option[Seq[wtf.Candidate]]] = Stitch + .run(manhattanKVEndpoint.get(realGraphDatasetKey.withPkey(viewerId), valueDesc)) + .map(_.map(mhResponse => mhResponse.contents.candidates)) +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/store/UserLanguagesStore.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/store/UserLanguagesStore.scala new file mode 100644 index 0000000000..3f78ddb876 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/store/UserLanguagesStore.scala @@ -0,0 +1,47 @@ +package com.twitter.home_mixer.store + +import com.twitter.bijection.Injection +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.home_mixer.store.ManhattanUserLanguagesKVDescriptor._ +import com.twitter.home_mixer.util.LanguageUtil +import com.twitter.search.common.constants.{thriftscala => scc} +import com.twitter.service.metastore.gen.{thriftscala => smg} +import com.twitter.stitch.Stitch +import com.twitter.storage.client.manhattan.bijections.Bijections +import com.twitter.storage.client.manhattan.kv.ManhattanKVEndpoint +import com.twitter.storage.client.manhattan.kv.ManhattanValue +import com.twitter.storage.client.manhattan.kv.impl.ReadOnlyKeyDescriptor +import com.twitter.storage.client.manhattan.kv.impl.ValueDescriptor +import com.twitter.storehaus.ReadableStore +import com.twitter.util.Future + +object ManhattanUserLanguagesKVDescriptor { + val userLanguagesDatasetName = "languages" + val keyInjection = Injection.connect[Long, Array[Byte]].andThen(Bijections.BytesInjection) + val keyDescriptor = ReadOnlyKeyDescriptor(keyInjection) + val valueDescriptor = ValueDescriptor(Bijections.BinaryScalaInjection(smg.UserLanguages)) + val userLanguagesDatasetKey = keyDescriptor.withDataset(userLanguagesDatasetName) +} + +class UserLanguagesStore( + manhattanKVEndpoint: ManhattanKVEndpoint, + statsReceiver: StatsReceiver) + extends ReadableStore[Long, Seq[scc.ThriftLanguage]] { + + private val scopedStatsReceiver = statsReceiver.scope(getClass.getSimpleName) + private val keyFoundCounter = scopedStatsReceiver.counter("key/found") + private val keyLossCounter = scopedStatsReceiver.counter("key/loss") + + override def get(viewerId: Long): Future[Option[Seq[scc.ThriftLanguage]]] = + Stitch + .run( + manhattanKVEndpoint.get(key = userLanguagesDatasetKey.withPkey(viewerId), valueDescriptor)) + .map { + case Some(mhResponse: ManhattanValue[smg.UserLanguages]) => + keyFoundCounter.incr() + Some(LanguageUtil.computeLanguages(mhResponse.contents)) + case _ => + keyLossCounter.incr() + None + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/util/BUILD.bazel b/home-mixer/server/src/main/scala/com/twitter/home_mixer/util/BUILD.bazel new file mode 100644 index 0000000000..2330dd5417 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/util/BUILD.bazel @@ -0,0 +1,22 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + strict_deps = True, + tags = ["bazel-compatible"], + dependencies = [ + "home-mixer/server/src/main/scala/com/twitter/home_mixer/model", + "home-mixer/thrift/src/main/thrift:thrift-scala", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/feature_hydrator/query/impressed_tweets", + "product-mixer/component-library/src/main/scala/com/twitter/product_mixer/component_library/model/candidate", + "product-mixer/core/src/main/scala/com/twitter/product_mixer/core/model/common/presentation", + "servo/repo/src/main/scala", + "snowflake/src/main/scala/com/twitter/snowflake/id", + "src/java/com/twitter/search/common/util/lang", + "src/scala/com/twitter/ml/api/util", + "src/thrift/com/twitter/search/common:constants-java", + "src/thrift/com/twitter/search/common:constants-scala", + "src/thrift/com/twitter/service/metastore/gen:thrift-java", + "src/thrift/com/twitter/service/metastore/gen:thrift-scala", + "storage/clients/manhattan/client/src/main/scala", + ], +) diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/util/CachedScoredTweetsHelper.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/util/CachedScoredTweetsHelper.scala new file mode 100644 index 0000000000..740fd03ec9 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/util/CachedScoredTweetsHelper.scala @@ -0,0 +1,49 @@ +package com.twitter.home_mixer.util + +import com.twitter.home_mixer.model.HomeFeatures.CachedScoredTweetsFeature +import com.twitter.home_mixer.{thriftscala => hmt} +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.model.common.identifier.CandidatePipelineIdentifier +import com.twitter.snowflake.id.SnowflakeId +import com.twitter.util.Time + +object CachedScoredTweetsHelper { + + def tweetImpressionsAndCachedScoredTweets( + features: FeatureMap, + candidatePipelineIdentifier: CandidatePipelineIdentifier + ): Seq[Long] = { + val tweetImpressions = TweetImpressionsHelper.tweetImpressions(features) + val cachedScoredTweets = features + .getOrElse(CachedScoredTweetsFeature, Seq.empty) + .filter { tweet => + tweet.candidatePipelineIdentifier.exists( + CandidatePipelineIdentifier(_).equals(candidatePipelineIdentifier)) + }.map(_.tweetId) + + (tweetImpressions ++ cachedScoredTweets).toSeq + } + + def tweetImpressionsAndCachedScoredTweetsInRange( + features: FeatureMap, + candidatePipelineIdentifier: CandidatePipelineIdentifier, + maxNumImpressions: Int, + sinceTime: Time, + untilTime: Time + ): Seq[Long] = + tweetImpressionsAndCachedScoredTweets(features, candidatePipelineIdentifier) + .filter { tweetId => + val creationTime = SnowflakeId.timeFromId(tweetId) + sinceTime <= creationTime && untilTime >= creationTime + }.take(maxNumImpressions) + + def unseenCachedScoredTweets( + features: FeatureMap + ): Seq[hmt.CachedScoredTweet] = { + val seenTweetIds = TweetImpressionsHelper.tweetImpressions(features) + + features + .getOrElse(CachedScoredTweetsFeature, Seq.empty) + .filter(tweet => !seenTweetIds.contains(tweet.tweetId)) + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/util/CandidatesUtil.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/util/CandidatesUtil.scala new file mode 100644 index 0000000000..4bf265fd13 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/util/CandidatesUtil.scala @@ -0,0 +1,105 @@ +package com.twitter.home_mixer.util + +import com.twitter.home_mixer.model.HomeFeatures.AuthorIdFeature +import com.twitter.home_mixer.model.HomeFeatures.FavoritedByUserIdsFeature +import com.twitter.home_mixer.model.HomeFeatures.HasImageFeature +import com.twitter.home_mixer.model.HomeFeatures.IsRetweetFeature +import com.twitter.home_mixer.model.HomeFeatures.MediaUnderstandingAnnotationIdsFeature +import com.twitter.home_mixer.model.HomeFeatures.RepliedByEngagerIdsFeature +import com.twitter.home_mixer.model.HomeFeatures.RetweetedByEngagerIdsFeature +import com.twitter.home_mixer.model.HomeFeatures.ScoreFeature +import com.twitter.home_mixer.model.HomeFeatures.SourceTweetIdFeature +import com.twitter.home_mixer.model.HomeFeatures.SourceUserIdFeature +import com.twitter.product_mixer.component_library.model.candidate.CursorCandidate +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap +import com.twitter.product_mixer.core.model.common.CandidateWithFeatures +import com.twitter.product_mixer.core.model.common.UniversalNoun +import com.twitter.product_mixer.core.model.common.presentation.CandidateWithDetails +import com.twitter.product_mixer.core.model.common.presentation.ItemCandidateWithDetails +import com.twitter.product_mixer.core.model.common.presentation.ModuleCandidateWithDetails +import com.twitter.product_mixer.core.pipeline.PipelineQuery +import com.twitter.product_mixer.core.pipeline.pipeline_failure.PipelineFailure +import com.twitter.product_mixer.core.pipeline.pipeline_failure.UnexpectedCandidateResult + +import scala.reflect.ClassTag + +object CandidatesUtil { + def getItemCandidates(candidates: Seq[CandidateWithDetails]): Seq[ItemCandidateWithDetails] = { + candidates.collect { + case item: ItemCandidateWithDetails if !item.isCandidateType[CursorCandidate] => Seq(item) + case module: ModuleCandidateWithDetails => module.candidates + }.flatten + } + + def getItemCandidatesWithOnlyModuleLast( + candidates: Seq[CandidateWithDetails] + ): Seq[ItemCandidateWithDetails] = { + candidates.collect { + case item: ItemCandidateWithDetails if !item.isCandidateType[CursorCandidate] => item + case module: ModuleCandidateWithDetails => module.candidates.last + } + } + + def containsType[CandidateType <: UniversalNoun[_]]( + candidates: Seq[CandidateWithDetails] + )( + implicit tag: ClassTag[CandidateType] + ): Boolean = candidates.exists { + case ItemCandidateWithDetails(_: CandidateType, _, _) => true + case module: ModuleCandidateWithDetails => + module.candidates.head.isCandidateType[CandidateType]() + case _ => false + } + + def getOriginalAuthorId(candidateFeatures: FeatureMap): Option[Long] = + if (candidateFeatures.getOrElse(IsRetweetFeature, false)) + candidateFeatures.getOrElse(SourceUserIdFeature, None) + else candidateFeatures.getOrElse(AuthorIdFeature, None) + + def getEngagerUserIds( + candidateFeatures: FeatureMap + ): Seq[Long] = { + candidateFeatures.getOrElse(FavoritedByUserIdsFeature, Seq.empty) ++ + candidateFeatures.getOrElse(RetweetedByEngagerIdsFeature, Seq.empty) ++ + candidateFeatures.getOrElse(RepliedByEngagerIdsFeature, Seq.empty) + } + + def getMediaUnderstandingAnnotationIds( + candidateFeatures: FeatureMap + ): Seq[Long] = { + if (candidateFeatures.get(HasImageFeature)) + candidateFeatures.getOrElse(MediaUnderstandingAnnotationIdsFeature, Seq.empty) + else Seq.empty + } + + def getTweetIdAndSourceId(candidate: CandidateWithFeatures[TweetCandidate]): Seq[Long] = + Seq(candidate.candidate.id) ++ candidate.features.getOrElse(SourceTweetIdFeature, None) + + def isAuthoredByViewer(query: PipelineQuery, candidateFeatures: FeatureMap): Boolean = + candidateFeatures.getOrElse(AuthorIdFeature, None).contains(query.getRequiredUserId) || + (candidateFeatures.getOrElse(IsRetweetFeature, false) && + candidateFeatures.getOrElse(SourceUserIdFeature, None).contains(query.getRequiredUserId)) + + val reverseChronTweetsOrdering: Ordering[CandidateWithDetails] = + Ordering.by[CandidateWithDetails, Long] { + case ItemCandidateWithDetails(candidate: TweetCandidate, _, _) => -candidate.id + case ModuleCandidateWithDetails(candidates, _, _) if candidates.nonEmpty => + -candidates.last.candidateIdLong + case _ => throw PipelineFailure(UnexpectedCandidateResult, "Invalid candidate type") + } + + val scoreOrdering: Ordering[CandidateWithDetails] = Ordering.by[CandidateWithDetails, Double] { + case ItemCandidateWithDetails(_, _, features) => + -features.getOrElse(ScoreFeature, None).getOrElse(0.0) + case ModuleCandidateWithDetails(candidates, _, _) => + -candidates.last.features.getOrElse(ScoreFeature, None).getOrElse(0.0) + case _ => throw PipelineFailure(UnexpectedCandidateResult, "Invalid candidate type") + } + + val conversationModuleTweetsOrdering: Ordering[CandidateWithDetails] = + Ordering.by[CandidateWithDetails, Long] { + case ItemCandidateWithDetails(candidate: TweetCandidate, _, _) => candidate.id + case _ => throw PipelineFailure(UnexpectedCandidateResult, "Only Item candidate expected") + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/util/InjectionTransformer.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/util/InjectionTransformer.scala new file mode 100644 index 0000000000..f45102f37e --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/util/InjectionTransformer.scala @@ -0,0 +1,43 @@ +package com.twitter.home_mixer.util + +import com.twitter.bijection.Injection +import com.twitter.io.Buf +import com.twitter.servo.util.Transformer +import com.twitter.storage.client.manhattan.bijections.Bijections +import com.twitter.util.Return +import com.twitter.util.Try +import java.nio.ByteBuffer + +object InjectionTransformerImplicits { + implicit class ByteArrayInjectionToByteBufferTransformer[A](baInj: Injection[A, Array[Byte]]) { + + private val bbInj: Injection[A, ByteBuffer] = baInj + .andThen(Bijections.byteArray2Buf) + .andThen(Bijections.byteBuffer2Buf.inverse) + + def toByteBufferTransformer(): Transformer[A, ByteBuffer] = new InjectionTransformer(bbInj) + def toByteArrayTransformer(): Transformer[A, Array[Byte]] = new InjectionTransformer(baInj) + } + + implicit class BufInjectionToByteBufferTransformer[A](bufInj: Injection[A, Buf]) { + + private val bbInj: Injection[A, ByteBuffer] = bufInj.andThen(Bijections.byteBuffer2Buf.inverse) + private val baInj: Injection[A, Array[Byte]] = bufInj.andThen(Bijections.byteArray2Buf.inverse) + + def toByteBufferTransformer(): Transformer[A, ByteBuffer] = new InjectionTransformer(bbInj) + def toByteArrayTransformer(): Transformer[A, Array[Byte]] = new InjectionTransformer(baInj) + } + + implicit class ByteBufferInjectionToByteBufferTransformer[A](bbInj: Injection[A, ByteBuffer]) { + + private val baInj: Injection[A, Array[Byte]] = bbInj.andThen(Bijections.bb2ba) + + def toByteBufferTransformer(): Transformer[A, ByteBuffer] = new InjectionTransformer(bbInj) + def toByteArrayTransformer(): Transformer[A, Array[Byte]] = new InjectionTransformer(baInj) + } +} + +class InjectionTransformer[A, B](inj: Injection[A, B]) extends Transformer[A, B] { + override def to(a: A): Try[B] = Return(inj(a)) + override def from(b: B): Try[A] = Try.fromScala(inj.invert(b)) +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/util/LanguageUtil.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/util/LanguageUtil.scala new file mode 100644 index 0000000000..3969ef64da --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/util/LanguageUtil.scala @@ -0,0 +1,93 @@ +package com.twitter.home_mixer.util + +import com.twitter.search.common.constants.{thriftscala => scc} +import com.twitter.search.common.util.lang.ThriftLanguageUtil +import com.twitter.service.metastore.gen.{thriftscala => smg} + +object LanguageUtil { + + private val DafaultMinProducedLanguageRatio = 0.05 + private val DefaultMinConsumedLanguageConfidence = 0.8 + + /** + * Computes a list of languages based on UserLanguages information retrieved from Metastore. + * + * The list is sorted in descending order of confidence score associated with each language. + * That is, language with highest confidence value is in index 0. + */ + def computeLanguages( + userLanguages: smg.UserLanguages, + minProducedLanguageRatio: Double = DafaultMinProducedLanguageRatio, + minConsumedLanguageConfidence: Double = DefaultMinConsumedLanguageConfidence + ): Seq[scc.ThriftLanguage] = { + val languageConfidenceMap = computeLanguageConfidenceMap( + userLanguages, + minProducedLanguageRatio, + minConsumedLanguageConfidence + ) + languageConfidenceMap.toSeq.sortWith(_._2 > _._2).map(_._1) // _1 = language, _2 = score + } + + /** + * Computes confidence map based on UserLanguages information retrieved from Metastore. + * where, + * key = language code + * value = level of confidence that the language is applicable to a user. + */ + private def computeLanguageConfidenceMap( + userLanguages: smg.UserLanguages, + minProducedLanguageRatio: Double, + minConsumedLanguageConfidence: Double + ): Map[scc.ThriftLanguage, Double] = { + + val producedLanguages = getLanguageMap(userLanguages.produced) + val consumedLanguages = getLanguageMap(userLanguages.consumed) + val languages = (producedLanguages.keys ++ consumedLanguages.keys).toSet + var maxConfidence = 0.0 + + val confidenceMap = languages.map { language => + val produceRatio = producedLanguages + .get(language) + .map { score => if (score < minProducedLanguageRatio) 0.0 else score } + .getOrElse(0.0) + + val consumeConfidence = consumedLanguages + .get(language) + .map { score => if (score < minConsumedLanguageConfidence) 0.0 else score } + .getOrElse(0.0) + + val overallConfidence = (0.3 + 4 * produceRatio) * (0.1 + consumeConfidence) + maxConfidence = Math.max(maxConfidence, overallConfidence) + + (language -> overallConfidence) + }.toMap + + val normalizedConfidenceMap = if (maxConfidence > 0) { + confidenceMap.map { + case (language, confidenceScore) => + val normalizedScore = (confidenceScore / maxConfidence * 0.9) + 0.1 + (language -> normalizedScore) + } + } else { + confidenceMap + } + normalizedConfidenceMap + } + + private def getLanguageMap( + scoredLanguages: Seq[smg.ScoredString] + ): Map[scc.ThriftLanguage, Double] = { + scoredLanguages.flatMap { scoredLanguage => + getThriftLanguage(scoredLanguage.item).map { language => (language -> scoredLanguage.weight) } + }.toMap + } + + private def getThriftLanguage(languageName: String): Option[scc.ThriftLanguage] = { + val languageOrdinal = ThriftLanguageUtil.getThriftLanguageOf(languageName).ordinal + val language = scc.ThriftLanguage(languageOrdinal) + language match { + case scc.ThriftLanguage.Unknown => None + case _ => Some(language) + } + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/util/MissingKeyException.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/util/MissingKeyException.scala new file mode 100644 index 0000000000..ae8fd4deda --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/util/MissingKeyException.scala @@ -0,0 +1,5 @@ +package com.twitter.home_mixer.util + +object MissingKeyException extends Exception("Missing key") { + override def toString: String = getMessage +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/util/ObservedKeyValueResultHandler.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/util/ObservedKeyValueResultHandler.scala new file mode 100644 index 0000000000..0cd0cd60bc --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/util/ObservedKeyValueResultHandler.scala @@ -0,0 +1,43 @@ +package com.twitter.home_mixer.util + +import com.twitter.finagle.stats.StatsReceiver +import com.twitter.servo.keyvalue.KeyValueResult +import com.twitter.util.Return +import com.twitter.util.Throw +import com.twitter.util.Try + +trait ObservedKeyValueResultHandler { + val statsReceiver: StatsReceiver + val statScope: String + + private lazy val scopedStatsReceiver = statsReceiver.scope(statScope) + private lazy val keyTotalCounter = scopedStatsReceiver.counter("key/total") + private lazy val keyFoundCounter = scopedStatsReceiver.counter("key/found") + private lazy val keyLossCounter = scopedStatsReceiver.counter("key/loss") + private lazy val keyFailureCounter = scopedStatsReceiver.counter("key/failure") + + def observedGet[K, V]( + key: Option[K], + keyValueResult: KeyValueResult[K, V], + ): Try[Option[V]] = { + if (key.nonEmpty) { + keyTotalCounter.incr() + keyValueResult(key.get) match { + case Return(Some(value)) => + keyFoundCounter.incr() + Return(Some(value)) + case Return(None) => + keyLossCounter.incr() + Return(None) + case Throw(exception) => + keyFailureCounter.incr() + Throw(exception) + case _ => + // never reaches here + Return(None) + } + } else { + Throw(MissingKeyException) + } + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/util/ReplyRetweetUtil.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/util/ReplyRetweetUtil.scala new file mode 100644 index 0000000000..13758ec5da --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/util/ReplyRetweetUtil.scala @@ -0,0 +1,120 @@ +package com.twitter.home_mixer.util + +import com.twitter.home_mixer.model.HomeFeatures._ +import com.twitter.product_mixer.component_library.model.candidate.TweetCandidate +import com.twitter.product_mixer.core.model.common.CandidateWithFeatures + +object ReplyRetweetUtil { + + def isEligibleReply(candidate: CandidateWithFeatures[TweetCandidate]): Boolean = { + candidate.features.getOrElse(InReplyToTweetIdFeature, None).nonEmpty && + !candidate.features.getOrElse(IsRetweetFeature, false) + } + + /** + * Builds a map from reply tweet to all ancestors that are also hydrated candidates. If a reply + * does not have any ancestors which are also candidates, it will not add to the returned Map. + * Make sure ancestors are bottom-up ordered such that: + * (1) if parent tweet is a candidate, it should be the first item at the returned ancestors; + * (2) if root tweet is a candidate, it should be the last item at the returned ancestors. + * Retweets of replies or replies to retweets are not included. + */ + def replyToAncestorTweetCandidatesMap( + candidates: Seq[CandidateWithFeatures[TweetCandidate]] + ): Map[Long, Seq[CandidateWithFeatures[TweetCandidate]]] = { + val replyToAncestorTweetIdsMap: Map[Long, Seq[Long]] = + candidates.flatMap { candidate => + if (isEligibleReply(candidate)) { + val ancestorIds = + if (candidate.features.getOrElse(AncestorsFeature, Seq.empty).nonEmpty) { + candidate.features.getOrElse(AncestorsFeature, Seq.empty).map(_.tweetId) + } else { + Seq( + candidate.features.getOrElse(InReplyToTweetIdFeature, None), + candidate.features.getOrElse(ConversationModuleIdFeature, None) + ).flatten.distinct + } + Some(candidate.candidate.id -> ancestorIds) + } else { + None + } + }.toMap + + val ancestorTweetIds = replyToAncestorTweetIdsMap.values.flatten.toSet + val ancestorTweetsMapById: Map[Long, CandidateWithFeatures[TweetCandidate]] = candidates + .filter { maybeAncestor => + ancestorTweetIds.contains(maybeAncestor.candidate.id) + }.map { ancestor => + ancestor.candidate.id -> ancestor + }.toMap + + replyToAncestorTweetIdsMap + .mapValues { ancestorTweetIds => + ancestorTweetIds.flatMap { ancestorTweetId => + ancestorTweetsMapById.get(ancestorTweetId) + } + }.filter { + case (reply, ancestors) => + ancestors.nonEmpty + } + } + + /** + * This map is the opposite of [[replyToAncestorTweetCandidatesMap]]. + * Builds a map from ancestor tweet to all descendant replies that are also hydrated candidates. + * Currently, we only return two ancestors at most: one is inReplyToTweetId and the other + * is conversationId. + * Retweets of replies are not included. + */ + def ancestorTweetIdToDescendantRepliesMap( + candidates: Seq[CandidateWithFeatures[TweetCandidate]] + ): Map[Long, Seq[CandidateWithFeatures[TweetCandidate]]] = { + val tweetToCandidateMap = candidates.map(c => c.candidate.id -> c).toMap + replyToAncestorTweetCandidatesMap(candidates).toSeq + .flatMap { + case (reply, ancestorTweets) => + ancestorTweets.map { ancestor => + (ancestor.candidate.id, reply) + } + }.groupBy { case (ancestor, reply) => ancestor } + .mapValues { ancestorReplyPairs => + ancestorReplyPairs.map(_._2).distinct + }.mapValues(tweetIds => tweetIds.map(tid => tweetToCandidateMap(tid))) + } + + /** + * Builds a map from reply tweet to inReplyToTweet which is also a candidate. + * Retweets of replies or replies to retweets are not included + */ + def replyTweetIdToInReplyToTweetMap( + candidates: Seq[CandidateWithFeatures[TweetCandidate]] + ): Map[Long, CandidateWithFeatures[TweetCandidate]] = { + val eligibleReplyCandidates = candidates.filter { candidate => + isEligibleReply(candidate) && candidate.features + .getOrElse(InReplyToTweetIdFeature, None) + .nonEmpty + } + + val inReplyToTweetIds = eligibleReplyCandidates + .flatMap(_.features.getOrElse(InReplyToTweetIdFeature, None)) + .toSet + + val inReplyToTweetIdToTweetMap: Map[Long, CandidateWithFeatures[TweetCandidate]] = candidates + .filter { maybeInReplyToTweet => + inReplyToTweetIds.contains(maybeInReplyToTweet.candidate.id) + }.map { inReplyToTweet => + inReplyToTweet.candidate.id -> inReplyToTweet + }.toMap + + eligibleReplyCandidates.flatMap { reply => + val inReplyToTweetId = reply.features.getOrElse(InReplyToTweetIdFeature, None) + if (inReplyToTweetId.nonEmpty) { + inReplyToTweetIdToTweetMap.get(inReplyToTweetId.get).map { inReplyToTweet => + reply.candidate.id -> inReplyToTweet + } + } else { + None + } + }.toMap + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/util/TensorFlowUtil.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/util/TensorFlowUtil.scala new file mode 100644 index 0000000000..05d7c21279 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/util/TensorFlowUtil.scala @@ -0,0 +1,32 @@ +package com.twitter.home_mixer.util + +import com.twitter.ml.api.thriftscala.FloatTensor +import com.twitter.ml.api.util.BufferToIterators.RichFloatBuffer +import java.nio.ByteBuffer +import java.nio.ByteOrder + +/** + * Contains functionality to transform data records and Tensors + */ + +object TensorFlowUtil { + + private def skipEmbeddingBBHeader(bb: ByteBuffer): ByteBuffer = { + val bb_copy = bb.duplicate() + bb_copy.getLong() + bb_copy + } + + private def byteBufferToFloatIterator( + bb: ByteBuffer + ): Iterator[Float] = { + bb.order(ByteOrder.LITTLE_ENDIAN).asFloatBuffer.iterator + } + + def embeddingByteBufferToFloatTensor( + bb: ByteBuffer + ): FloatTensor = { + val bb_content = skipEmbeddingBBHeader(bb) + FloatTensor(byteBufferToFloatIterator(bb_content).map(_.toDouble).toList) + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/util/TweetImpressionsHelper.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/util/TweetImpressionsHelper.scala new file mode 100644 index 0000000000..eabaac9e3a --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/util/TweetImpressionsHelper.scala @@ -0,0 +1,15 @@ +package com.twitter.home_mixer.util + +import com.twitter.home_mixer.model.HomeFeatures.TweetImpressionsFeature +import com.twitter.product_mixer.component_library.feature_hydrator.query.impressed_tweets.ImpressedTweets +import com.twitter.product_mixer.core.feature.featuremap.FeatureMap + +object TweetImpressionsHelper { + def tweetImpressions(features: FeatureMap): Set[Long] = { + val manhattanImpressions = + features.getOrElse(TweetImpressionsFeature, Seq.empty).flatMap(_.tweetIds) + val memcacheImpressions = features.getOrElse(ImpressedTweets, Seq.empty) + + (manhattanImpressions ++ memcacheImpressions).toSet + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/util/earlybird/BUILD.bazel b/home-mixer/server/src/main/scala/com/twitter/home_mixer/util/earlybird/BUILD.bazel new file mode 100644 index 0000000000..0ca1f89aaa --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/util/earlybird/BUILD.bazel @@ -0,0 +1,18 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + strict_deps = True, + tags = ["bazel-compatible"], + dependencies = [ + "home-mixer/thrift/src/main/thrift:thrift-scala", + "src/java/com/twitter/search/common/schema/base", + "src/java/com/twitter/search/common/schema/earlybird", + "src/java/com/twitter/search/common/util/lang", + "src/java/com/twitter/search/queryparser/query:core-query-nodes", + "src/java/com/twitter/search/queryparser/query/search:search-query-nodes", + "src/thrift/com/twitter/search:earlybird-scala", + "src/thrift/com/twitter/search/common:constants-scala", + "src/thrift/com/twitter/search/common:query-scala", + "src/thrift/com/twitter/search/common:ranking-scala", + ], +) diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/util/earlybird/EarlybirdRequestUtil.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/util/earlybird/EarlybirdRequestUtil.scala new file mode 100644 index 0000000000..19df02a200 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/util/earlybird/EarlybirdRequestUtil.scala @@ -0,0 +1,58 @@ +package com.twitter.home_mixer.util.earlybird + +import com.twitter.conversions.DurationOps._ +import com.twitter.search.common.query.thriftjava.{thriftscala => scq} +import com.twitter.search.earlybird.{thriftscala => eb} +import com.twitter.util.Duration + +object EarlybirdRequestUtil { + + // If no EarlybirdOptions.maxNumHitsPerShard is set then default to this value. + val DefaultMaxHitsToProcess = 1000 + val DefaultSearchProcessingTimeout: Duration = 200.milliseconds + val DefaultMaxNumResultsPerShard = 100 + val DeafultCollectorParams = scq.CollectorParams( + // numResultsToReturn defines how many results each EB shard will return to search root + numResultsToReturn = DefaultMaxNumResultsPerShard, + // terminationParams.maxHitsToProcess is used for early terminating per shard results fetching. + terminationParams = Some( + scq.CollectorTerminationParams( + maxHitsToProcess = Some(DefaultMaxHitsToProcess), + timeoutMs = DefaultSearchProcessingTimeout.inMilliseconds.toInt + )) + ) + + def getTweetsEBFeaturesRequest( + userId: Option[Long], + tweetIds: Option[Seq[Long]], + clientId: Option[String], + getTweetsFromArchiveIndex: Boolean = false, + getOnlyProtectedTweets: Boolean = false, + ): eb.EarlybirdRequest = { + + val candidateSize = tweetIds.getOrElse(Seq.empty).size + val thriftQuery = eb.ThriftSearchQuery( + numResults = candidateSize, + collectConversationId = true, + rankingMode = eb.ThriftSearchRankingMode.Relevance, + relevanceOptions = Some(RelevanceSearchUtil.RelevanceOptions), + collectorParams = Some(DeafultCollectorParams), + facetFieldNames = Some(RelevanceSearchUtil.FacetsToFetch), + resultMetadataOptions = Some(RelevanceSearchUtil.MetadataOptions), + searcherId = userId, + searchStatusIds = tweetIds.map(_.toSet), + ) + + eb.EarlybirdRequest( + searchQuery = thriftQuery, + clientId = clientId, + getOlderResults = Some(getTweetsFromArchiveIndex), + getProtectedTweetsOnly = Some(getOnlyProtectedTweets), + timeoutMs = DefaultSearchProcessingTimeout.inMilliseconds.toInt, + skipVeryRecentTweets = true, + // This param decides # of tweets to return from search superRoot and realtime/protected/Archive roots. + // It takes higher precedence than ThriftSearchQuery.numResults + numResultsToReturnAtRoot = Some(candidateSize) + ) + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/util/earlybird/EarlybirdResponseUtil.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/util/earlybird/EarlybirdResponseUtil.scala new file mode 100644 index 0000000000..fb52e96890 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/util/earlybird/EarlybirdResponseUtil.scala @@ -0,0 +1,369 @@ +package com.twitter.home_mixer.util.earlybird + +import com.twitter.search.common.constants.{thriftscala => scc} +import com.twitter.search.common.features.{thriftscala => sc} +import com.twitter.search.common.schema.earlybird.EarlybirdFieldConstants.EarlybirdFieldConstant +import com.twitter.search.common.schema.earlybird.EarlybirdFieldConstants.EarlybirdFieldConstant._ +import com.twitter.search.common.util.lang.ThriftLanguageUtil +import com.twitter.search.earlybird.{thriftscala => eb} + +object EarlybirdResponseUtil { + + private[earlybird] val Mentions: String = "mentions" + private[earlybird] val Hashtags: String = "hashtags" + private val CharsToRemoveFromMentions: Set[Char] = "@".toSet + private val CharsToRemoveFromHashtags: Set[Char] = "#".toSet + + // Default value of settings of ThriftTweetFeatures. + private[earlybird] val DefaultEarlybirdFeatures: sc.ThriftTweetFeatures = sc.ThriftTweetFeatures() + private[earlybird] val DefaultCount = 0 + private[earlybird] val DefaultLanguage = 0 + private[earlybird] val DefaultScore = 0.0 + + private[earlybird] def getTweetCountByAuthorId( + searchResults: Seq[eb.ThriftSearchResult] + ): Map[Long, Int] = { + searchResults + .groupBy { result => + result.metadata.map(_.fromUserId).getOrElse(0L) + }.mapValues(_.size).withDefaultValue(0) + } + + private[earlybird] def getLanguage(uiLanguageCode: Option[String]): Option[scc.ThriftLanguage] = { + uiLanguageCode.flatMap { languageCode => + scc.ThriftLanguage.get(ThriftLanguageUtil.getThriftLanguageOf(languageCode).getValue) + } + } + + private def getMentions(result: eb.ThriftSearchResult): Seq[String] = { + val facetLabels = result.metadata.flatMap(_.facetLabels).getOrElse(Seq.empty) + getFacets(facetLabels, Mentions, CharsToRemoveFromMentions) + } + + private def getHashtags(result: eb.ThriftSearchResult): Seq[String] = { + val facetLabels = result.metadata.flatMap(_.facetLabels).getOrElse(Seq.empty) + getFacets(facetLabels, Hashtags, CharsToRemoveFromHashtags) + } + + private def getFacets( + facetLabels: Seq[eb.ThriftFacetLabel], + facetName: String, + charsToRemove: Set[Char] + ): Seq[String] = { + facetLabels.filter(_.fieldName == facetName).map(_.label.filterNot(charsToRemove)) + } + + private def isUserMentioned( + screenName: Option[String], + mentions: Seq[String] + ): Boolean = { + screenName + .exists { screenName => mentions.exists(_.equalsIgnoreCase(screenName)) } + } + + private[earlybird] def isUsersMainLanguage( + tweetLanguage: scc.ThriftLanguage, + userLanguages: Seq[scc.ThriftLanguage] + ): Boolean = { + (tweetLanguage != scc.ThriftLanguage.Unknown) && userLanguages.headOption.contains( + tweetLanguage) + } + + private[earlybird] def isUsersLanguage( + tweetLanguage: scc.ThriftLanguage, + userLanguages: Seq[scc.ThriftLanguage] + ): Boolean = { + (tweetLanguage != scc.ThriftLanguage.Unknown) && userLanguages.contains(tweetLanguage) + } + + private[earlybird] def isUILanguage( + tweetLanguage: scc.ThriftLanguage, + uiLanguage: Option[scc.ThriftLanguage] + ): Boolean = { + (tweetLanguage != scc.ThriftLanguage.Unknown) && uiLanguage.contains(tweetLanguage) + } + + private def getBooleanOptFeature( + featureName: EarlybirdFieldConstant, + resultMapOpt: Option[scala.collection.Map[Int, Boolean]], + defaultValue: Boolean = false, + ): Option[Boolean] = { + resultMapOpt.map { + _.getOrElse(featureName.getFieldId, defaultValue) + } + } + + private def getDoubleAsIntOptFeature( + featureName: EarlybirdFieldConstant, + resultMapOpt: Option[scala.collection.Map[Int, Double]] + ): Option[Int] = { + if (resultMapOpt.exists(_.contains(featureName.getFieldId))) + resultMapOpt + .map { + _.get(featureName.getFieldId) + } + .flatMap { doubleValue => + doubleValue.map(_.toInt) + } + else + None + } + + private def getIntOptFeature( + featureName: EarlybirdFieldConstant, + resultMapOpt: Option[scala.collection.Map[Int, Int]] + ): Option[Int] = { + if (resultMapOpt.exists(_.contains(featureName.getFieldId))) + resultMapOpt.flatMap { + _.get(featureName.getFieldId) + } + else + None + } + + def getOONTweetThriftFeaturesByTweetId( + searcherUserId: Long, + screenName: Option[String], + userLanguages: Seq[scc.ThriftLanguage], + uiLanguageCode: Option[String] = None, + searchResults: Seq[eb.ThriftSearchResult], + ): Map[Long, sc.ThriftTweetFeatures] = { + + searchResults.map { searchResult => + val features = getOONThriftTweetFeaturesFromSearchResult( + searcherUserId, + screenName, + userLanguages, + getLanguage(uiLanguageCode), + getTweetCountByAuthorId(searchResults), + searchResult + ) + (searchResult.id -> features) + }.toMap + } + + private[earlybird] def getOONThriftTweetFeaturesFromSearchResult( + searcherUserId: Long, + screenName: Option[String], + userLanguages: Seq[scc.ThriftLanguage], + uiLanguage: Option[scc.ThriftLanguage], + tweetCountByAuthorId: Map[Long, Int], + searchResult: eb.ThriftSearchResult + ): sc.ThriftTweetFeatures = { + val applyFeatures = (applyUserIndependentFeatures( + searchResult + )(_)).andThen( + applyOONUserDependentFeatures( + searcherUserId, + screenName, + userLanguages, + uiLanguage, + tweetCountByAuthorId, + searchResult + )(_) + ) + val tweetFeatures = searchResult.tweetFeatures.getOrElse(DefaultEarlybirdFeatures) + applyFeatures(tweetFeatures) + } + + private[earlybird] def applyUserIndependentFeatures( + result: eb.ThriftSearchResult + )( + thriftTweetFeatures: sc.ThriftTweetFeatures + ): sc.ThriftTweetFeatures = { + + val features = result.metadata + .map { metadata => + val isRetweet = metadata.isRetweet.getOrElse(false) + val isReply = metadata.isReply.getOrElse(false) + + // Facets. + val mentions = getMentions(result) + val hashtags = getHashtags(result) + + val searchResultSchemaFeatures = metadata.extraMetadata.flatMap(_.features) + val booleanSearchResultSchemaFeatures = searchResultSchemaFeatures.flatMap(_.boolValues) + val intSearchResultSchemaFeatures = searchResultSchemaFeatures.flatMap(_.intValues) + val doubleSearchResultSchemaFeatures = searchResultSchemaFeatures.flatMap(_.doubleValues) + + thriftTweetFeatures.copy( + // Info about the Tweet. + isRetweet = isRetweet, + isOffensive = metadata.isOffensive.getOrElse(false), + isReply = isReply, + fromVerifiedAccount = metadata.fromVerifiedAccount.getOrElse(false), + cardType = metadata.cardType, + signature = metadata.signature, + language = metadata.language, + isAuthorNSFW = metadata.isUserNSFW.getOrElse(false), + isAuthorBot = metadata.isUserBot.getOrElse(false), + isAuthorSpam = metadata.isUserSpam.getOrElse(false), + isSensitiveContent = + metadata.extraMetadata.flatMap(_.isSensitiveContent).getOrElse(false), + isAuthorProfileEgg = metadata.extraMetadata.flatMap(_.profileIsEggFlag).getOrElse(false), + isAuthorNew = metadata.extraMetadata.flatMap(_.isUserNewFlag).getOrElse(false), + linkLanguage = metadata.extraMetadata.flatMap(_.linkLanguage).getOrElse(DefaultLanguage), + // Info about Tweet content/media. + hasCard = metadata.hasCard.getOrElse(false), + hasImage = metadata.hasImage.getOrElse(false), + hasNews = metadata.hasNews.getOrElse(false), + hasVideo = metadata.hasVideo.getOrElse(false), + hasConsumerVideo = metadata.hasConsumerVideo.getOrElse(false), + hasProVideo = metadata.hasProVideo.getOrElse(false), + hasVine = metadata.hasVine.getOrElse(false), + hasPeriscope = metadata.hasPeriscope.getOrElse(false), + hasNativeVideo = metadata.hasNativeVideo.getOrElse(false), + hasNativeImage = metadata.hasNativeImage.getOrElse(false), + hasLink = metadata.hasLink.getOrElse(false), + hasVisibleLink = metadata.hasVisibleLink.getOrElse(false), + hasTrend = metadata.hasTrend.getOrElse(false), + hasMultipleHashtagsOrTrends = metadata.hasMultipleHashtagsOrTrends.getOrElse(false), + hasQuote = metadata.extraMetadata.flatMap(_.hasQuote), + urlsList = metadata.tweetUrls.map { + _.map(_.originalUrl) + }, + hasMultipleMedia = + metadata.extraMetadata.flatMap(_.hasMultipleMediaFlag).getOrElse(false), + visibleTokenRatio = getIntOptFeature(VISIBLE_TOKEN_RATIO, intSearchResultSchemaFeatures), + // Various counts. + favCount = metadata.favCount.getOrElse(DefaultCount), + replyCount = metadata.replyCount.getOrElse(DefaultCount), + retweetCount = metadata.retweetCount.getOrElse(DefaultCount), + quoteCount = metadata.extraMetadata.flatMap(_.quotedCount), + embedsImpressionCount = metadata.embedsImpressionCount.getOrElse(DefaultCount), + embedsUrlCount = metadata.embedsUrlCount.getOrElse(DefaultCount), + videoViewCount = metadata.videoViewCount.getOrElse(DefaultCount), + numMentions = metadata.extraMetadata.flatMap(_.numMentions).getOrElse(DefaultCount), + numHashtags = metadata.extraMetadata.flatMap(_.numHashtags).getOrElse(DefaultCount), + favCountV2 = metadata.extraMetadata.flatMap(_.favCountV2), + replyCountV2 = metadata.extraMetadata.flatMap(_.replyCountV2), + retweetCountV2 = metadata.extraMetadata.flatMap(_.retweetCountV2), + weightedFavoriteCount = metadata.extraMetadata.flatMap(_.weightedFavCount), + weightedReplyCount = metadata.extraMetadata.flatMap(_.weightedReplyCount), + weightedRetweetCount = metadata.extraMetadata.flatMap(_.weightedRetweetCount), + weightedQuoteCount = metadata.extraMetadata.flatMap(_.weightedQuoteCount), + embedsImpressionCountV2 = + getDoubleAsIntOptFeature(EMBEDS_IMPRESSION_COUNT_V2, doubleSearchResultSchemaFeatures), + embedsUrlCountV2 = + getDoubleAsIntOptFeature(EMBEDS_URL_COUNT_V2, doubleSearchResultSchemaFeatures), + decayedFavoriteCount = + getDoubleAsIntOptFeature(DECAYED_FAVORITE_COUNT, doubleSearchResultSchemaFeatures), + decayedRetweetCount = + getDoubleAsIntOptFeature(DECAYED_RETWEET_COUNT, doubleSearchResultSchemaFeatures), + decayedReplyCount = + getDoubleAsIntOptFeature(DECAYED_REPLY_COUNT, doubleSearchResultSchemaFeatures), + decayedQuoteCount = + getDoubleAsIntOptFeature(DECAYED_QUOTE_COUNT, doubleSearchResultSchemaFeatures), + fakeFavoriteCount = + getDoubleAsIntOptFeature(FAKE_FAVORITE_COUNT, doubleSearchResultSchemaFeatures), + fakeRetweetCount = + getDoubleAsIntOptFeature(FAKE_RETWEET_COUNT, doubleSearchResultSchemaFeatures), + fakeReplyCount = + getDoubleAsIntOptFeature(FAKE_REPLY_COUNT, doubleSearchResultSchemaFeatures), + fakeQuoteCount = + getDoubleAsIntOptFeature(FAKE_QUOTE_COUNT, doubleSearchResultSchemaFeatures), + // Scores. + textScore = metadata.textScore.getOrElse(DefaultScore), + earlybirdScore = metadata.score.getOrElse(DefaultScore), + parusScore = metadata.parusScore.getOrElse(DefaultScore), + userRep = metadata.userRep.getOrElse(DefaultScore), + pBlockScore = metadata.extraMetadata.flatMap(_.pBlockScore), + toxicityScore = metadata.extraMetadata.flatMap(_.toxicityScore), + pSpammyTweetScore = metadata.extraMetadata.flatMap(_.pSpammyTweetScore), + pReportedTweetScore = metadata.extraMetadata.flatMap(_.pReportedTweetScore), + pSpammyTweetContent = metadata.extraMetadata.flatMap(_.spammyTweetContentScore), + // Safety Signals + labelAbusiveFlag = + getBooleanOptFeature(LABEL_ABUSIVE_FLAG, booleanSearchResultSchemaFeatures), + labelAbusiveHiRclFlag = + getBooleanOptFeature(LABEL_ABUSIVE_HI_RCL_FLAG, booleanSearchResultSchemaFeatures), + labelDupContentFlag = + getBooleanOptFeature(LABEL_DUP_CONTENT_FLAG, booleanSearchResultSchemaFeatures), + labelNsfwHiPrcFlag = + getBooleanOptFeature(LABEL_NSFW_HI_PRC_FLAG, booleanSearchResultSchemaFeatures), + labelNsfwHiRclFlag = + getBooleanOptFeature(LABEL_NSFW_HI_RCL_FLAG, booleanSearchResultSchemaFeatures), + labelSpamFlag = getBooleanOptFeature(LABEL_SPAM_FLAG, booleanSearchResultSchemaFeatures), + labelSpamHiRclFlag = + getBooleanOptFeature(LABEL_SPAM_HI_RCL_FLAG, booleanSearchResultSchemaFeatures), + // Periscope Features + periscopeExists = + getBooleanOptFeature(PERISCOPE_EXISTS, booleanSearchResultSchemaFeatures), + periscopeHasBeenFeatured = + getBooleanOptFeature(PERISCOPE_HAS_BEEN_FEATURED, booleanSearchResultSchemaFeatures), + periscopeIsCurrentlyFeatured = getBooleanOptFeature( + PERISCOPE_IS_CURRENTLY_FEATURED, + booleanSearchResultSchemaFeatures), + periscopeIsFromQualitySource = getBooleanOptFeature( + PERISCOPE_IS_FROM_QUALITY_SOURCE, + booleanSearchResultSchemaFeatures), + periscopeIsLive = + getBooleanOptFeature(PERISCOPE_IS_LIVE, booleanSearchResultSchemaFeatures), + // Last Engagement Features + lastFavSinceCreationHrs = + getIntOptFeature(LAST_FAVORITE_SINCE_CREATION_HRS, intSearchResultSchemaFeatures), + lastRetweetSinceCreationHrs = + getIntOptFeature(LAST_RETWEET_SINCE_CREATION_HRS, intSearchResultSchemaFeatures), + lastReplySinceCreationHrs = + getIntOptFeature(LAST_REPLY_SINCE_CREATION_HRS, intSearchResultSchemaFeatures), + lastQuoteSinceCreationHrs = + getIntOptFeature(LAST_QUOTE_SINCE_CREATION_HRS, intSearchResultSchemaFeatures), + likedByUserIds = metadata.extraMetadata.flatMap(_.likedByUserIds), + mentionsList = if (mentions.nonEmpty) Some(mentions) else None, + hashtagsList = if (hashtags.nonEmpty) Some(hashtags) else None, + isComposerSourceCamera = + getBooleanOptFeature(COMPOSER_SOURCE_IS_CAMERA_FLAG, booleanSearchResultSchemaFeatures), + ) + } + .getOrElse(thriftTweetFeatures) + + if (result.tweetSource.contains(eb.ThriftTweetSource.RealtimeProtectedCluster)) { + features.copy(isProtected = true) + } else { + features + } + } + + // Omitting inNetwork features e.g source tweet features and follow graph. + // Can be expanded to include InNetwork in the future. + def applyOONUserDependentFeatures( + searcherUserId: Long, + screenName: Option[String], + userLanguages: Seq[scc.ThriftLanguage], + uiLanguage: Option[scc.ThriftLanguage], + tweetCountByAuthorId: Map[Long, Int], + result: eb.ThriftSearchResult + )( + thriftTweetFeatures: sc.ThriftTweetFeatures + ): sc.ThriftTweetFeatures = { + result.metadata + .map { metadata => + val isRetweet = metadata.isRetweet.getOrElse(false) + val isReply = metadata.isReply.getOrElse(false) + val replyToSearcher = isReply && (metadata.referencedTweetAuthorId == searcherUserId) + val replyOther = isReply && !replyToSearcher + val retweetOther = isRetweet && (metadata.referencedTweetAuthorId != searcherUserId) + val tweetLanguage = metadata.language.getOrElse(scc.ThriftLanguage.Unknown) + + thriftTweetFeatures.copy( + // Info about the Tweet. + fromSearcher = metadata.fromUserId == searcherUserId, + probablyFromFollowedAuthor = false, + fromMutualFollow = false, + replySearcher = replyToSearcher, + replyOther = replyOther, + retweetOther = retweetOther, + mentionSearcher = isUserMentioned(screenName, getMentions(result)), + // Info about Tweet content/media. + matchesSearcherMainLang = isUsersMainLanguage(tweetLanguage, userLanguages), + matchesSearcherLangs = isUsersLanguage(tweetLanguage, userLanguages), + matchesUILang = isUILanguage(tweetLanguage, uiLanguage), + // Various counts. + prevUserTweetEngagement = + metadata.extraMetadata.flatMap(_.prevUserTweetEngagement).getOrElse(DefaultCount), + tweetCountFromUserInSnapshot = tweetCountByAuthorId(metadata.fromUserId), + ) + } + .getOrElse(thriftTweetFeatures) + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/util/earlybird/RelevanceSearchUtil.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/util/earlybird/RelevanceSearchUtil.scala new file mode 100644 index 0000000000..30be20d606 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/util/earlybird/RelevanceSearchUtil.scala @@ -0,0 +1,71 @@ +package com.twitter.home_mixer.util.earlybird + +import com.twitter.search.common.schema.earlybird.EarlybirdFieldConstants.EarlybirdFieldConstant +import com.twitter.search.common.ranking.{thriftscala => scr} +import com.twitter.search.earlybird.{thriftscala => eb} + +object RelevanceSearchUtil { + + val Mentions: String = EarlybirdFieldConstant.MENTIONS_FACET + val Hashtags: String = EarlybirdFieldConstant.HASHTAGS_FACET + val FacetsToFetch: Seq[String] = Seq(Mentions, Hashtags) + + private val RankingParams: scr.ThriftRankingParams = { + scr.ThriftRankingParams( + `type` = Some(scr.ThriftScoringFunctionType.TensorflowBased), + selectedTensorflowModel = Some("timelines_rectweet_replica"), + minScore = -1.0e100, + retweetCountParams = Some(scr.ThriftLinearFeatureRankingParams(weight = 20.0)), + replyCountParams = Some(scr.ThriftLinearFeatureRankingParams(weight = 1.0)), + reputationParams = Some(scr.ThriftLinearFeatureRankingParams(weight = 0.2)), + luceneScoreParams = Some(scr.ThriftLinearFeatureRankingParams(weight = 2.0)), + textScoreParams = Some(scr.ThriftLinearFeatureRankingParams(weight = 0.18)), + urlParams = Some(scr.ThriftLinearFeatureRankingParams(weight = 2.0)), + isReplyParams = Some(scr.ThriftLinearFeatureRankingParams(weight = 1.0)), + favCountParams = Some(scr.ThriftLinearFeatureRankingParams(weight = 30.0)), + langEnglishUIBoost = 0.5, + langEnglishTweetBoost = 0.2, + langDefaultBoost = 0.02, + unknownLanguageBoost = 0.05, + offensiveBoost = 0.1, + inTrustedCircleBoost = 3.0, + multipleHashtagsOrTrendsBoost = 0.6, + inDirectFollowBoost = 4.0, + tweetHasTrendBoost = 1.1, + selfTweetBoost = 2.0, + tweetHasImageUrlBoost = 2.0, + tweetHasVideoUrlBoost = 2.0, + useUserLanguageInfo = true, + ageDecayParams = Some(scr.ThriftAgeDecayRankingParams(slope = 0.005, base = 1.0)), + selectedModels = Some(Map("home_mixer_unified_engagement_prod" -> 1.0)), + applyBoosts = false, + ) + } + + val MetadataOptions: eb.ThriftSearchResultMetadataOptions = { + eb.ThriftSearchResultMetadataOptions( + getTweetUrls = true, + getResultLocation = false, + getLuceneScore = false, + getInReplyToStatusId = true, + getReferencedTweetAuthorId = true, + getMediaBits = true, + getAllFeatures = true, + returnSearchResultFeatures = true, + // Set getExclusiveConversationAuthorId in order to retrieve Exclusive / SuperFollow tweets. + getExclusiveConversationAuthorId = true + ) + } + + val RelevanceOptions: eb.ThriftSearchRelevanceOptions = { + eb.ThriftSearchRelevanceOptions( + proximityScoring = true, + maxConsecutiveSameUser = Some(2), + rankingParams = Some(RankingParams), + maxHitsToProcess = Some(500), + maxUserBlendCount = Some(3), + proximityPhraseWeight = 9.0, + returnAllResults = Some(true) + ) + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/util/tweetypie/BUILD.bazel b/home-mixer/server/src/main/scala/com/twitter/home_mixer/util/tweetypie/BUILD.bazel new file mode 100644 index 0000000000..dbfee21f4a --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/util/tweetypie/BUILD.bazel @@ -0,0 +1,10 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + strict_deps = True, + tags = ["bazel-compatible"], + dependencies = [ + "home-mixer/server/src/main/scala/com/twitter/home_mixer/model", + "home-mixer/thrift/src/main/thrift:thrift-scala", + ], +) diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/util/tweetypie/RequestFields.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/util/tweetypie/RequestFields.scala new file mode 100644 index 0000000000..9d048005ef --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/util/tweetypie/RequestFields.scala @@ -0,0 +1,57 @@ +package com.twitter.home_mixer.util.tweetypie + +import com.twitter.tweetypie.{thriftscala => tp} + +object RequestFields { + + val CoreTweetFields: Set[tp.TweetInclude] = Set[tp.TweetInclude]( + tp.TweetInclude.TweetFieldId(tp.Tweet.IdField.id), + tp.TweetInclude.TweetFieldId(tp.Tweet.CoreDataField.id) + ) + val MediaFields: Set[tp.TweetInclude] = Set[tp.TweetInclude]( + tp.TweetInclude.TweetFieldId(tp.Tweet.MediaField.id), + ) + val SelfThreadFields: Set[tp.TweetInclude] = Set[tp.TweetInclude]( + tp.TweetInclude.TweetFieldId(tp.Tweet.SelfThreadMetadataField.id) + ) + val MentionsTweetFields: Set[tp.TweetInclude] = Set[tp.TweetInclude]( + tp.TweetInclude.TweetFieldId(tp.Tweet.MentionsField.id) + ) + val SemanticAnnotationTweetFields: Set[tp.TweetInclude] = Set[tp.TweetInclude]( + tp.TweetInclude.TweetFieldId(tp.Tweet.EscherbirdEntityAnnotationsField.id) + ) + val NsfwLabelFields: Set[tp.TweetInclude] = Set[tp.TweetInclude]( + // Tweet fields containing NSFW related attributes. + tp.TweetInclude.TweetFieldId(tp.Tweet.NsfwHighRecallLabelField.id), + tp.TweetInclude.TweetFieldId(tp.Tweet.NsfwHighPrecisionLabelField.id), + tp.TweetInclude.TweetFieldId(tp.Tweet.NsfaHighRecallLabelField.id) + ) + val SafetyLabelFields: Set[tp.TweetInclude] = Set[tp.TweetInclude]( + // Tweet fields containing RTF labels for abuse and spam. + tp.TweetInclude.TweetFieldId(tp.Tweet.SpamLabelField.id), + tp.TweetInclude.TweetFieldId(tp.Tweet.AbusiveLabelField.id) + ) + val ConversationControlField: Set[tp.TweetInclude] = + Set[tp.TweetInclude](tp.TweetInclude.TweetFieldId(tp.Tweet.ConversationControlField.id)) + + val TweetTPHydrationFields: Set[tp.TweetInclude] = CoreTweetFields ++ + NsfwLabelFields ++ + SafetyLabelFields ++ + SemanticAnnotationTweetFields ++ + Set( + tp.TweetInclude.TweetFieldId(tp.Tweet.TakedownCountryCodesField.id), + // QTs imply a TweetyPie -> SGS request dependency + tp.TweetInclude.TweetFieldId(tp.Tweet.QuotedTweetField.id), + tp.TweetInclude.TweetFieldId(tp.Tweet.CommunitiesField.id), + // Field required for determining if a Tweet was created via News Camera. + tp.TweetInclude.TweetFieldId(tp.Tweet.ComposerSourceField.id) + ) + + val TweetStaticEntitiesFields: Set[tp.TweetInclude] = + MentionsTweetFields ++ CoreTweetFields ++ SemanticAnnotationTweetFields ++ MediaFields + + val ContentFields: Set[tp.TweetInclude] = CoreTweetFields ++ MediaFields ++ SelfThreadFields ++ + ConversationControlField ++ SemanticAnnotationTweetFields ++ + Set[tp.TweetInclude]( + tp.TweetInclude.MediaEntityFieldId(tp.MediaEntity.AdditionalMetadataField.id)) +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/util/tweetypie/content/BUILD.bazel b/home-mixer/server/src/main/scala/com/twitter/home_mixer/util/tweetypie/content/BUILD.bazel new file mode 100644 index 0000000000..5acfd98e0b --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/util/tweetypie/content/BUILD.bazel @@ -0,0 +1,19 @@ +scala_library( + sources = ["*.scala"], + compiler_option_sets = ["fatal_warnings"], + strict_deps = True, + tags = ["bazel-compatible"], + dependencies = [ + "home-mixer/server/src/main/scala/com/twitter/home_mixer/model", + "home-mixer/thrift/src/main/thrift:thrift-scala", + "src/java/com/twitter/common/text/tagger", + "src/java/com/twitter/common/text/token", + "src/java/com/twitter/common_internal/text", + "src/java/com/twitter/common_internal/text/version", + "src/java/com/twitter/search/common/util/text", + "src/thrift/com/twitter/search/common:features-scala", + "src/thrift/com/twitter/tweetypie:media-entity-scala", + "src/thrift/com/twitter/tweetypie:service-scala", + "src/thrift/com/twitter/tweetypie:tweet-scala", + ], +) diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/util/tweetypie/content/FeatureExtractionHelper.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/util/tweetypie/content/FeatureExtractionHelper.scala new file mode 100644 index 0000000000..07cbdebe4a --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/util/tweetypie/content/FeatureExtractionHelper.scala @@ -0,0 +1,29 @@ +package com.twitter.home_mixer.util.tweetypie.content + +import com.twitter.home_mixer.model.ContentFeatures +import com.twitter.tweetypie.{thriftscala => tp} + +object FeatureExtractionHelper { + + def extractFeatures( + tweet: tp.Tweet + ): ContentFeatures = { + val contentFeaturesFromTweet = ContentFeatures.Empty.copy( + selfThreadMetadata = tweet.selfThreadMetadata + ) + + val contentFeaturesWithText = TweetTextFeaturesExtractor.addTextFeaturesFromTweet( + contentFeaturesFromTweet, + tweet + ) + val contentFeaturesWithMedia = TweetMediaFeaturesExtractor.addMediaFeaturesFromTweet( + contentFeaturesWithText, + tweet + ) + + contentFeaturesWithMedia.copy( + conversationControl = tweet.conversationControl, + semanticCoreAnnotations = tweet.escherbirdEntityAnnotations.map(_.entityAnnotations) + ) + } +} diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/util/tweetypie/content/TweetMediaFeaturesExtractor.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/util/tweetypie/content/TweetMediaFeaturesExtractor.scala new file mode 100644 index 0000000000..0a5a93a2e9 --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/util/tweetypie/content/TweetMediaFeaturesExtractor.scala @@ -0,0 +1,285 @@ +package com.twitter.home_mixer.util.tweetypie.content + +import com.twitter.home_mixer.model.ContentFeatures +import com.twitter.mediaservices.commons.mediainformation.{thriftscala => mi} +import com.twitter.mediaservices.commons.tweetmedia.{thriftscala => tm} +import com.twitter.mediaservices.commons.{thriftscala => ms} +import com.twitter.tweetypie.{thriftscala => tp} +import scala.collection.Map + +object TweetMediaFeaturesExtractor { + + private val ImageCategories = Set( + ms.MediaCategory.TweetImage.value, + ms.MediaCategory.TweetGif.value + ) + private val VideoCategories = Set( + ms.MediaCategory.TweetVideo.value, + ms.MediaCategory.AmplifyVideo.value + ) + + def hasImage(tweet: tp.Tweet): Boolean = hasMediaByCategory(tweet, ImageCategories) + + def hasVideo(tweet: tp.Tweet): Boolean = hasMediaByCategory(tweet, VideoCategories) + + private def hasMediaByCategory(tweet: tp.Tweet, categories: Set[Int]): Boolean = { + tweet.media.exists { mediaEntities => + mediaEntities.exists { mediaEntity => + mediaEntity.mediaKey.map(_.mediaCategory).exists { mediaCategory => + categories.contains(mediaCategory.value) + } + } + } + } + + def addMediaFeaturesFromTweet( + inputFeatures: ContentFeatures, + tweet: tp.Tweet, + ): ContentFeatures = { + val featuresWithMediaEntity = tweet.media + .map { mediaEntities => + val sizeFeatures = getSizeFeatures(mediaEntities) + val playbackFeatures = getPlaybackFeatures(mediaEntities) + val mediaWidths = sizeFeatures.map(_.width.toShort) + val mediaHeights = sizeFeatures.map(_.height.toShort) + val resizeMethods = sizeFeatures.map(_.resizeMethod.toShort) + val faceMapAreas = getFaceMapAreas(mediaEntities) + val sortedColorPalette = getSortedColorPalette(mediaEntities) + val stickerFeatures = getStickerFeatures(mediaEntities) + val mediaOriginProviders = getMediaOriginProviders(mediaEntities) + val isManaged = getIsManaged(mediaEntities) + val is360 = getIs360(mediaEntities) + val viewCount = getViewCount(mediaEntities) + val userDefinedProductMetadataFeatures = + getUserDefinedProductMetadataFeatures(mediaEntities) + val isMonetizable = + getOptBooleanFromSeqOpt(userDefinedProductMetadataFeatures.map(_.isMonetizable)) + val isEmbeddable = + getOptBooleanFromSeqOpt(userDefinedProductMetadataFeatures.map(_.isEmbeddable)) + val hasSelectedPreviewImage = + getOptBooleanFromSeqOpt(userDefinedProductMetadataFeatures.map(_.hasSelectedPreviewImage)) + val hasTitle = getOptBooleanFromSeqOpt(userDefinedProductMetadataFeatures.map(_.hasTitle)) + val hasDescription = + getOptBooleanFromSeqOpt(userDefinedProductMetadataFeatures.map(_.hasDescription)) + val hasVisitSiteCallToAction = getOptBooleanFromSeqOpt( + userDefinedProductMetadataFeatures.map(_.hasVisitSiteCallToAction)) + val hasAppInstallCallToAction = getOptBooleanFromSeqOpt( + userDefinedProductMetadataFeatures.map(_.hasAppInstallCallToAction)) + val hasWatchNowCallToAction = + getOptBooleanFromSeqOpt(userDefinedProductMetadataFeatures.map(_.hasWatchNowCallToAction)) + + inputFeatures.copy( + videoDurationMs = playbackFeatures.durationMs, + bitRate = playbackFeatures.bitRate, + aspectRatioNum = playbackFeatures.aspectRatioNum, + aspectRatioDen = playbackFeatures.aspectRatioDen, + widths = Some(mediaWidths), + heights = Some(mediaHeights), + resizeMethods = Some(resizeMethods), + faceAreas = Some(faceMapAreas), + dominantColorRed = sortedColorPalette.headOption.map(_.rgb.red), + dominantColorBlue = sortedColorPalette.headOption.map(_.rgb.blue), + dominantColorGreen = sortedColorPalette.headOption.map(_.rgb.green), + dominantColorPercentage = sortedColorPalette.headOption.map(_.percentage), + numColors = Some(sortedColorPalette.size.toShort), + stickerIds = Some(stickerFeatures), + mediaOriginProviders = Some(mediaOriginProviders), + isManaged = Some(isManaged), + is360 = Some(is360), + viewCount = viewCount, + isMonetizable = isMonetizable, + isEmbeddable = isEmbeddable, + hasSelectedPreviewImage = hasSelectedPreviewImage, + hasTitle = hasTitle, + hasDescription = hasDescription, + hasVisitSiteCallToAction = hasVisitSiteCallToAction, + hasAppInstallCallToAction = hasAppInstallCallToAction, + hasWatchNowCallToAction = hasWatchNowCallToAction + ) + } + .getOrElse(inputFeatures) + + val featuresWithMediaTags = tweet.mediaTags + .map { mediaTags => + val mediaTagScreenNames = getMediaTagScreenNames(mediaTags.tagMap) + val numMediaTags = mediaTagScreenNames.size + + featuresWithMediaEntity.copy( + mediaTagScreenNames = Some(mediaTagScreenNames), + numMediaTags = Some(numMediaTags.toShort) + ) + } + .getOrElse(featuresWithMediaEntity) + + featuresWithMediaTags + .copy(media = tweet.media) + } + + private def getSizeFeatures(mediaEntities: Seq[tp.MediaEntity]): Seq[MediaSizeFeatures] = { + mediaEntities.map { mediaEntity => + mediaEntity.sizes.foldLeft(MediaSizeFeatures(0, 0, 0))((accDimensions, dimensions) => + MediaSizeFeatures( + width = math.max(dimensions.width, accDimensions.width), + height = math.max(dimensions.height, accDimensions.height), + resizeMethod = math.max(dimensions.resizeMethod.getValue, accDimensions.resizeMethod) + )) + } + } + + private def getPlaybackFeatures(mediaEntities: Seq[tp.MediaEntity]): PlaybackFeatures = { + val allPlaybackFeatures = mediaEntities + .flatMap { mediaEntity => + mediaEntity.mediaInfo map { + case videoEntity: tm.MediaInfo.VideoInfo => + PlaybackFeatures( + durationMs = Some(videoEntity.videoInfo.durationMillis), + bitRate = videoEntity.videoInfo.variants.maxBy(_.bitRate).bitRate, + aspectRatioNum = Some(videoEntity.videoInfo.aspectRatio.numerator), + aspectRatioDen = Some(videoEntity.videoInfo.aspectRatio.denominator) + ) + case gifEntity: tm.MediaInfo.AnimatedGifInfo => + PlaybackFeatures( + durationMs = None, + bitRate = gifEntity.animatedGifInfo.variants.maxBy(_.bitRate).bitRate, + aspectRatioNum = Some(gifEntity.animatedGifInfo.aspectRatio.numerator), + aspectRatioDen = Some(gifEntity.animatedGifInfo.aspectRatio.denominator) + ) + case _ => PlaybackFeatures(None, None, None, None) + } + } + .collect { + case playbackFeatures: PlaybackFeatures => playbackFeatures + } + + if (allPlaybackFeatures.nonEmpty) allPlaybackFeatures.maxBy(_.durationMs) + else PlaybackFeatures(None, None, None, None) + } + + private def getMediaTagScreenNames(tagMap: Map[Long, Seq[tp.MediaTag]]): Seq[String] = + tagMap.values + .flatMap(seqMediaTag => seqMediaTag.flatMap(_.screenName)) + .toSeq + + // Areas of the faces identified in the media entities + private def getFaceMapAreas(mediaEntities: Seq[tp.MediaEntity]): Seq[Int] = { + for { + mediaEntity <- mediaEntities + metadata <- mediaEntity.additionalMetadata.toSeq + faceData <- metadata.faceData + faces <- faceData.faces + } yield { + faces + .getOrElse("orig", Seq.empty[mi.Face]) + .flatMap(f => f.boundingBox.map(bb => bb.width * bb.height)) + } + }.flatten + + // All ColorPalettes in the media sorted by the percentage in descending order + private def getSortedColorPalette( + mediaEntities: Seq[tp.MediaEntity] + ): Seq[mi.ColorPaletteItem] = { + for { + mediaEntity <- mediaEntities + metadata <- mediaEntity.additionalMetadata.toSeq + colorInfo <- metadata.colorInfo + } yield { + colorInfo.palette + } + }.flatten.sortBy(-_.percentage) + + // Id's of stickers applied by the user + private def getStickerFeatures(mediaEntities: Seq[tp.MediaEntity]): Seq[Long] = { + for { + mediaEntity <- mediaEntities + metadata <- mediaEntity.additionalMetadata.toSeq + stickerInfo <- metadata.stickerInfo + } yield { + stickerInfo.stickers.map(_.id) + } + }.flatten + + // 3rd party media providers. eg. giphy for gifs + private def getMediaOriginProviders(mediaEntities: Seq[tp.MediaEntity]): Seq[String] = + for { + mediaEntity <- mediaEntities + metadata <- mediaEntity.additionalMetadata.toSeq + mediaOrigin <- metadata.foundMediaOrigin + } yield { + mediaOrigin.provider + } + + private def getIsManaged(mediaEntities: Seq[tp.MediaEntity]): Boolean = { + for { + mediaEntity <- mediaEntities + metadata <- mediaEntity.additionalMetadata.toSeq + managementInfo <- metadata.managementInfo + } yield { + managementInfo.managed + } + }.contains(true) + + private def getIs360(mediaEntities: Seq[tp.MediaEntity]): Boolean = { + for { + mediaEntity <- mediaEntities + metadata <- mediaEntity.additionalMetadata.toSeq + info360 <- metadata.info360 + } yield { + info360.is360 + } + }.contains(Some(true)) + + private def getViewCount(mediaEntities: Seq[tp.MediaEntity]): Option[Long] = { + for { + mediaEntity <- mediaEntities + metadata <- mediaEntity.additionalMetadata.toSeq + engagementInfo <- metadata.engagementInfo + viewCounts <- engagementInfo.viewCount + } yield { + viewCounts + } + }.reduceOption(_ max _) + + // metadata defined by the user when uploading the image + private def getUserDefinedProductMetadataFeatures( + mediaEntities: Seq[tp.MediaEntity] + ): Seq[UserDefinedProductMetadataFeatures] = + for { + mediaEntity <- mediaEntities + userDefinedMetadata <- mediaEntity.metadata + } yield { + UserDefinedProductMetadataFeatures( + isMonetizable = userDefinedMetadata.monetizable, + isEmbeddable = userDefinedMetadata.embeddable, + hasSelectedPreviewImage = Some(userDefinedMetadata.previewImage.nonEmpty), + hasTitle = userDefinedMetadata.title.map(_.nonEmpty), + hasDescription = userDefinedMetadata.description.map(_.nonEmpty), + hasVisitSiteCallToAction = userDefinedMetadata.callToActions.map(_.visitSite.nonEmpty), + hasAppInstallCallToAction = userDefinedMetadata.callToActions.map(_.appInstall.nonEmpty), + hasWatchNowCallToAction = userDefinedMetadata.callToActions.map(_.watchNow.nonEmpty) + ) + } + + private def getOptBooleanFromSeqOpt( + seqOpt: Seq[Option[Boolean]] + ): Option[Boolean] = Some( + seqOpt.exists(boolOpt => boolOpt.contains(true)) + ) +} + +case class MediaSizeFeatures(width: Int, height: Int, resizeMethod: Int) + +case class PlaybackFeatures( + durationMs: Option[Int], + bitRate: Option[Int], + aspectRatioNum: Option[Short], + aspectRatioDen: Option[Short]) + +case class UserDefinedProductMetadataFeatures( + isMonetizable: Option[Boolean], + isEmbeddable: Option[Boolean], + hasSelectedPreviewImage: Option[Boolean], + hasTitle: Option[Boolean], + hasDescription: Option[Boolean], + hasVisitSiteCallToAction: Option[Boolean], + hasAppInstallCallToAction: Option[Boolean], + hasWatchNowCallToAction: Option[Boolean]) diff --git a/home-mixer/server/src/main/scala/com/twitter/home_mixer/util/tweetypie/content/TweetTextFeaturesExtractor.scala b/home-mixer/server/src/main/scala/com/twitter/home_mixer/util/tweetypie/content/TweetTextFeaturesExtractor.scala new file mode 100644 index 0000000000..0a403d98af --- /dev/null +++ b/home-mixer/server/src/main/scala/com/twitter/home_mixer/util/tweetypie/content/TweetTextFeaturesExtractor.scala @@ -0,0 +1,44 @@ +package com.twitter.home_mixer.util.tweetypie.content + +import com.twitter.home_mixer.model.ContentFeatures +import com.twitter.tweetypie.{thriftscala => tp} + +object TweetTextFeaturesExtractor { + + private val QUESTION_MARK_CHARS = Set( + '\u003F', '\u00BF', '\u037E', '\u055E', '\u061F', '\u1367', '\u1945', '\u2047', '\u2048', + '\u2049', '\u2753', '\u2754', '\u2CFA', '\u2CFB', '\u2E2E', '\uA60F', '\uA6F7', '\uFE16', + '\uFE56', '\uFF1F', '\u1114', '\u1E95' + ) + private val NEW_LINE_REGEX = "\r\n|\r|\n".r + + def addTextFeaturesFromTweet( + inputFeatures: ContentFeatures, + tweet: tp.Tweet + ): ContentFeatures = { + tweet.coreData + .map { coreData => + val tweetText = coreData.text + + inputFeatures.copy( + hasQuestion = hasQuestionCharacter(tweetText), + length = getLength(tweetText).toShort, + numCaps = getCaps(tweetText).toShort, + numWhiteSpaces = getSpaces(tweetText).toShort, + numNewlines = Some(getNumNewlines(tweetText)), + ) + } + .getOrElse(inputFeatures) + } + + def getLength(text: String): Int = + text.codePointCount(0, text.length()) + + def getCaps(text: String): Int = text.count(Character.isUpperCase) + + def getSpaces(text: String): Int = text.count(Character.isWhitespace) + + def hasQuestionCharacter(text: String): Boolean = text.exists(QUESTION_MARK_CHARS.contains) + + def getNumNewlines(text: String): Short = NEW_LINE_REGEX.findAllIn(text).length.toShort +} diff --git a/navi/dr_transform/Cargo.toml b/navi/dr_transform/Cargo.toml new file mode 100644 index 0000000000..47f097eb90 --- /dev/null +++ b/navi/dr_transform/Cargo.toml @@ -0,0 +1,29 @@ +[package] +name = "dr_transform" +version = "0.1.0" +edition = "2021" + +# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html +[dependencies] +serde = { version = "1.0", features = ["derive"] } +serde_json = "1.0" +json = "0.12.4" +bpr_thrift = { path = "../thrift_bpr_adapter/thrift/"} +segdense = { path = "../segdense/"} +thrift = "0.17.0" +ndarray = "0.15" +ort = {git ="https://github.com/pykeio/ort.git", tag="v1.14.2"} +base64 = "0.20.0" +npyz = "0.7.2" +log = "0.4.17" +env_logger = "0.9.0" +prometheus = "0.13.1" +once_cell = "1.17.0" +rand = "0.8.5" +itertools = "0.10.5" +[dev-dependencies] +criterion = "0.3.0" + +[[bench]] +name = "bpr_benchmark" +harness = false diff --git a/navi/dr_transform/src/all_config.rs b/navi/dr_transform/src/all_config.rs new file mode 100644 index 0000000000..426d11cef5 --- /dev/null +++ b/navi/dr_transform/src/all_config.rs @@ -0,0 +1,49 @@ +use serde::{Deserialize, Serialize}; + +use serde_json::Error; + +#[derive(Default, Debug, Clone, PartialEq, Serialize, Deserialize)] +#[serde(rename_all = "camelCase")] +pub struct AllConfig { + #[serde(rename = "train_data")] + pub train_data: TrainData, +} + +#[derive(Default, Debug, Clone, PartialEq, Serialize, Deserialize)] +#[serde(rename_all = "camelCase")] +pub struct TrainData { + #[serde(rename = "seg_dense_schema")] + pub seg_dense_schema: SegDenseSchema, +} + +#[derive(Default, Debug, Clone, PartialEq, Serialize, Deserialize)] +#[serde(rename_all = "camelCase")] +pub struct SegDenseSchema { + #[serde(rename = "renamed_features")] + pub renamed_features: RenamedFeatures, +} + +#[derive(Default, Debug, Clone, PartialEq, Serialize, Deserialize)] +#[serde(rename_all = "camelCase")] +pub struct RenamedFeatures { + pub continuous: String, + pub binary: String, + pub discrete: String, + #[serde(rename = "author_embedding")] + pub author_embedding: String, + #[serde(rename = "user_embedding")] + pub user_embedding: String, + #[serde(rename = "user_eng_embedding")] + pub user_eng_embedding: String, + #[serde(rename = "meta__author_id")] + pub meta_author_id: String, + #[serde(rename = "meta__user_id")] + pub meta_user_id: String, + #[serde(rename = "meta__tweet_id")] + pub meta_tweet_id: String, +} + +pub fn parse(json_str: &str) -> Result { + let all_config: AllConfig = serde_json::from_str(json_str)?; + return std::result::Result::Ok(all_config); +} diff --git a/navi/dr_transform/src/converter.rs b/navi/dr_transform/src/converter.rs new file mode 100644 index 0000000000..30d3ad0a64 --- /dev/null +++ b/navi/dr_transform/src/converter.rs @@ -0,0 +1,621 @@ +use std::collections::BTreeSet; +use std::fmt::{self, Debug, Display}; +use std::fs; + +use bpr_thrift::data::DataRecord; +use bpr_thrift::prediction_service::BatchPredictionRequest; +use bpr_thrift::tensor::GeneralTensor; +use log::debug; +use ndarray::Array2; +use once_cell::sync::OnceCell; +use ort::tensor::InputTensor; +use prometheus::{HistogramOpts, HistogramVec}; +use segdense::mapper::{FeatureMapper, MapReader}; +use segdense::segdense_transform_spec_home_recap_2022::{DensificationTransformSpec, Root}; +use segdense::util; +use thrift::protocol::{TBinaryInputProtocol, TSerializable}; +use thrift::transport::TBufferChannel; + +use crate::{all_config}; +use crate::all_config::AllConfig; + +pub fn log_feature_match( + dr: &DataRecord, + seg_dense_config: &DensificationTransformSpec, + dr_type: String, +) { + // Note the following algorithm matches features from config using linear search. + // Also the record source is MinDataRecord. This includes only binary and continous features for now. + + for (feature_id, feature_value) in dr.continuous_features.as_ref().unwrap().into_iter() { + debug!( + "{} - Continous Datarecord => Feature ID: {}, Feature value: {}", + dr_type, feature_id, feature_value + ); + for input_feature in &seg_dense_config.cont.input_features { + if input_feature.feature_id == *feature_id { + debug!("Matching input feature: {:?}", input_feature) + } + } + } + + for feature_id in dr.binary_features.as_ref().unwrap().into_iter() { + debug!( + "{} - Binary Datarecord => Feature ID: {}", + dr_type, feature_id + ); + for input_feature in &seg_dense_config.binary.input_features { + if input_feature.feature_id == *feature_id { + debug!("Found input feature: {:?}", input_feature) + } + } + } +} + +pub fn log_feature_matches(drs: &Vec, seg_dense_config: &DensificationTransformSpec) { + for dr in drs { + log_feature_match(dr, seg_dense_config, String::from("individual")); + } +} + +pub trait Converter: Send + Sync + Debug + 'static + Display { + fn convert(&self, input: Vec>) -> (Vec, Vec); +} + +#[derive(Debug)] +#[allow(dead_code)] +pub struct BatchPredictionRequestToTorchTensorConverter { + all_config: AllConfig, + seg_dense_config: Root, + all_config_path: String, + seg_dense_config_path: String, + feature_mapper: FeatureMapper, + user_embedding_feature_id: i64, + user_eng_embedding_feature_id: i64, + author_embedding_feature_id: i64, + discrete_features_to_report: BTreeSet, + continuous_features_to_report: BTreeSet, + discrete_feature_metrics: &'static HistogramVec, + continuous_feature_metrics: &'static HistogramVec, +} + +impl Display for BatchPredictionRequestToTorchTensorConverter { + fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { + write!( + f, + "all_config_path: {}, seg_dense_config_path:{}", + self.all_config_path, self.seg_dense_config_path + ) + } +} + +impl BatchPredictionRequestToTorchTensorConverter { + pub fn new( + model_dir: &str, + model_version: &str, + reporting_feature_ids: Vec<(i64, &str)>, + register_metric_fn: Option, + ) -> BatchPredictionRequestToTorchTensorConverter { + let all_config_path = format!("{}/{}/all_config.json", model_dir, model_version); + let seg_dense_config_path = format!( + "{}/{}/segdense_transform_spec_home_recap_2022.json", + model_dir, model_version + ); + let seg_dense_config = util::load_config(&seg_dense_config_path); + let all_config = all_config::parse( + &fs::read_to_string(&all_config_path) + .unwrap_or_else(|error| panic!("error loading all_config.json - {}", error)), + ) + .unwrap(); + + let feature_mapper = util::load_from_parsed_config_ref(&seg_dense_config); + + let user_embedding_feature_id = Self::get_feature_id( + &all_config + .train_data + .seg_dense_schema + .renamed_features + .user_embedding, + &seg_dense_config, + ); + let user_eng_embedding_feature_id = Self::get_feature_id( + &all_config + .train_data + .seg_dense_schema + .renamed_features + .user_eng_embedding, + &seg_dense_config, + ); + let author_embedding_feature_id = Self::get_feature_id( + &all_config + .train_data + .seg_dense_schema + .renamed_features + .author_embedding, + &seg_dense_config, + ); + static METRICS: OnceCell<(HistogramVec, HistogramVec)> = OnceCell::new(); + let (discrete_feature_metrics, continuous_feature_metrics) = METRICS.get_or_init(|| { + let discrete = HistogramVec::new( + HistogramOpts::new(":navi:feature_id:discrete", "Discrete Feature ID values") + .buckets(Vec::from(&[ + 0.0, 10.0, 20.0, 30.0, 40.0, 50.0, 60.0, 70.0, 80.0, 90.0, 100.0, 110.0, + 120.0, 130.0, 140.0, 150.0, 160.0, 170.0, 180.0, 190.0, 200.0, 250.0, + 300.0, 500.0, 1000.0, 10000.0, 100000.0, + ] as &'static [f64])), + &["feature_id"], + ) + .expect("metric cannot be created"); + let continuous = HistogramVec::new( + HistogramOpts::new( + ":navi:feature_id:continuous", + "continuous Feature ID values", + ) + .buckets(Vec::from(&[ + 0.0, 10.0, 20.0, 30.0, 40.0, 50.0, 60.0, 70.0, 80.0, 90.0, 100.0, 110.0, 120.0, + 130.0, 140.0, 150.0, 160.0, 170.0, 180.0, 190.0, 200.0, 250.0, 300.0, 500.0, + 1000.0, 10000.0, 100000.0, + ] as &'static [f64])), + &["feature_id"], + ) + .expect("metric cannot be created"); + register_metric_fn.map(|r| { + r(&discrete); + r(&continuous); + }); + (discrete, continuous) + }); + + let mut discrete_features_to_report = BTreeSet::new(); + let mut continuous_features_to_report = BTreeSet::new(); + + for (feature_id, feature_type) in reporting_feature_ids.iter() { + match *feature_type { + "discrete" => discrete_features_to_report.insert(feature_id.clone()), + "continuous" => continuous_features_to_report.insert(feature_id.clone()), + _ => panic!( + "Invalid feature type {} for reporting metrics!", + feature_type + ), + }; + } + + return BatchPredictionRequestToTorchTensorConverter { + all_config, + seg_dense_config, + all_config_path, + seg_dense_config_path, + feature_mapper, + user_embedding_feature_id, + user_eng_embedding_feature_id, + author_embedding_feature_id, + discrete_features_to_report, + continuous_features_to_report, + discrete_feature_metrics, + continuous_feature_metrics, + }; + } + + fn get_feature_id(feature_name: &str, seg_dense_config: &Root) -> i64 { + // given a feature name, we get the complex feature type id + for feature in &seg_dense_config.complex_feature_type_transform_spec { + if feature.full_feature_name == feature_name { + return feature.feature_id; + } + } + return -1; + } + + fn parse_batch_prediction_request(bytes: Vec) -> BatchPredictionRequest { + // parse batch prediction request into a struct from byte array repr. + let mut bc = TBufferChannel::with_capacity(bytes.len(), 0); + bc.set_readable_bytes(&bytes); + let mut protocol = TBinaryInputProtocol::new(bc, true); + return BatchPredictionRequest::read_from_in_protocol(&mut protocol).unwrap(); + } + + fn get_embedding_tensors( + &self, + bprs: &[BatchPredictionRequest], + feature_id: i64, + batch_size: &[usize], + ) -> Array2 { + // given an embedding feature id, extract the float tensor array into tensors. + let cols: usize = 200; + let rows: usize = batch_size[batch_size.len() - 1]; + let total_size = rows * cols; + + let mut working_set = vec![0 as f32; total_size]; + let mut bpr_start = 0; + for (bpr, &bpr_end) in bprs.iter().zip(batch_size) { + if bpr.common_features.is_some() { + if bpr.common_features.as_ref().unwrap().tensors.is_some() { + if bpr + .common_features + .as_ref() + .unwrap() + .tensors + .as_ref() + .unwrap() + .contains_key(&feature_id) + { + let source_tensor = bpr + .common_features + .as_ref() + .unwrap() + .tensors + .as_ref() + .unwrap() + .get(&feature_id) + .unwrap(); + let tensor = match source_tensor { + GeneralTensor::FloatTensor(float_tensor) => + //Tensor::of_slice( + { + float_tensor + .floats + .iter() + .map(|x| x.into_inner() as f32) + .collect::>() + } + _ => vec![0 as f32; cols], + }; + + // since the tensor is found in common feature, add it in all batches + for row in bpr_start..bpr_end { + for col in 0..cols { + working_set[row * cols + col] = tensor[col]; + } + } + } + } + } + // find the feature in individual feature list and add to corresponding batch. + for (index, datarecord) in bpr.individual_features_list.iter().enumerate() { + if datarecord.tensors.is_some() + && datarecord + .tensors + .as_ref() + .unwrap() + .contains_key(&feature_id) + { + let source_tensor = datarecord + .tensors + .as_ref() + .unwrap() + .get(&feature_id) + .unwrap(); + let tensor = match source_tensor { + GeneralTensor::FloatTensor(float_tensor) => float_tensor + .floats + .iter() + .map(|x| x.into_inner() as f32) + .collect::>(), + _ => vec![0 as f32; cols], + }; + for col in 0..cols { + working_set[(bpr_start + index) * cols + col] = tensor[col]; + } + } + } + bpr_start = bpr_end; + } + return Array2::::from_shape_vec([rows, cols], working_set).unwrap(); + } + + // Todo : Refactor, create a generic version with different type and field accessors + // Example paramterize and then instiantiate the following + // (FLOAT --> FLOAT, DataRecord.continuous_feature) + // (BOOL --> INT64, DataRecord.binary_feature) + // (INT64 --> INT64, DataRecord.discrete_feature) + fn get_continuous(&self, bprs: &[BatchPredictionRequest], batch_ends: &[usize]) -> InputTensor { + // These need to be part of model schema + let rows: usize = batch_ends[batch_ends.len() - 1]; + let cols: usize = 5293; + let full_size: usize = (rows * cols).try_into().unwrap(); + let default_val = f32::NAN; + + let mut tensor = vec![default_val; full_size]; + + let mut bpr_start = 0; + for (bpr, &bpr_end) in bprs.iter().zip(batch_ends) { + // Common features + if bpr.common_features.is_some() + && bpr + .common_features + .as_ref() + .unwrap() + .continuous_features + .is_some() + { + let common_features = bpr + .common_features + .as_ref() + .unwrap() + .continuous_features + .as_ref() + .unwrap(); + + for feature in common_features { + match self.feature_mapper.get(feature.0) { + Some(f_info) => { + let idx = f_info.index_within_tensor as usize; + if idx < cols { + // Set value in each row + for r in bpr_start..bpr_end { + let flat_index: usize = (r * cols + idx).try_into().unwrap(); + tensor[flat_index] = feature.1.into_inner() as f32; + } + } + } + None => (), + } + if self.continuous_features_to_report.contains(feature.0) { + self.continuous_feature_metrics + .with_label_values(&[feature.0.to_string().as_str()]) + .observe(feature.1.into_inner() as f64) + } else if self.discrete_features_to_report.contains(feature.0) { + self.discrete_feature_metrics + .with_label_values(&[feature.0.to_string().as_str()]) + .observe(feature.1.into_inner() as f64) + } + } + } + + // Process the batch of datarecords + for r in bpr_start..bpr_end { + let dr: &DataRecord = + &bpr.individual_features_list[usize::try_from(r - bpr_start).unwrap()]; + if dr.continuous_features.is_some() { + for feature in dr.continuous_features.as_ref().unwrap() { + match self.feature_mapper.get(&feature.0) { + Some(f_info) => { + let idx = f_info.index_within_tensor as usize; + let flat_index: usize = (r * cols + idx).try_into().unwrap(); + if flat_index < tensor.len() && idx < cols { + tensor[flat_index] = feature.1.into_inner() as f32; + } + } + None => (), + } + if self.continuous_features_to_report.contains(feature.0) { + self.continuous_feature_metrics + .with_label_values(&[feature.0.to_string().as_str()]) + .observe(feature.1.into_inner() as f64) + } else if self.discrete_features_to_report.contains(feature.0) { + self.discrete_feature_metrics + .with_label_values(&[feature.0.to_string().as_str()]) + .observe(feature.1.into_inner() as f64) + } + } + } + } + bpr_start = bpr_end; + } + + return InputTensor::FloatTensor( + Array2::::from_shape_vec( + [rows.try_into().unwrap(), cols.try_into().unwrap()], + tensor, + ) + .unwrap() + .into_dyn(), + ); + } + + fn get_binary(&self, bprs: &[BatchPredictionRequest], batch_ends: &[usize]) -> InputTensor { + // These need to be part of model schema + let rows: usize = batch_ends[batch_ends.len() - 1]; + let cols: usize = 149; + let full_size: usize = (rows * cols).try_into().unwrap(); + let default_val: i64 = 0; + + let mut v = vec![default_val; full_size]; + + let mut bpr_start = 0; + for (bpr, &bpr_end) in bprs.iter().zip(batch_ends) { + // Common features + if bpr.common_features.is_some() + && bpr + .common_features + .as_ref() + .unwrap() + .binary_features + .is_some() + { + let common_features = bpr + .common_features + .as_ref() + .unwrap() + .binary_features + .as_ref() + .unwrap(); + + for feature in common_features { + match self.feature_mapper.get(feature) { + Some(f_info) => { + let idx = f_info.index_within_tensor as usize; + if idx < cols { + // Set value in each row + for r in bpr_start..bpr_end { + let flat_index: usize = (r * cols + idx).try_into().unwrap(); + v[flat_index] = 1; + } + } + } + None => (), + } + } + } + + // Process the batch of datarecords + for r in bpr_start..bpr_end { + let dr: &DataRecord = + &bpr.individual_features_list[usize::try_from(r - bpr_start).unwrap()]; + if dr.binary_features.is_some() { + for feature in dr.binary_features.as_ref().unwrap() { + match self.feature_mapper.get(&feature) { + Some(f_info) => { + let idx = f_info.index_within_tensor as usize; + let flat_index: usize = (r * cols + idx).try_into().unwrap(); + v[flat_index] = 1; + } + None => (), + } + } + } + } + bpr_start = bpr_end; + } + return InputTensor::Int64Tensor( + Array2::::from_shape_vec([rows.try_into().unwrap(), cols.try_into().unwrap()], v) + .unwrap() + .into_dyn(), + ); + } + + #[allow(dead_code)] + fn get_discrete(&self, bprs: &[BatchPredictionRequest], batch_ends: &[usize]) -> InputTensor { + // These need to be part of model schema + let rows: usize = batch_ends[batch_ends.len() - 1]; + let cols: usize = 320; + let full_size: usize = (rows * cols).try_into().unwrap(); + let default_val: i64 = 0; + + let mut v = vec![default_val; full_size]; + + let mut bpr_start = 0; + for (bpr, &bpr_end) in bprs.iter().zip(batch_ends) { + // Common features + if bpr.common_features.is_some() + && bpr + .common_features + .as_ref() + .unwrap() + .discrete_features + .is_some() + { + let common_features = bpr + .common_features + .as_ref() + .unwrap() + .discrete_features + .as_ref() + .unwrap(); + + for feature in common_features { + match self.feature_mapper.get(feature.0) { + Some(f_info) => { + let idx = f_info.index_within_tensor as usize; + if idx < cols { + // Set value in each row + for r in bpr_start..bpr_end { + let flat_index: usize = (r * cols + idx).try_into().unwrap(); + v[flat_index] = *feature.1; + } + } + } + None => (), + } + if self.discrete_features_to_report.contains(feature.0) { + self.discrete_feature_metrics + .with_label_values(&[feature.0.to_string().as_str()]) + .observe(*feature.1 as f64) + } + } + } + + // Process the batch of datarecords + for r in bpr_start..bpr_end { + let dr: &DataRecord = &bpr.individual_features_list[usize::try_from(r).unwrap()]; + if dr.discrete_features.is_some() { + for feature in dr.discrete_features.as_ref().unwrap() { + match self.feature_mapper.get(&feature.0) { + Some(f_info) => { + let idx = f_info.index_within_tensor as usize; + let flat_index: usize = (r * cols + idx).try_into().unwrap(); + if flat_index < v.len() && idx < cols { + v[flat_index] = *feature.1; + } + } + None => (), + } + if self.discrete_features_to_report.contains(feature.0) { + self.discrete_feature_metrics + .with_label_values(&[feature.0.to_string().as_str()]) + .observe(*feature.1 as f64) + } + } + } + } + bpr_start = bpr_end; + } + return InputTensor::Int64Tensor( + Array2::::from_shape_vec([rows.try_into().unwrap(), cols.try_into().unwrap()], v) + .unwrap() + .into_dyn(), + ); + } + + fn get_user_embedding( + &self, + bprs: &[BatchPredictionRequest], + batch_ends: &[usize], + ) -> InputTensor { + InputTensor::FloatTensor( + self.get_embedding_tensors(bprs, self.user_embedding_feature_id, batch_ends) + .into_dyn(), + ) + } + + fn get_eng_embedding( + &self, + bpr: &[BatchPredictionRequest], + batch_ends: &[usize], + ) -> InputTensor { + InputTensor::FloatTensor( + self.get_embedding_tensors(bpr, self.user_eng_embedding_feature_id, batch_ends) + .into_dyn(), + ) + } + + fn get_author_embedding( + &self, + bpr: &[BatchPredictionRequest], + batch_ends: &[usize], + ) -> InputTensor { + InputTensor::FloatTensor( + self.get_embedding_tensors(bpr, self.author_embedding_feature_id, batch_ends) + .into_dyn(), + ) + } +} + +impl Converter for BatchPredictionRequestToTorchTensorConverter { + fn convert(&self, batched_bytes: Vec>) -> (Vec, Vec) { + let bprs = batched_bytes + .into_iter() + .map(|bytes| { + BatchPredictionRequestToTorchTensorConverter::parse_batch_prediction_request(bytes) + }) + .collect::>(); + let batch_ends = bprs + .iter() + .map(|bpr| bpr.individual_features_list.len()) + .scan(0usize, |acc, e| { + //running total + *acc = *acc + e; + Some(*acc) + }) + .collect::>(); + + let t1 = self.get_continuous(&bprs, &batch_ends); + let t2 = self.get_binary(&bprs, &batch_ends); + //let _t3 = self.get_discrete(&bprs, &batch_ends); + let t4 = self.get_user_embedding(&bprs, &batch_ends); + let t5 = self.get_eng_embedding(&bprs, &batch_ends); + let t6 = self.get_author_embedding(&bprs, &batch_ends); + + (vec![t1, t2, t4, t5, t6], batch_ends) + } +} diff --git a/navi/dr_transform/src/lib.rs b/navi/dr_transform/src/lib.rs new file mode 100644 index 0000000000..25b7cd2d3f --- /dev/null +++ b/navi/dr_transform/src/lib.rs @@ -0,0 +1,5 @@ +pub mod all_config; +pub mod converter; +#[cfg(test)] +mod test; +pub mod util; diff --git a/navi/dr_transform/src/util.rs b/navi/dr_transform/src/util.rs new file mode 100644 index 0000000000..8c87731856 --- /dev/null +++ b/navi/dr_transform/src/util.rs @@ -0,0 +1,30 @@ +use npyz::WriterBuilder; +use npyz::{AutoSerialize, WriteOptions}; +use std::io::BufWriter; +use std::{ + fs::File, + io::{self, BufRead}, +}; + +pub fn load_batch_prediction_request_base64(file_name: &str) -> Vec> { + let file = File::open(file_name).expect("could not read file"); + let mut result = vec![]; + for line in io::BufReader::new(file).lines() { + match base64::decode(line.unwrap().trim()) { + Ok(payload) => result.push(payload), + Err(err) => println!("error decoding line {}", err), + } + } + println!("reslt len: {}", result.len()); + return result; +} +pub fn save_to_npy(data: &[T], save_to: String) { + let mut writer = WriteOptions::new() + .default_dtype() + .shape(&[data.len() as u64, 1]) + .writer(BufWriter::new(File::create(save_to).unwrap())) + .begin_nd() + .unwrap(); + writer.extend(data.to_owned()).unwrap(); + writer.finish().unwrap(); +} diff --git a/navi/navi/Cargo.toml b/navi/navi/Cargo.toml new file mode 100644 index 0000000000..a942b1ae4a --- /dev/null +++ b/navi/navi/Cargo.toml @@ -0,0 +1,76 @@ +[package] +name = "navi" +version = "2.0.42" +edition = "2021" +# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html + +[[bin]] +name = "navi" +path = "src/bin/navi.rs" +required-features=["tf"] +[[bin]] +name = "navi_torch" +path = "src/bin/navi_torch.rs" +required-features=["torch"] +[[bin]] +name = "navi_onnx" +path = "src/bin/navi_onnx.rs" +required-features=["onnx"] + +[features] +default=[] +navi_console=[] +torch=["tch"] +onnx=["ort"] +tf=["tensorflow"] +[dependencies] +itertools = "0.10.5" +anyhow = "1.0.57" +arrayvec = "0.7.2" +clap = { version = "4.0.32", features = ["derive"] } +console-subscriber = "0.1.6" +time = { version = "0.3.20", features = ["parsing"] } +env_logger = "0.10.0" +flamegraph = "0.6.1" +fnv = "1.0.7" +futures = { version = "0.3", default-features = false } +image = "0.24.5" +indexmap = "1.8.1" +lazy_static = "1.4" +libloading = "0.7" +log = "0.4.17" +ndarray-rand = "0.14.0" +prometheus = "0.13.1" +prost = "0.9" +prost-types = "0.9" +parking_lot = "0.12.1" +rand = "0.8.5" +rand_pcg = "0.3.1" +random = "0.12.2" +sha256 = "1.0.3" +tonic = { version = "0.6.2", features=['compression', 'tls'] } +tokio = { version = "1.17.0", features = ["macros", "rt-multi-thread", "fs", "process"] } +warp = "0.3" +npyz = "0.7.3" +base64 = "0.21.0" +histogram = "0.6.9" +tch = {version = "0.10.3", optional = true} +tensorflow = { version = "0.20.0", optional = true } +once_cell = {version = "1.17.1"} +ndarray = "0.15" +serde = "1.0.154" +serde_json = "1.0.94" +dr_transform = { path = "../dr_transform"} +[target.'cfg(not(target_os="linux"))'.dependencies] +ort = {git ="https://github.com/pykeio/ort.git", features=["profiling"], optional = true, tag="v1.14.2"} +[target.'cfg(target_os="linux")'.dependencies] +ort = {git ="https://github.com/pykeio/ort.git", features=["profiling", "tensorrt", "cuda", "copy-dylibs"], optional = true, tag="v1.14.2"} +[build-dependencies] +tonic-build = {version = "0.6.2", features=['prost', "compression"] } +[profile.release] +debug = true +[dev-dependencies] +ndarray-rand = "0.14.0" +tokio-test = "*" +assert_cmd = "2.0" +criterion = "0.4.0" diff --git a/navi/navi/README.md b/navi/navi/README.md new file mode 100644 index 0000000000..d8962daf45 --- /dev/null +++ b/navi/navi/README.md @@ -0,0 +1,34 @@ +# Navi: High-Performance Machine Learning Serving Server in Rust + +Navi is a high-performance, versatile machine learning serving server implemented in Rust, tailored for production usage. It's designed to efficiently serve within the Twitter tech stack, offering top-notch performance while focusing on core features. + +## Key Features + +- **Minimalist Design Optimized for Production Use Cases**: Navi delivers ultra-high performance, stability, and availability, engineered to handle real-world application demands with a streamlined codebase. +- **gRPC API Compatibility with TensorFlow Serving**: Seamless integration with existing TensorFlow Serving clients via its gRPC API, enabling easy integration, smooth deployment, and scaling in production environments. +- **Plugin Architecture for Different Runtimes**: Navi's pluggable architecture supports various machine learning runtimes, providing adaptability and extensibility for diverse use cases. Out-of-the-box support is available for TensorFlow and Onnx Runtime, with PyTorch in an experimental state. + +## Current State + +While Navi's features may not be as comprehensive as its open-source counterparts, its performance-first mindset makes it highly efficient. +- Navi for TensorFlow is currently the most feature-complete, supporting multiple input tensors of different types (float, int, string, etc.). +- Navi for Onnx primarily supports one input tensor of type string, used in Twitter's home recommendation with a proprietary BatchPredictRequest format. +- Navi for Pytorch is compilable and runnable but not yet production-ready in terms of performance and stability. + +## Directory Structure + +- `navi`: The main code repository for Navi +- `dr_transform`: Twitter-specific converter that converts BatchPredictionRequest Thrift to ndarray +- `segdense`: Twitter-specific config to specify how to retrieve feature values from BatchPredictionRequest +- `thrift_bpr_adapter`: generated thrift code for BatchPredictionRequest + +## Content +We include all *.rs source code that makes up the main navi binaries for you to examine. The test and benchmark code, as well as configuration files are not included due to data security concerns. + +## Run +in navi/navi you can run. Note you need to create a models directory and create some versions, preferably using epoch time, e.g., 1679693908377 +- scripts/run_tf2.sh +- scripts/run_onnx.sh + +## Build +you can adapt the above scripts to build using Cargo diff --git a/navi/navi/build.rs b/navi/navi/build.rs new file mode 100644 index 0000000000..8757a18231 --- /dev/null +++ b/navi/navi/build.rs @@ -0,0 +1,13 @@ +fn main() -> Result<(), Box> { + //::compile_protos("proto/tensorflow_serving/apis/prediction_service.proto")?; + tonic_build::configure().compile( + &[ + "proto/tensorflow_serving/apis/prediction_service.proto", + "proto/tensorflow/core/protobuf/config.proto", + "proto/tensorflow_serving/apis/prediction_log.proto", + "proto/kfserving/grpc_predict_v2.proto", + ], + &["proto"], + )?; + Ok(()) +} diff --git a/navi/navi/proto/kfserving/grpc_predict_v2.proto b/navi/navi/proto/kfserving/grpc_predict_v2.proto new file mode 100644 index 0000000000..6b2475a2e8 --- /dev/null +++ b/navi/navi/proto/kfserving/grpc_predict_v2.proto @@ -0,0 +1,326 @@ +// Copyright 2020 kubeflow.org. +// +// 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. + +syntax = "proto3"; +package inference; + +// Inference Server GRPC endpoints. +service GRPCInferenceService +{ + // The ServerLive API indicates if the inference server is able to receive + // and respond to metadata and inference requests. + rpc ServerLive(ServerLiveRequest) returns (ServerLiveResponse) {} + + // The ServerReady API indicates if the server is ready for inferencing. + rpc ServerReady(ServerReadyRequest) returns (ServerReadyResponse) {} + + // The ModelReady API indicates if a specific model is ready for inferencing. + rpc ModelReady(ModelReadyRequest) returns (ModelReadyResponse) {} + + // The ServerMetadata API provides information about the server. Errors are + // indicated by the google.rpc.Status returned for the request. The OK code + // indicates success and other codes indicate failure. + rpc ServerMetadata(ServerMetadataRequest) returns (ServerMetadataResponse) {} + + // The per-model metadata API provides information about a model. Errors are + // indicated by the google.rpc.Status returned for the request. The OK code + // indicates success and other codes indicate failure. + rpc ModelMetadata(ModelMetadataRequest) returns (ModelMetadataResponse) {} + + // The ModelInfer API performs inference using the specified model. Errors are + // indicated by the google.rpc.Status returned for the request. The OK code + // indicates success and other codes indicate failure. + rpc ModelInfer(ModelInferRequest) returns (ModelInferResponse) {} +} + +message ServerLiveRequest {} + +message ServerLiveResponse +{ + // True if the inference server is live, false if not live. + bool live = 1; +} + +message ServerReadyRequest {} + +message ServerReadyResponse +{ + // True if the inference server is ready, false if not ready. + bool ready = 1; +} + +message ModelReadyRequest +{ + // The name of the model to check for readiness. + string name = 1; + + // The version of the model to check for readiness. If not given the + // server will choose a version based on the model and internal policy. + string version = 2; +} + +message ModelReadyResponse +{ + // True if the model is ready, false if not ready. + bool ready = 1; +} + +message ServerMetadataRequest {} + +message ServerMetadataResponse +{ + // The server name. + string name = 1; + + // The server version. + string version = 2; + + // The extensions supported by the server. + repeated string extensions = 3; +} + +message ModelMetadataRequest +{ + // The name of the model. + string name = 1; + + // The version of the model to check for readiness. If not given the + // server will choose a version based on the model and internal policy. + string version = 2; +} + +message ModelMetadataResponse +{ + // Metadata for a tensor. + message TensorMetadata + { + // The tensor name. + string name = 1; + + // The tensor data type. + string datatype = 2; + + // The tensor shape. A variable-size dimension is represented + // by a -1 value. + repeated int64 shape = 3; + } + + // The model name. + string name = 1; + + // The versions of the model available on the server. + repeated string versions = 2; + + // The model's platform. See Platforms. + string platform = 3; + + // The model's inputs. + repeated TensorMetadata inputs = 4; + + // The model's outputs. + repeated TensorMetadata outputs = 5; +} + +message ModelInferRequest +{ + // An input tensor for an inference request. + message InferInputTensor + { + // The tensor name. + string name = 1; + + // The tensor data type. + string datatype = 2; + + // The tensor shape. + repeated int64 shape = 3; + + // Optional inference input tensor parameters. + map parameters = 4; + + // The tensor contents using a data-type format. This field must + // not be specified if "raw" tensor contents are being used for + // the inference request. + InferTensorContents contents = 5; + } + + // An output tensor requested for an inference request. + message InferRequestedOutputTensor + { + // The tensor name. + string name = 1; + + // Optional requested output tensor parameters. + map parameters = 2; + } + + // The name of the model to use for inferencing. + string model_name = 1; + + // The version of the model to use for inference. If not given the + // server will choose a version based on the model and internal policy. + string model_version = 2; + + // Optional identifier for the request. If specified will be + // returned in the response. + string id = 3; + + // Optional inference parameters. + map parameters = 4; + + // The input tensors for the inference. + repeated InferInputTensor inputs = 5; + + // The requested output tensors for the inference. Optional, if not + // specified all outputs produced by the model will be returned. + repeated InferRequestedOutputTensor outputs = 6; + + // The data contained in an input tensor can be represented in "raw" + // bytes form or in the repeated type that matches the tensor's data + // type. To use the raw representation 'raw_input_contents' must be + // initialized with data for each tensor in the same order as + // 'inputs'. For each tensor, the size of this content must match + // what is expected by the tensor's shape and data type. The raw + // data must be the flattened, one-dimensional, row-major order of + // the tensor elements without any stride or padding between the + // elements. Note that the FP16 and BF16 data types must be represented as + // raw content as there is no specific data type for a 16-bit float type. + // + // If this field is specified then InferInputTensor::contents must + // not be specified for any input tensor. + repeated bytes raw_input_contents = 7; +} + +message ModelInferResponse +{ + // An output tensor returned for an inference request. + message InferOutputTensor + { + // The tensor name. + string name = 1; + + // The tensor data type. + string datatype = 2; + + // The tensor shape. + repeated int64 shape = 3; + + // Optional output tensor parameters. + map parameters = 4; + + // The tensor contents using a data-type format. This field must + // not be specified if "raw" tensor contents are being used for + // the inference response. + InferTensorContents contents = 5; + } + + // The name of the model used for inference. + string model_name = 1; + + // The version of the model used for inference. + string model_version = 2; + + // The id of the inference request if one was specified. + string id = 3; + + // Optional inference response parameters. + map parameters = 4; + + // The output tensors holding inference results. + repeated InferOutputTensor outputs = 5; + + // The data contained in an output tensor can be represented in + // "raw" bytes form or in the repeated type that matches the + // tensor's data type. To use the raw representation 'raw_output_contents' + // must be initialized with data for each tensor in the same order as + // 'outputs'. For each tensor, the size of this content must match + // what is expected by the tensor's shape and data type. The raw + // data must be the flattened, one-dimensional, row-major order of + // the tensor elements without any stride or padding between the + // elements. Note that the FP16 and BF16 data types must be represented as + // raw content as there is no specific data type for a 16-bit float type. + // + // If this field is specified then InferOutputTensor::contents must + // not be specified for any output tensor. + repeated bytes raw_output_contents = 6; +} + +// An inference parameter value. The Parameters message describes a +// “name”/”value” pair, where the “name” is the name of the parameter +// and the “value” is a boolean, integer, or string corresponding to +// the parameter. +message InferParameter +{ + // The parameter value can be a string, an int64, a boolean + // or a message specific to a predefined parameter. + oneof parameter_choice + { + // A boolean parameter value. + bool bool_param = 1; + + // An int64 parameter value. + int64 int64_param = 2; + + // A string parameter value. + string string_param = 3; + } +} + +// The data contained in a tensor represented by the repeated type +// that matches the tensor's data type. Protobuf oneof is not used +// because oneofs cannot contain repeated fields. +message InferTensorContents +{ + // Representation for BOOL data type. The size must match what is + // expected by the tensor's shape. The contents must be the flattened, + // one-dimensional, row-major order of the tensor elements. + repeated bool bool_contents = 1; + + // Representation for INT8, INT16, and INT32 data types. The size + // must match what is expected by the tensor's shape. The contents + // must be the flattened, one-dimensional, row-major order of the + // tensor elements. + repeated int32 int_contents = 2; + + // Representation for INT64 data types. The size must match what + // is expected by the tensor's shape. The contents must be the + // flattened, one-dimensional, row-major order of the tensor elements. + repeated int64 int64_contents = 3; + + // Representation for UINT8, UINT16, and UINT32 data types. The size + // must match what is expected by the tensor's shape. The contents + // must be the flattened, one-dimensional, row-major order of the + // tensor elements. + repeated uint32 uint_contents = 4; + + // Representation for UINT64 data types. The size must match what + // is expected by the tensor's shape. The contents must be the + // flattened, one-dimensional, row-major order of the tensor elements. + repeated uint64 uint64_contents = 5; + + // Representation for FP32 data type. The size must match what is + // expected by the tensor's shape. The contents must be the flattened, + // one-dimensional, row-major order of the tensor elements. + repeated float fp32_contents = 6; + + // Representation for FP64 data type. The size must match what is + // expected by the tensor's shape. The contents must be the flattened, + // one-dimensional, row-major order of the tensor elements. + repeated double fp64_contents = 7; + + // Representation for BYTES data type. The size must match what is + // expected by the tensor's shape. The contents must be the flattened, + // one-dimensional, row-major order of the tensor elements. + repeated bytes bytes_contents = 8; +} diff --git a/navi/navi/proto/tensorflow/core/example/example.proto b/navi/navi/proto/tensorflow/core/example/example.proto new file mode 100644 index 0000000000..0b49514e5e --- /dev/null +++ b/navi/navi/proto/tensorflow/core/example/example.proto @@ -0,0 +1,306 @@ +// Protocol messages for describing input data Examples for machine learning +// model training or inference. +syntax = "proto3"; + +package tensorflow; + +import "tensorflow/core/example/feature.proto"; + +option cc_enable_arenas = true; +option java_outer_classname = "ExampleProtos"; +option java_multiple_files = true; +option java_package = "org.tensorflow.example"; +option go_package = "github.com/tensorflow/tensorflow/tensorflow/go/core/example"; + +// LINT.IfChange +// An Example is a mostly-normalized data format for storing data for +// training and inference. It contains a key-value store (features); where +// each key (string) maps to a Feature message (which is oneof packed BytesList, +// FloatList, or Int64List). This flexible and compact format allows the +// storage of large amounts of typed data, but requires that the data shape +// and use be determined by the configuration files and parsers that are used to +// read and write this format. That is, the Example is mostly *not* a +// self-describing format. In TensorFlow, Examples are read in row-major +// format, so any configuration that describes data with rank-2 or above +// should keep this in mind. For example, to store an M x N matrix of Bytes, +// the BytesList must contain M*N bytes, with M rows of N contiguous values +// each. That is, the BytesList value must store the matrix as: +// .... row 0 .... .... row 1 .... // ........... // ... row M-1 .... +// +// An Example for a movie recommendation application: +// features { +// feature { +// key: "age" +// value { float_list { +// value: 29.0 +// }} +// } +// feature { +// key: "movie" +// value { bytes_list { +// value: "The Shawshank Redemption" +// value: "Fight Club" +// }} +// } +// feature { +// key: "movie_ratings" +// value { float_list { +// value: 9.0 +// value: 9.7 +// }} +// } +// feature { +// key: "suggestion" +// value { bytes_list { +// value: "Inception" +// }} +// } +// # Note that this feature exists to be used as a label in training. +// # E.g., if training a logistic regression model to predict purchase +// # probability in our learning tool we would set the label feature to +// # "suggestion_purchased". +// feature { +// key: "suggestion_purchased" +// value { float_list { +// value: 1.0 +// }} +// } +// # Similar to "suggestion_purchased" above this feature exists to be used +// # as a label in training. +// # E.g., if training a linear regression model to predict purchase +// # price in our learning tool we would set the label feature to +// # "purchase_price". +// feature { +// key: "purchase_price" +// value { float_list { +// value: 9.99 +// }} +// } +// } +// +// A conformant Example data set obeys the following conventions: +// - If a Feature K exists in one example with data type T, it must be of +// type T in all other examples when present. It may be omitted. +// - The number of instances of Feature K list data may vary across examples, +// depending on the requirements of the model. +// - If a Feature K doesn't exist in an example, a K-specific default will be +// used, if configured. +// - If a Feature K exists in an example but contains no items, the intent +// is considered to be an empty tensor and no default will be used. + +message Example { + Features features = 1; +} + +// A SequenceExample is an Example representing one or more sequences, and +// some context. The context contains features which apply to the entire +// example. The feature_lists contain a key, value map where each key is +// associated with a repeated set of Features (a FeatureList). +// A FeatureList thus represents the values of a feature identified by its key +// over time / frames. +// +// Below is a SequenceExample for a movie recommendation application recording a +// sequence of ratings by a user. The time-independent features ("locale", +// "age", "favorites") describing the user are part of the context. The sequence +// of movies the user rated are part of the feature_lists. For each movie in the +// sequence we have information on its name and actors and the user's rating. +// This information is recorded in three separate feature_list(s). +// In the example below there are only two movies. All three feature_list(s), +// namely "movie_ratings", "movie_names", and "actors" have a feature value for +// both movies. Note, that "actors" is itself a bytes_list with multiple +// strings per movie. +// +// context: { +// feature: { +// key : "locale" +// value: { +// bytes_list: { +// value: [ "pt_BR" ] +// } +// } +// } +// feature: { +// key : "age" +// value: { +// float_list: { +// value: [ 19.0 ] +// } +// } +// } +// feature: { +// key : "favorites" +// value: { +// bytes_list: { +// value: [ "Majesty Rose", "Savannah Outen", "One Direction" ] +// } +// } +// } +// } +// feature_lists: { +// feature_list: { +// key : "movie_ratings" +// value: { +// feature: { +// float_list: { +// value: [ 4.5 ] +// } +// } +// feature: { +// float_list: { +// value: [ 5.0 ] +// } +// } +// } +// } +// feature_list: { +// key : "movie_names" +// value: { +// feature: { +// bytes_list: { +// value: [ "The Shawshank Redemption" ] +// } +// } +// feature: { +// bytes_list: { +// value: [ "Fight Club" ] +// } +// } +// } +// } +// feature_list: { +// key : "actors" +// value: { +// feature: { +// bytes_list: { +// value: [ "Tim Robbins", "Morgan Freeman" ] +// } +// } +// feature: { +// bytes_list: { +// value: [ "Brad Pitt", "Edward Norton", "Helena Bonham Carter" ] +// } +// } +// } +// } +// } +// +// A conformant SequenceExample data set obeys the following conventions: +// +// Context: +// - All conformant context features K must obey the same conventions as +// a conformant Example's features (see above). +// Feature lists: +// - A FeatureList L may be missing in an example; it is up to the +// parser configuration to determine if this is allowed or considered +// an empty list (zero length). +// - If a FeatureList L exists, it may be empty (zero length). +// - If a FeatureList L is non-empty, all features within the FeatureList +// must have the same data type T. Even across SequenceExamples, the type T +// of the FeatureList identified by the same key must be the same. An entry +// without any values may serve as an empty feature. +// - If a FeatureList L is non-empty, it is up to the parser configuration +// to determine if all features within the FeatureList must +// have the same size. The same holds for this FeatureList across multiple +// examples. +// - For sequence modeling, e.g.: +// http://colah.github.io/posts/2015-08-Understanding-LSTMs/ +// https://github.com/tensorflow/nmt +// the feature lists represent a sequence of frames. +// In this scenario, all FeatureLists in a SequenceExample have the same +// number of Feature messages, so that the ith element in each FeatureList +// is part of the ith frame (or time step). +// Examples of conformant and non-conformant examples' FeatureLists: +// +// Conformant FeatureLists: +// feature_lists: { feature_list: { +// key: "movie_ratings" +// value: { feature: { float_list: { value: [ 4.5 ] } } +// feature: { float_list: { value: [ 5.0 ] } } } +// } } +// +// Non-conformant FeatureLists (mismatched types): +// feature_lists: { feature_list: { +// key: "movie_ratings" +// value: { feature: { float_list: { value: [ 4.5 ] } } +// feature: { int64_list: { value: [ 5 ] } } } +// } } +// +// Conditionally conformant FeatureLists, the parser configuration determines +// if the feature sizes must match: +// feature_lists: { feature_list: { +// key: "movie_ratings" +// value: { feature: { float_list: { value: [ 4.5 ] } } +// feature: { float_list: { value: [ 5.0, 6.0 ] } } } +// } } +// +// Conformant pair of SequenceExample +// feature_lists: { feature_list: { +// key: "movie_ratings" +// value: { feature: { float_list: { value: [ 4.5 ] } } +// feature: { float_list: { value: [ 5.0 ] } } } +// } } +// and: +// feature_lists: { feature_list: { +// key: "movie_ratings" +// value: { feature: { float_list: { value: [ 4.5 ] } } +// feature: { float_list: { value: [ 5.0 ] } } +// feature: { float_list: { value: [ 2.0 ] } } } +// } } +// +// Conformant pair of SequenceExample +// feature_lists: { feature_list: { +// key: "movie_ratings" +// value: { feature: { float_list: { value: [ 4.5 ] } } +// feature: { float_list: { value: [ 5.0 ] } } } +// } } +// and: +// feature_lists: { feature_list: { +// key: "movie_ratings" +// value: { } +// } } +// +// Conditionally conformant pair of SequenceExample, the parser configuration +// determines if the second feature_lists is consistent (zero-length) or +// invalid (missing "movie_ratings"): +// feature_lists: { feature_list: { +// key: "movie_ratings" +// value: { feature: { float_list: { value: [ 4.5 ] } } +// feature: { float_list: { value: [ 5.0 ] } } } +// } } +// and: +// feature_lists: { } +// +// Non-conformant pair of SequenceExample (mismatched types) +// feature_lists: { feature_list: { +// key: "movie_ratings" +// value: { feature: { float_list: { value: [ 4.5 ] } } +// feature: { float_list: { value: [ 5.0 ] } } } +// } } +// and: +// feature_lists: { feature_list: { +// key: "movie_ratings" +// value: { feature: { int64_list: { value: [ 4 ] } } +// feature: { int64_list: { value: [ 5 ] } } +// feature: { int64_list: { value: [ 2 ] } } } +// } } +// +// Conditionally conformant pair of SequenceExample; the parser configuration +// determines if the feature sizes must match: +// feature_lists: { feature_list: { +// key: "movie_ratings" +// value: { feature: { float_list: { value: [ 4.5 ] } } +// feature: { float_list: { value: [ 5.0 ] } } } +// } } +// and: +// feature_lists: { feature_list: { +// key: "movie_ratings" +// value: { feature: { float_list: { value: [ 4.0 ] } } +// feature: { float_list: { value: [ 5.0, 3.0 ] } } +// } } + +message SequenceExample { + Features context = 1; + FeatureLists feature_lists = 2; +} +// LINT.ThenChange( +// https://www.tensorflow.org/code/tensorflow/python/training/training.py) diff --git a/navi/navi/proto/tensorflow/core/example/feature.proto b/navi/navi/proto/tensorflow/core/example/feature.proto new file mode 100644 index 0000000000..e532747f35 --- /dev/null +++ b/navi/navi/proto/tensorflow/core/example/feature.proto @@ -0,0 +1,110 @@ +// Protocol messages for describing features for machine learning model +// training or inference. +// +// There are three base Feature types: +// - bytes +// - float +// - int64 +// +// A Feature contains Lists which may hold zero or more values. These +// lists are the base values BytesList, FloatList, Int64List. +// +// Features are organized into categories by name. The Features message +// contains the mapping from name to Feature. +// +// Example Features for a movie recommendation application: +// feature { +// key: "age" +// value { float_list { +// value: 29.0 +// }} +// } +// feature { +// key: "movie" +// value { bytes_list { +// value: "The Shawshank Redemption" +// value: "Fight Club" +// }} +// } +// feature { +// key: "movie_ratings" +// value { float_list { +// value: 9.0 +// value: 9.7 +// }} +// } +// feature { +// key: "suggestion" +// value { bytes_list { +// value: "Inception" +// }} +// } +// feature { +// key: "suggestion_purchased" +// value { int64_list { +// value: 1 +// }} +// } +// feature { +// key: "purchase_price" +// value { float_list { +// value: 9.99 +// }} +// } +// + +syntax = "proto3"; + +package tensorflow; + +option cc_enable_arenas = true; +option java_outer_classname = "FeatureProtos"; +option java_multiple_files = true; +option java_package = "org.tensorflow.example"; +option go_package = "github.com/tensorflow/tensorflow/tensorflow/go/core/example"; + +// LINT.IfChange +// Containers to hold repeated fundamental values. +message BytesList { + repeated bytes value = 1; +} +message FloatList { + repeated float value = 1 [packed = true]; +} +message Int64List { + repeated int64 value = 1 [packed = true]; +} + +// Containers for non-sequential data. +message Feature { + // Each feature can be exactly one kind. + oneof kind { + BytesList bytes_list = 1; + FloatList float_list = 2; + Int64List int64_list = 3; + } +} + +message Features { + // Map from feature name to feature. + map feature = 1; +} + +// Containers for sequential data. +// +// A FeatureList contains lists of Features. These may hold zero or more +// Feature values. +// +// FeatureLists are organized into categories by name. The FeatureLists message +// contains the mapping from name to FeatureList. +// +message FeatureList { + repeated Feature feature = 1; +} + +message FeatureLists { + // Map from feature name to feature list. + map feature_list = 1; +} +// LINT.ThenChange( +// https://www.tensorflow.org/code/tensorflow/python/training/training.py) diff --git a/navi/navi/proto/tensorflow/core/framework/allocation_description.proto b/navi/navi/proto/tensorflow/core/framework/allocation_description.proto new file mode 100644 index 0000000000..f18caa40b2 --- /dev/null +++ b/navi/navi/proto/tensorflow/core/framework/allocation_description.proto @@ -0,0 +1,29 @@ +syntax = "proto3"; + +package tensorflow; + +option cc_enable_arenas = true; +option java_outer_classname = "AllocationDescriptionProtos"; +option java_multiple_files = true; +option java_package = "org.tensorflow.framework"; +option go_package = "github.com/tensorflow/tensorflow/tensorflow/go/core/framework/allocation_description_go_proto"; + +message AllocationDescription { + // Total number of bytes requested + int64 requested_bytes = 1; + + // Total number of bytes allocated if known + int64 allocated_bytes = 2; + + // Name of the allocator used + string allocator_name = 3; + + // Identifier of the allocated buffer if known + int64 allocation_id = 4; + + // Set if this tensor only has one remaining reference + bool has_single_reference = 5; + + // Address of the allocation. + uint64 ptr = 6; +} diff --git a/navi/navi/proto/tensorflow/core/framework/api_def.proto b/navi/navi/proto/tensorflow/core/framework/api_def.proto new file mode 100644 index 0000000000..1823ce64f2 --- /dev/null +++ b/navi/navi/proto/tensorflow/core/framework/api_def.proto @@ -0,0 +1,138 @@ +// Defines the text format for including per-op API definition and +// overrides for client language op code generators. + +syntax = "proto3"; + +package tensorflow; + +import "tensorflow/core/framework/attr_value.proto"; + +option cc_enable_arenas = true; +option java_outer_classname = "ApiDefProtos"; +option java_multiple_files = true; +option java_package = "org.tensorflow.framework"; +option go_package = "github.com/tensorflow/tensorflow/tensorflow/go/core/framework/api_def_go_proto"; + +// Used to specify and override the default API & behavior in the +// generated code for client languages, from what you would get from +// the OpDef alone. There will be a set of ApiDefs that are common +// to all client languages, and another set per client language. +// The per-client-language ApiDefs will inherit values from the +// common ApiDefs which it can either replace or modify. +// +// We separate the API definition from the OpDef so we can evolve the +// API while remaining backwards compatible when interpreting old +// graphs. Overrides go in an "api_def.pbtxt" file with a text-format +// ApiDefs message. +// +// WARNING: Be *very* careful changing the API for any existing op -- +// you can change the semantics of existing code. These changes may +// need to wait until a major release of TensorFlow to avoid breaking +// our compatibility promises. +message ApiDef { + // Name of the op (in the OpDef) to specify the API for. + string graph_op_name = 1; + // If this op is deprecated, set deprecation message to the message + // that should be logged when this op is used. + // The message should indicate alternative op to use, if any. + string deprecation_message = 12; + // Major version when the op will be deleted. For e.g. set this + // value to 2 if op API should be removed in TensorFlow 2.0 and + // deprecated in versions before that. + int32 deprecation_version = 13; + + enum Visibility { + // Normally this is "VISIBLE" unless you are inheriting a + // different value from another ApiDef. + DEFAULT_VISIBILITY = 0; + // Publicly visible in the API. + VISIBLE = 1; + // Do not include this op in the generated API. If visibility is + // set to 'SKIP', other fields are ignored for this op. + SKIP = 2; + // Hide this op by putting it into an internal namespace (or whatever + // is appropriate in the target language). + HIDDEN = 3; + } + Visibility visibility = 2; + + // If you specify any endpoint, this will replace all of the + // inherited endpoints. The first endpoint should be the + // "canonical" endpoint, and should not be deprecated (unless all + // endpoints are deprecated). + message Endpoint { + // Name should be either like "CamelCaseName" or + // "Package.CamelCaseName". Client-language-specific ApiDefs may + // use a snake_case convention instead of CamelCase. + string name = 1; + + // Set if this endpoint is deprecated. If set to true, a message suggesting + // to use a non-deprecated endpoint instead will be printed. If all + // endpoints are deprecated, set deprecation_message in ApiDef instead. + bool deprecated = 3; + + // Major version when an endpoint will be deleted. For e.g. set this + // value to 2 if endpoint should be removed in TensorFlow 2.0 and + // deprecated in versions before that. + int32 deprecation_version = 4; + } + repeated Endpoint endpoint = 3; + + message Arg { + string name = 1; + + // Change the name used to access this arg in the API from what + // is used in the GraphDef. Note that these names in `backticks` + // will also be replaced in the summary & description fields. + string rename_to = 2; + + // Note: this will replace any inherited arg doc. There is no + // current way of modifying arg descriptions (other than replacing + // them entirely) as can be done with op descriptions. + string description = 3; + } + repeated Arg in_arg = 4; + repeated Arg out_arg = 5; + // List of original in_arg names to specify new argument order. + // Length of arg_order should be either empty to keep current order + // or match size of in_arg. + repeated string arg_order = 11; + + // Description of the graph-construction-time configuration of this + // Op. That is to say, this describes the attr fields that will + // be specified in the NodeDef. + message Attr { + string name = 1; + + // Change the name used to access this attr in the API from what + // is used in the GraphDef. Note that these names in `backticks` + // will also be replaced in the summary & description fields. + string rename_to = 2; + + // Specify a new default value to use for this attr. This default + // will be used when creating new graphs, as opposed to the + // default in the OpDef, which will be used when interpreting old + // GraphDefs. + AttrValue default_value = 3; + + // Note: this will replace any inherited attr doc, there is no current + // way of modifying attr descriptions as can be done with op descriptions. + string description = 4; + } + repeated Attr attr = 6; + + // One-line human-readable description of what the Op does. + string summary = 7; + + // Additional, longer human-readable description of what the Op does. + string description = 8; + + // Modify an existing/inherited description by adding text to the beginning + // or end. + string description_prefix = 9; + string description_suffix = 10; +} + +message ApiDefs { + repeated ApiDef op = 1; +} diff --git a/navi/navi/proto/tensorflow/core/framework/attr_value.proto b/navi/navi/proto/tensorflow/core/framework/attr_value.proto new file mode 100644 index 0000000000..2e913130df --- /dev/null +++ b/navi/navi/proto/tensorflow/core/framework/attr_value.proto @@ -0,0 +1,64 @@ +syntax = "proto3"; + +package tensorflow; + +import "tensorflow/core/framework/tensor.proto"; +import "tensorflow/core/framework/tensor_shape.proto"; +import "tensorflow/core/framework/types.proto"; + +option cc_enable_arenas = true; +option java_outer_classname = "AttrValueProtos"; +option java_multiple_files = true; +option java_package = "org.tensorflow.framework"; +option go_package = "github.com/tensorflow/tensorflow/tensorflow/go/core/framework/attr_value_go_proto"; + +// Protocol buffer representing the value for an attr used to configure an Op. +// Comment indicates the corresponding attr type. Only the field matching the +// attr type may be filled. +message AttrValue { + // LINT.IfChange + message ListValue { + repeated bytes s = 2; // "list(string)" + repeated int64 i = 3 [packed = true]; // "list(int)" + repeated float f = 4 [packed = true]; // "list(float)" + repeated bool b = 5 [packed = true]; // "list(bool)" + repeated DataType type = 6 [packed = true]; // "list(type)" + repeated TensorShapeProto shape = 7; // "list(shape)" + repeated TensorProto tensor = 8; // "list(tensor)" + repeated NameAttrList func = 9; // "list(attr)" + } + // LINT.ThenChange(https://www.tensorflow.org/code/tensorflow/c/c_api.cc) + + oneof value { + bytes s = 2; // "string" + int64 i = 3; // "int" + float f = 4; // "float" + bool b = 5; // "bool" + DataType type = 6; // "type" + TensorShapeProto shape = 7; // "shape" + TensorProto tensor = 8; // "tensor" + ListValue list = 1; // any "list(...)" + + // "func" represents a function. func.name is a function's name or + // a primitive op's name. func.attr.first is the name of an attr + // defined for that function. func.attr.second is the value for + // that attr in the instantiation. + NameAttrList func = 10; + + // This is a placeholder only used in nodes defined inside a + // function. It indicates the attr value will be supplied when + // the function is instantiated. For example, let us suppose a + // node "N" in function "FN". "N" has an attr "A" with value + // placeholder = "foo". When FN is instantiated with attr "foo" + // set to "bar", the instantiated node N's attr A will have been + // given the value "bar". + string placeholder = 9; + } +} + +// A list of attr names and their values. The whole list is attached +// with a string name. E.g., MatMul[T=float]. +message NameAttrList { + string name = 1; + map attr = 2; +} diff --git a/navi/navi/proto/tensorflow/core/framework/cost_graph.proto b/navi/navi/proto/tensorflow/core/framework/cost_graph.proto new file mode 100644 index 0000000000..42c9e23cfa --- /dev/null +++ b/navi/navi/proto/tensorflow/core/framework/cost_graph.proto @@ -0,0 +1,89 @@ +syntax = "proto3"; + +package tensorflow; + +import "tensorflow/core/framework/tensor_shape.proto"; +import "tensorflow/core/framework/types.proto"; + +option cc_enable_arenas = true; +option java_outer_classname = "CostGraphProtos"; +option java_multiple_files = true; +option java_package = "org.tensorflow.framework"; +option go_package = "github.com/tensorflow/tensorflow/tensorflow/go/core/framework/cost_graph_go_proto"; + +message CostGraphDef { + message Node { + // The name of the node. Names are globally unique. + string name = 1; + + // The device of the node. Can be empty if the node is mapped to the + // default partition or partitioning hasn't been run yet. + string device = 2; + + // The id of the node. Node ids are only unique inside a partition. + int32 id = 3; + + // Inputs of this node. They must be executed before this node can be + // executed. An input is a particular output of another node, specified + // by the node id and the output index. + message InputInfo { + int32 preceding_node = 1; + int32 preceding_port = 2; + } + repeated InputInfo input_info = 4; + + // Outputs of this node. + message OutputInfo { + int64 size = 1; + // If >= 0, the output is an alias of an input. Note that an alias input + // may itself be an alias. The algorithm will therefore need to follow + // those pointers. + int64 alias_input_port = 2; + TensorShapeProto shape = 3; + DataType dtype = 4; + } + repeated OutputInfo output_info = 5; + + // Temporary memory used by this node. + int64 temporary_memory_size = 6; + + // Persistent memory used by this node. + int64 persistent_memory_size = 12; + + int64 host_temp_memory_size = 10 [deprecated = true]; + int64 device_temp_memory_size = 11 [deprecated = true]; + int64 device_persistent_memory_size = 16 [deprecated = true]; + + // Estimate of the computational cost of this node, in microseconds. + int64 compute_cost = 9; + + // Analytical estimate of the computational cost of this node, in + // microseconds. + int64 compute_time = 14; + + // Analytical estimate of the memory access cost of this node, in + // microseconds. + int64 memory_time = 15; + + // If true, the output is permanent: it can't be discarded, because this + // node is part of the "final output". Nodes may depend on final nodes. + bool is_final = 7; + + // Ids of the control inputs for this node. + repeated int32 control_input = 8; + + // Are the costs inaccurate? + bool inaccurate = 17; + } + repeated Node node = 1; + + // Total cost of this graph, typically used for balancing decisions. + message AggregatedCost { + // Aggregated cost value. + float cost = 1; + + // Aggregated cost dimension (e.g. 'memory', 'compute', 'network'). + string dimension = 2; + } + repeated AggregatedCost cost = 2; +} diff --git a/navi/navi/proto/tensorflow/core/framework/dataset_metadata.proto b/navi/navi/proto/tensorflow/core/framework/dataset_metadata.proto new file mode 100644 index 0000000000..0e667dd48d --- /dev/null +++ b/navi/navi/proto/tensorflow/core/framework/dataset_metadata.proto @@ -0,0 +1,10 @@ +syntax = "proto3"; + +package tensorflow.data; + +option go_package = "github.com/tensorflow/tensorflow/tensorflow/go/core/framework/dataset_metadata_go_proto"; + +// next: 2 +message Metadata { + bytes name = 1; +} diff --git a/navi/navi/proto/tensorflow/core/framework/dataset_options.proto b/navi/navi/proto/tensorflow/core/framework/dataset_options.proto new file mode 100644 index 0000000000..3919d51c16 --- /dev/null +++ b/navi/navi/proto/tensorflow/core/framework/dataset_options.proto @@ -0,0 +1,196 @@ +syntax = "proto3"; + +package tensorflow.data; + +import "tensorflow/core/framework/model.proto"; + +option go_package = "github.com/tensorflow/tensorflow/tensorflow/go/core/framework/dataset_options_go_proto"; + +// Represents the type of auto-sharding we enable. +enum AutoShardPolicy { + // AUTO: Attempts FILE-based sharding, falling back to DATA-based sharding. + AUTO = 0; + // FILE: Shards by input files (i.e. each worker will get a set of files to + // process). When this option is selected, make sure that there is at least as + // many files as workers. If there are fewer input files than workers, a + // runtime error will be raised. + FILE = 1; + // DATA: Shards by elements produced by the dataset. Each worker will process + // the whole dataset and discard the portion that is not for itself. Note that + // for this mode to correctly partitions the dataset elements, the dataset + // needs to produce elements in a deterministic order. + DATA = 2; + // HINT: Looks for the presence of `shard(SHARD_HINT, ...)` which is treated + // as a placeholder to replace with `shard(num_workers, worker_index)`. + HINT = 3; + // OFF: No sharding will be performed. + OFF = -1; +} + +// next: 5 +message AutotuneOptions { + // Whether to automatically tune performance knobs. + oneof optional_enabled { + bool enabled = 1; + } + // When autotuning is enabled (through autotune), determines the CPU budget to + // use. Values greater than the number of schedulable CPU cores are allowed + // but may result in CPU contention. + oneof optional_cpu_budget { + int32 cpu_budget = 2; + } + // When autotuning is enabled (through autotune), determines the RAM budget to + // use. Values greater than the available RAM in bytes may result in OOM. If + // 0, defaults to half of the available RAM in bytes. + oneof optional_ram_budget { + int64 ram_budget = 3; + } + + // When autotuning is enabled (through autotune), determines the algorithm to + // use. If not explicitly set by user, autotuning will follow HILL_CLIMB + // algorithm but has more flexibility to tune parameters more aggressively, + // in which case the behavior is implementation specific and may change over + // time. + oneof optional_autotune_algorithm { + model.AutotuneAlgorithm autotune_algorithm = 4; + } +} + +// next: 2 +message CardinalityOptions { + enum ComputeLevel { + CARDINALITY_COMPUTE_UNSPECIFIED = 0; + // Cardinality will only be computed if it can be determined in a cheap + // manner (ie. without reading from file sources). If the cardinality would + // be nontrivial to compute, Cardinality() will return UNKNOWN_CARDINALITY. + CARDINALITY_COMPUTE_LOW = 1; + // Moderate effort will be made to determine cardinality, such as reading + // index data from source files. If significant work is needed to compute + // cardinality (e.g. reading entire source file contents or executing user + // defined functions), Cardinality() will return UNKNOWN_CARDINALITY. + CARDINALITY_COMPUTE_MODERATE = 2; + } + ComputeLevel compute_level = 1; +} + +// next: 3 +message DistributeOptions { + AutoShardPolicy auto_shard_policy = 1; + // The number of devices attached to this input pipeline. + oneof optional_num_devices { + int32 num_devices = 2; + } +} + +// next: 18 +message OptimizationOptions { + // Whether to apply default graph optimizations. If False, only graph + // optimizations that have been explicitly enabled will be applied. + oneof optional_apply_default_optimizations { + bool apply_default_optimizations = 1; + } + reserved 2; + reserved 3; + reserved 4; + reserved 5; + // Whether to fuse filter transformations. + oneof optional_filter_fusion { + bool filter_fusion = 6; + } + // NOTE: field id 7 deleted in June 2021. + reserved 7; + // NOTE: field id 8 deleted in June 2021. + reserved 8; + // Whether to fuse map and batch transformations. + oneof optional_map_and_batch_fusion { + bool map_and_batch_fusion = 9; + } + // Whether to fuse map and filter transformations. + oneof optional_map_and_filter_fusion { + bool map_and_filter_fusion = 10; + } + // Whether to fuse map transformations. + oneof optional_map_fusion { + bool map_fusion = 11; + } + // Whether to parallelize stateless map transformations. + oneof optional_map_parallelization { + bool map_parallelization = 12; + } + + // NOTE: field id 13 deleted in June 2021. + reserved 13; + + // Whether to eliminate no-op transformations. + oneof optional_noop_elimination { + bool noop_elimination = 14; + } + // Whether to parallelize copying of batch elements. This optimization is + // highly experimental and can cause performance degradation (e.g. when the + // parallelization overhead exceeds the benefits of performing the data copies + // in parallel). You should only enable this optimization if a) your input + // pipeline is bottlenecked on batching and b) you have validated that this + // optimization improves performance. + oneof optional_parallel_batch { + bool parallel_batch = 15; + } + // Field id 16 was removed in 06/2021. + reserved 16; + // Whether to fuse shuffle and repeat transformations. + oneof optional_shuffle_and_repeat_fusion { + bool shuffle_and_repeat_fusion = 17; + } +} + +// next: 3 +message ThreadingOptions { + // If set, it overrides the maximum degree of intra-op parallelism. + oneof optional_max_intra_op_parallelism { + int32 max_intra_op_parallelism = 1; + } + // If set, the dataset will use a private threadpool of the given size. + oneof optional_private_threadpool_size { + int32 private_threadpool_size = 2; + } +} + +// Represents how to handle external state during serialization. +enum ExternalStatePolicy { + POLICY_WARN = 0; + POLICY_IGNORE = 1; + POLICY_FAIL = 2; +} + +// Message stored with Dataset objects to control how datasets are processed and +// optimized. +// +// next: 8 +message Options { + // Whether the outputs need to be produced in deterministic order. + oneof optional_deterministic { + bool deterministic = 1; + } + // The distribution strategy options associated with the dataset. + AutotuneOptions autotune_options = 7; + // The distribution strategy options associated with the dataset. + DistributeOptions distribute_options = 2; + // The optimization options associated with the dataset. + OptimizationOptions optimization_options = 3; + // Whether to introduce 'slack' in the last `prefetch` of the input pipeline, + // if it exists. This may reduce CPU contention with accelerator host-side + // activity at the start of a step. The slack frequency is determined by the + // number of devices attached to this input pipeline. + oneof optional_slack { + bool slack = 4; + } + // The threading options associated with the dataset. + ThreadingOptions threading_options = 5; + // This option can be used to override the default policy for how to handle + // external state when serializing a dataset or checkpointing its iterator. + // There are three settings available - IGNORE: External state is ignored + // without a warning; WARN: External state is ignored and a warning is logged; + // FAIL: External state results in an error. + oneof optional_external_state_policy { + ExternalStatePolicy external_state_policy = 6; + } +} diff --git a/navi/navi/proto/tensorflow/core/framework/device_attributes.proto b/navi/navi/proto/tensorflow/core/framework/device_attributes.proto new file mode 100644 index 0000000000..5f568e255f --- /dev/null +++ b/navi/navi/proto/tensorflow/core/framework/device_attributes.proto @@ -0,0 +1,58 @@ +syntax = "proto3"; + +package tensorflow; + +option cc_enable_arenas = true; +option java_outer_classname = "DeviceAttributesProtos"; +option java_multiple_files = true; +option java_package = "org.tensorflow.framework"; +option go_package = "github.com/tensorflow/tensorflow/tensorflow/go/core/framework/device_attributes_go_proto"; + +message InterconnectLink { + int32 device_id = 1; + string type = 2; + int32 strength = 3; +} + +message LocalLinks { + repeated InterconnectLink link = 1; +} + +message DeviceLocality { + // Optional bus locality of device. Default value of 0 means + // no specific locality. Specific localities are indexed from 1. + int32 bus_id = 1; + + // Optional NUMA locality of device. + int32 numa_node = 2; + + // Optional local interconnect links to other devices. + LocalLinks links = 3; +} + +message DeviceAttributes { + // Fully specified name of the device within a cluster. + string name = 1; + + // String representation of device_type. + string device_type = 2; + + // Memory capacity of device in bytes. + int64 memory_limit = 4; + + // Platform-specific data about device that may be useful + // for supporting efficient data transfers. + DeviceLocality locality = 5; + + // A device is assigned a global unique number each time it is + // initialized. "incarnation" should never be 0. + fixed64 incarnation = 6; + + // String representation of the physical device that this device maps to. + string physical_device_desc = 7; + + // A physical device ID for use in XLA DeviceAssignments, unique across + // clients in a multi-client setup. Set to -1 if unavailable, non-negative + // otherwise. + int64 xla_global_id = 8; +} diff --git a/navi/navi/proto/tensorflow/core/framework/full_type.proto b/navi/navi/proto/tensorflow/core/framework/full_type.proto new file mode 100644 index 0000000000..e8175ed3d1 --- /dev/null +++ b/navi/navi/proto/tensorflow/core/framework/full_type.proto @@ -0,0 +1,276 @@ +syntax = "proto3"; + +package tensorflow; + +option cc_enable_arenas = true; +option java_outer_classname = "FullTypeProtos"; +option java_multiple_files = true; +option java_package = "org.tensorflow.framework"; +option go_package = "github.com/tensorflow/tensorflow/tensorflow/go/core/framework/full_type_go_proto"; + +// Experimental. Represents the complete type information of a TensorFlow value. +enum FullTypeId { + // The default represents an uninitialized values. + TFT_UNSET = 0; + + // Type symbols. Used to construct more complex type expressions like + // algebraic data types. + + // Type variables may serve as placeholder for any other type ID in type + // templates. + // + // Examples: + // TFT_DATASET[TFT_VAR["T"]] is a Dataset returning a type indicated by "T". + // TFT_TENSOR[TFT_VAR["T"]] is a Tensor of n element type indicated by "T". + // TFT_TENSOR[TFT_VAR["T"]], TFT_TENSOR[TFT_VAR["T"]] are two tensors of + // identical element types. + // TFT_TENSOR[TFT_VAR["P"]], TFT_TENSOR[TFT_VAR["Q"]] are two tensors of + // independent element types. + // + TFT_VAR = 1; + + // Wildcard type. Describes a parameter of unknown type. In TensorFlow, that + // can mean either a "Top" type (accepts any type), or a dynamically typed + // object whose type is unknown in context. + // Important: "unknown" does not necessarily mean undeterminable! + TFT_ANY = 2; + + // The algebraic product type. This is an algebraic type that may be used just + // for logical grouping. Not to confused with TFT_TUPLE which describes a + // concrete object of several elements. + // + // Example: + // TFT_DATASET[TFT_PRODUCT[TFT_TENSOR[TFT_INT32], TFT_TENSOR[TFT_FLOAT64]]] + // is a Dataset producing two tensors, an integer one and a float one. + // + TFT_PRODUCT = 3; + + // Represents a named field, with the name stored in the attribute. + // + // Parametrization: + // TFT_NAMED[]{} + // * is the type of the field + // * is the field name, as string (thpugh can theoretically be an int + // as well) + // + // Example: + // TFT_RECORD[ + // TFT_NAMED[TFT_TENSOR[TFT_INT32]]{'foo'}, + // TFT_NAMED[TFT_TENSOR[TFT_FLOAT32]]{'bar'}, + // ] + // is a structure with two fields, an int tensor "foo" and a float tensor + // "bar". + TFT_NAMED = 4; + + // Template definition. Expands the variables by repeating a template as + // arguments of container. + // + // Parametrization: + // TFT_FOR_EACH[,