001/* 002 * Copyright (C) 2022 - 2025, the original author or authors. 003 * 004 * Licensed under the Apache License, Version 2.0 (the "License"); 005 * you may not use this file except in compliance with the License. 006 * You may obtain a copy of the License at 007 * 008 * http://www.apache.org/licenses/LICENSE-2.0 009 * 010 * Unless required by applicable law or agreed to in writing, software 011 * distributed under the License is distributed on an "AS IS" BASIS, 012 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 013 * See the License for the specific language governing permissions and 014 * limitations under the License. 015 */ 016package io.github.ascopes.jct.filemanagers; 017 018import io.github.ascopes.jct.ex.JctIllegalInputException; 019import io.github.ascopes.jct.utils.ToStringBuilder; 020import java.util.Objects; 021import java.util.Optional; 022import javax.tools.JavaFileManager.Location; 023import org.jspecify.annotations.Nullable; 024 025/** 026 * Handle that represents the location of a module. 027 * 028 * @author Ashley Scopes 029 * @since 0.0.1 030 */ 031public final class ModuleLocation implements Location { 032 033 private final Location parent; 034 private final String moduleName; 035 private final String name; 036 037 /** 038 * Initialize the location. 039 * 040 * @param parent the parent location. 041 * @param moduleName the module name. 042 * @throws JctIllegalInputException if the parent location is not an output location or a 043 * module-oriented location. 044 */ 045 public ModuleLocation(Location parent, String moduleName) { 046 Objects.requireNonNull(parent, "parent"); 047 Objects.requireNonNull(moduleName, "moduleName"); 048 049 if (!parent.isOutputLocation() && !parent.isModuleOrientedLocation()) { 050 throw new JctIllegalInputException( 051 "The parent of a module location must be either an output location " 052 + "or be module-oriented, but got " 053 + parent.getName() 054 ); 055 } 056 057 this.parent = parent; 058 this.moduleName = moduleName; 059 name = parent.getName() + "[" + moduleName + "]"; 060 } 061 062 /** 063 * Get the parent location. 064 * 065 * @return the parent location. 066 */ 067 public Location getParent() { 068 return parent; 069 } 070 071 /** 072 * Get the module name. 073 * 074 * @return the module name. 075 */ 076 public String getModuleName() { 077 return moduleName; 078 } 079 080 @Override 081 public String getName() { 082 return name; 083 } 084 085 @Override 086 public boolean isOutputLocation() { 087 return parent.isOutputLocation(); 088 } 089 090 @Override 091 public boolean isModuleOrientedLocation() { 092 // Module locations cannot be module-oriented, but their parents can be. 093 return false; 094 } 095 096 @Override 097 public boolean equals(@Nullable Object other) { 098 if (!(other instanceof ModuleLocation)) { 099 return false; 100 } 101 102 var that = (ModuleLocation) other; 103 104 return parent.equals(that.parent) 105 && moduleName.equals(that.moduleName); 106 } 107 108 @Override 109 public int hashCode() { 110 return Objects.hash(parent, moduleName); 111 } 112 113 @Override 114 public String toString() { 115 return new ToStringBuilder(this) 116 .attribute("parent", parent) 117 .attribute("moduleName", moduleName) 118 .toString(); 119 } 120 121 /** 122 * Attempt to upcast a given location to a ModuleLocation if it is an instance of 123 * ModuleLocation. 124 * 125 * @param location the location to attempt to upcast. 126 * @return an optional containing the upcast location if it is a module location, 127 * or an empty optional if not. 128 * @since 1.1.5 129 */ 130 public static Optional<ModuleLocation> upcast(Location location) { 131 return Optional.of(location) 132 .filter(ModuleLocation.class::isInstance) 133 .map(ModuleLocation.class::cast); 134 } 135}