This repository has been archived by the owner on Jan 28, 2022. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 1
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Basic annotations and code injection working but had to use the DSL d…
…ue to issues with importing spring classes in the generated code.
- Loading branch information
0 parents
commit aeaea3a
Showing
22 changed files
with
802 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,28 @@ | ||
subprojects { | ||
version = '0.0.1' | ||
group = 'com.stehno.effigy' | ||
|
||
apply plugin:'groovy' | ||
apply plugin:'maven-publish' | ||
|
||
compileJava { | ||
sourceCompatibility = 1.8 | ||
targetCompatibility = 1.8 | ||
} | ||
|
||
compileGroovy { | ||
groovyOptions.optimizationOptions.indy = false | ||
} | ||
|
||
repositories { | ||
jcenter() | ||
} | ||
|
||
dependencies { | ||
compile 'org.codehaus.groovy:groovy-all:2.3.7' | ||
} | ||
} | ||
|
||
task wrapper(type: Wrapper) { | ||
gradleVersion = '2.2' | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,5 @@ | ||
dependencies { | ||
compile 'org.springframework:spring-jdbc:4.1.2.RELEASE' | ||
|
||
//compile 'org.slf4j:slf4j-log4j12:1.7.7' | ||
} |
21 changes: 21 additions & 0 deletions
21
effigy-core/src/main/groovy/com/stehno/effigy/annotation/Column.groovy
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,21 @@ | ||
package com.stehno.effigy.annotation | ||
|
||
import java.lang.annotation.Documented | ||
import java.lang.annotation.ElementType | ||
import java.lang.annotation.Retention | ||
import java.lang.annotation.RetentionPolicy | ||
import java.lang.annotation.Target | ||
|
||
/** | ||
* Created by cjstehno on 11/26/2014. | ||
*/ | ||
@Target(ElementType.FIELD) | ||
@Retention(RetentionPolicy.SOURCE) | ||
@Documented | ||
@interface Column { | ||
|
||
/** | ||
* The name of the column in the database. | ||
*/ | ||
String value() | ||
} |
20 changes: 20 additions & 0 deletions
20
effigy-core/src/main/groovy/com/stehno/effigy/annotation/Effigy.groovy
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,20 @@ | ||
package com.stehno.effigy.annotation | ||
|
||
import com.stehno.effigy.transform.EffigyTransformer | ||
import org.codehaus.groovy.transform.GroovyASTTransformationClass | ||
|
||
import java.lang.annotation.* | ||
/** | ||
* Created by cjstehno on 11/26/2014. | ||
*/ | ||
@Target(ElementType.TYPE) | ||
@Retention(RetentionPolicy.SOURCE) | ||
@Documented | ||
@GroovyASTTransformationClass(classes=[EffigyTransformer]) | ||
@interface Effigy { | ||
|
||
/** | ||
* The name of the database table represented by the entity. The default will be to use the pluralized name of the entity. | ||
*/ | ||
String table() default '' | ||
} |
25 changes: 25 additions & 0 deletions
25
effigy-core/src/main/groovy/com/stehno/effigy/annotation/EffigyRepository.groovy
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,25 @@ | ||
package com.stehno.effigy.annotation | ||
|
||
import com.stehno.effigy.transform.EffigyRepositoryTransformer | ||
import org.codehaus.groovy.transform.GroovyASTTransformationClass | ||
|
||
import java.lang.annotation.Documented | ||
import java.lang.annotation.ElementType | ||
import java.lang.annotation.Retention | ||
import java.lang.annotation.RetentionPolicy | ||
import java.lang.annotation.Target | ||
|
||
/** | ||
* Created by cjstehno on 11/26/2014. | ||
*/ | ||
@Target(ElementType.TYPE) | ||
@Retention(RetentionPolicy.SOURCE) | ||
@Documented | ||
@GroovyASTTransformationClass(classes=[EffigyRepositoryTransformer]) | ||
@interface EffigyRepository { | ||
|
||
/** | ||
* The entity type handled by the repository (must be annotated with @Effigy) | ||
*/ | ||
Class forEntity() | ||
} |
8 changes: 8 additions & 0 deletions
8
effigy-core/src/main/groovy/com/stehno/effigy/annotation/Id.groovy
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,8 @@ | ||
package com.stehno.effigy.annotation | ||
/** | ||
* Created by cjstehno on 11/26/2014. | ||
*/ | ||
|
||
@interface Id { | ||
|
||
} |
28 changes: 28 additions & 0 deletions
28
effigy-core/src/main/groovy/com/stehno/effigy/transform/AnnotationUtils.groovy
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,28 @@ | ||
package com.stehno.effigy.transform | ||
|
||
import org.codehaus.groovy.ast.AnnotationNode | ||
import org.codehaus.groovy.ast.ClassNode | ||
import org.codehaus.groovy.ast.expr.ClassExpression | ||
|
||
/** | ||
* Created by cjstehno on 11/26/2014. | ||
*/ | ||
class AnnotationUtils { | ||
|
||
public static ClassNode extractClass(AnnotationNode annotation, String key) { | ||
def pair = annotation.members.find { pair -> pair.key == key; }; | ||
if (pair == null) return null; | ||
|
||
assert (pair.value instanceof ClassExpression) | ||
return pair.value.type; | ||
} | ||
|
||
public static String extractString(AnnotationNode annotation, String key) { | ||
def pair = annotation.members.find { pair -> pair.key == key; }; | ||
if (pair) { | ||
return pair.value.value; | ||
} else { | ||
return null; | ||
} | ||
} | ||
} |
241 changes: 241 additions & 0 deletions
241
effigy-core/src/main/groovy/com/stehno/effigy/transform/EffigyRepositoryTransformer.groovy
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,241 @@ | ||
package com.stehno.effigy.transform | ||
|
||
import static com.stehno.effigy.transform.AnnotationUtils.extractClass | ||
import static com.stehno.effigy.transform.AnnotationUtils.extractString | ||
|
||
import com.stehno.effigy.annotation.Column | ||
import com.stehno.effigy.annotation.Effigy | ||
import com.stehno.effigy.annotation.Id | ||
import groovy.text.Template | ||
import groovy.transform.ToString | ||
import org.codehaus.groovy.ast.* | ||
import org.codehaus.groovy.ast.builder.AstBuilder | ||
import org.codehaus.groovy.ast.expr.EmptyExpression | ||
import org.codehaus.groovy.ast.stmt.Statement | ||
import org.codehaus.groovy.control.CompilePhase | ||
import org.codehaus.groovy.control.SourceUnit | ||
import org.codehaus.groovy.transform.ASTTransformation | ||
import org.codehaus.groovy.transform.GroovyASTTransformation | ||
import org.springframework.beans.factory.annotation.Autowired | ||
import org.springframework.jdbc.core.JdbcTemplate | ||
import org.springframework.jdbc.core.PreparedStatementCreatorFactory | ||
import org.springframework.jdbc.support.GeneratedKeyHolder | ||
|
||
import java.lang.reflect.Modifier | ||
import java.sql.Types | ||
|
||
/** | ||
* Created by cjstehno on 11/26/2014. | ||
*/ | ||
@GroovyASTTransformation(phase = CompilePhase.CANONICALIZATION) | ||
class EffigyRepositoryTransformer implements ASTTransformation { | ||
|
||
@Override | ||
void visit(ASTNode[] nodes, SourceUnit source) { | ||
println "EffigyRepository: ${nodes[0]}" | ||
|
||
source.getAST().addImport(null, ClassHelper.make(GeneratedKeyHolder)) | ||
|
||
injectJdbcTemplate(nodes[1]) | ||
|
||
ClassNode entityClassNode = extractClass(nodes[0], 'forEntity') | ||
AnnotationNode effigyAnnotNode = entityClassNode.getAnnotations(new ClassNode(Effigy))[0] | ||
|
||
println "Entity has Effigy annotation: ${effigyAnnotNode != null}" | ||
|
||
String tableName = extractString(effigyAnnotNode, 'table') | ||
if (!tableName) { | ||
tableName = entityClassNode.nameWithoutPackage.toLowerCase() + 's' | ||
} | ||
|
||
// list fields | ||
EntityInfo entityInfo = new EntityInfo( | ||
table: tableName, | ||
type: entityClassNode | ||
) | ||
|
||
entityClassNode.fields.each { FieldNode field -> | ||
String fieldName | ||
|
||
AnnotationNode fieldColumnAnnot = field.getAnnotations(new ClassNode(Column))[0] | ||
if (fieldColumnAnnot) { | ||
fieldName = extractString(fieldColumnAnnot, 'value') | ||
|
||
} else { | ||
fieldName = StringUtils.camelCaseToUnderscore(field.name) | ||
} | ||
|
||
def idAnnot = field.getAnnotations(new ClassNode(Id))[0] | ||
|
||
entityInfo.props << new EntityPropertyInfo( | ||
id: idAnnot != null, | ||
fieldName: fieldName, | ||
propertyName: field.name, | ||
type: field.type | ||
) | ||
} | ||
|
||
println entityInfo | ||
|
||
injectSaveMethod(nodes[1] as ClassNode, entityInfo) | ||
} | ||
|
||
private static void injectSaveMethod(final ClassNode repositoryClassNode, final EntityInfo entityInfo) { | ||
def nodes = new AstBuilder().buildFromSpec { | ||
block { | ||
expression { | ||
declaration { | ||
variable('keys') | ||
token('=') | ||
constructorCall(GeneratedKeyHolder) { | ||
argumentList {} | ||
} | ||
} | ||
} | ||
|
||
expression { | ||
declaration { | ||
variable('factory') | ||
token('=') | ||
constructorCall(PreparedStatementCreatorFactory) { | ||
argumentList { | ||
constant("insert into ${entityInfo.table} (${entityInfo.fieldNamesString(false)}) values (${entityInfo.placeholderString(false)})" as String) | ||
array(int.class) { | ||
entityInfo.types(false).each { typ -> | ||
constant(typ) | ||
} | ||
} | ||
} | ||
} | ||
} | ||
} | ||
|
||
expression { | ||
declaration { | ||
variable('paramValues') | ||
token('=') | ||
array(Object) { | ||
entityInfo.propertyInfo(false).each { pi -> | ||
property { | ||
variable('entity') | ||
constant(pi.propertyName) | ||
} | ||
} | ||
} | ||
} | ||
} | ||
|
||
expression { | ||
declaration { | ||
variable('psc') | ||
token('=') | ||
methodCall { | ||
variable 'factory' | ||
constant 'newPreparedStatementCreator' | ||
argumentList { | ||
variable('paramValues') | ||
} | ||
} | ||
} | ||
} | ||
|
||
expression { | ||
methodCall { | ||
variable('jdbcTemplate') | ||
constant('update') | ||
argumentList { | ||
variable('psc') | ||
variable('keys') | ||
} | ||
} | ||
} | ||
|
||
returnStatement { | ||
property { | ||
variable('keys') | ||
constant('key') | ||
} | ||
} | ||
} | ||
} | ||
|
||
repositoryClassNode.addMethod(new MethodNode( | ||
'save', | ||
Modifier.PUBLIC, | ||
entityInfo.idType, | ||
[new Parameter(entityInfo.type, 'entity')] as Parameter[], | ||
null, | ||
nodes[0] as Statement | ||
)) | ||
} | ||
|
||
private void injectJdbcTemplate(ClassNode repositoryClassNode) { | ||
FieldNode jdbcTemplateFieldNode = new FieldNode( | ||
'jdbcTemplate', Modifier.PRIVATE, new ClassNode(JdbcTemplate), repositoryClassNode, new EmptyExpression() | ||
) | ||
|
||
jdbcTemplateFieldNode.addAnnotation(new AnnotationNode(new ClassNode(Autowired))) | ||
|
||
repositoryClassNode.addField(jdbcTemplateFieldNode) | ||
} | ||
|
||
private static List<ASTNode> build(Template template, Map attrs) { | ||
def text = template.make(attrs).toString() | ||
println text | ||
|
||
try { | ||
new AstBuilder().buildFromString(CompilePhase.CANONICALIZATION, true, text) | ||
} catch (ex) { | ||
ex.printStackTrace() | ||
return null | ||
} | ||
} | ||
} | ||
|
||
@ToString(includeNames = true) | ||
class EntityInfo { | ||
ClassNode type | ||
String table | ||
List<EntityPropertyInfo> props = [] | ||
|
||
ClassNode getIdType() { | ||
props.find { it.id }.type | ||
} | ||
|
||
String fieldNamesString(boolean includeId) { | ||
propertyInfo(includeId).collect { it.fieldName }.join(',') | ||
} | ||
|
||
String placeholderString(boolean includeId) { | ||
propertyInfo(includeId).collect { '?' }.join(',') | ||
} | ||
|
||
List<EntityPropertyInfo> propertyInfo(boolean includeId) { | ||
(includeId ? props : props.findAll { !it.id }) | ||
} | ||
|
||
List<Integer> types(boolean includeId) { | ||
propertyInfo(includeId).collect { | ||
switch (it.type.nameWithoutPackage) { | ||
case 'String': return Types.VARCHAR | ||
case 'Date': return Types.TIMESTAMP | ||
case 'Boolean': | ||
case 'boolean': | ||
return Types.BOOLEAN | ||
case 'Integer': return Types.INTEGER | ||
case 'Long': | ||
case 'long': | ||
return Types.BIGINT | ||
default: return Types.JAVA_OBJECT | ||
} | ||
} | ||
} | ||
} | ||
|
||
@ToString(includeNames = true) | ||
class EntityPropertyInfo { | ||
boolean id | ||
String fieldName | ||
String propertyName | ||
ClassNode type | ||
} |
Oops, something went wrong.