001/* 002 * Copyright (c) 2012-2021 Institut National des Sciences Appliquées de Lyon (INSA Lyon) and others 003 * 004 * This program and the accompanying materials are made available under the 005 * terms of the Eclipse Public License 2.0 which is available at 006 * http://www.eclipse.org/legal/epl-2.0. 007 * 008 * SPDX-License-Identifier: EPL-2.0 009 */ 010 011package org.eclipse.golo.runtime.augmentation; 012 013import org.eclipse.golo.runtime.Loader; 014import org.eclipse.golo.runtime.Module; 015 016import java.util.function.Predicate; 017import java.util.stream.Stream; 018 019import static org.eclipse.golo.runtime.augmentation.AugmentationApplication.Kind; 020 021/** 022 * Encapsulate a module defining an augmentation. 023 */ 024public final class DefiningModule { 025 026 public enum Scope { LOCAL, IMPORT, CALLSTACK } 027 028 private final Class<?> module; 029 private final Scope scope; 030 031 DefiningModule(Class<?> module, Scope scope) { 032 this.module = module; 033 this.scope = scope; 034 } 035 036 public Class<?> module() { 037 return this.module; 038 } 039 040 public static DefiningModule of(Class<?> module, Scope scope) { 041 return new DefiningModule(module, scope); 042 } 043 044 public static DefiningModule ofLocal(Class<?> module) { 045 return new DefiningModule(module, Scope.LOCAL); 046 } 047 048 public static DefiningModule ofImport(Class<?> module) { 049 return new DefiningModule(module, Scope.IMPORT); 050 } 051 052 public static DefiningModule ofCallstack(Class<?> module) { 053 return new DefiningModule(module, Scope.CALLSTACK); 054 } 055 056 /** 057 * Returns a stream of augmentations definitions defined directly on a type in the corresponding 058 * module. 059 * e.g. 060 * <pre class="listing"><code class="lang-golo" data-lang="golo"> 061 * augment module.Type { 062 * # ... 063 * } 064 * </code></pre> 065 */ 066 private Stream<AugmentationApplication> simpleAugmentationsFor(Loader loader, Class<?> receiverType) { 067 return Stream.of(Module.augmentations(module)) 068 .map(loader) 069 .filter(isAssignableFrom(receiverType)) 070 .map(target -> new AugmentationApplication( 071 loader.load(module.getName() + "$" + target.getName().replace(".", "$")), 072 target, scope, Kind.SIMPLE)); 073 } 074 075 private static Predicate<Class<?>> isAssignableFrom(Class<?> receiver) { 076 return target -> target != null && target.isAssignableFrom(receiver); 077 } 078 079 private Stream<AugmentationApplication> fullyNamedAugmentationsFor(Loader loader, Class<?> receiverType) { 080 return Stream.of(Module.augmentationApplications(module)) 081 .map(loader::load) 082 .filter(target -> target != null && target.isAssignableFrom(receiverType)) 083 .flatMap(target -> qualifyAugmentations(loader, target)); 084 } 085 086 private Stream<AugmentationApplication> qualifyAugmentations(Loader loader, Class<?> target) { 087 return Stream.of(Module.augmentationApplications(module, target)) 088 .flatMap(this::fullyQualifiedName) 089 .map(augmentName -> new AugmentationApplication( 090 loader.load(augmentName), 091 target, scope, Kind.NAMED)); 092 } 093 094 /** 095 * Fully qualify an augmentation name. 096 * <p> 097 * Given an augmentation name, this generate a stream of alternative names by prepending 098 * names of the defining module as well as imported modules. 099 */ 100 private Stream<String> fullyQualifiedName(String augmentationName) { 101 Stream.Builder<String> names = Stream.builder(); 102 int idx = augmentationName.lastIndexOf("."); 103 if (idx == -1) { 104 names.add(augmentationName); 105 } else { 106 names.add(new StringBuilder(augmentationName).replace(idx, idx + 1, "$").toString()); 107 } 108 names.add(module.getName() + "$" + augmentationName.replace(".", "$")); 109 return Stream.concat( 110 names.build(), 111 Stream.of(Module.imports(module)) 112 .map(prefix -> prefix + "$" + augmentationName.replace(".", "$"))); 113 } 114 115 public Stream<AugmentationApplication> augmentationsFor(Loader loader, Class<?> receiverType) { 116 if (module == null) { 117 return Stream.empty(); 118 } 119 return Stream.concat( 120 simpleAugmentationsFor(loader, receiverType), 121 fullyNamedAugmentationsFor(loader, receiverType)); 122 123 } 124 125 @Override 126 public String toString() { 127 return "DefiningModule<" + module + "," + scope + ">"; 128 } 129} 130 131