All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.gmetrics.metric.AbstractMethodMetric.groovy Maven / Gradle / Ivy

Go to download

The GMetrics project provides calculation and reporting of size and complexity metrics for Groovy source code.

The newest version!
/*
* Copyright 2012 the original author or authors.
*
* 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.
*/
package org.gmetrics.metric

import org.codehaus.groovy.ast.ClassNode
import org.codehaus.groovy.ast.MethodNode
import org.codehaus.groovy.ast.expr.ClosureExpression
import org.gmetrics.source.SourceCode
import org.gmetrics.util.AstUtil
import org.gmetrics.result.ClassMetricResult
import org.gmetrics.result.MetricResult
import org.gmetrics.result.MethodKey

/**
 * Abstract superclass for method-based metrics.
 *
 * @author Chris Mair
 */
abstract class AbstractMethodMetric extends AbstractMetric implements MethodMetric {

    final MetricLevel baseLevel = MetricLevel.METHOD

    boolean includeClosureFields = true

    abstract MetricResult calculate(MethodNode methodNode, SourceCode sourceCode)
    abstract MetricResult calculate(ClosureExpression closureExpression, SourceCode sourceCode)

    @Override
    MetricResult applyToMethod(MethodNode methodNode, SourceCode sourceCode) {
        if (!enabled) {
            return null
        }
        return calculate(methodNode, sourceCode)
    }

    @Override
    MetricResult applyToClosure(ClosureExpression closureExpression, SourceCode sourceCode) {
        if (!enabled) {
            return null
        }
        return calculate(closureExpression, sourceCode)
    }

    protected ClassMetricResult calculateForClass(ClassNode classNode, SourceCode sourceCode) {
        Map childMetricResults = [:]

        if (isNotAnInterface(classNode)) {
            addMethodsToMetricResults(sourceCode, classNode, childMetricResults)
            addClosureFieldsToMetricResults(sourceCode, classNode, childMetricResults)
        }

        if (childMetricResults.isEmpty()) {
            return null
        }

        def aggregateMetricResults = createAggregateMetricResult(MetricLevel.CLASS, childMetricResults.values(), classNode)

        return new ClassMetricResult(aggregateMetricResults, childMetricResults)
    }

    protected int lineNumberForMethod(MethodNode methodNode, SourceCode sourceCode) {
        int lineNumber = methodNode.lineNumber
        if (methodNode.annotations) {
            int firstLine = methodNode.lineNumber
            int lastLine = methodNode.code.lineNumber
            def matchingLine = (firstLine..lastLine).find { int line ->
                String lineString = sourceCode.line(line - 1)
                lineString.contains(methodNode.name)
            }
            lineNumber = matchingLine != null ? matchingLine : lineNumber
        }
        return lineNumber
    }

    private void addClosureFieldsToMetricResults(SourceCode sourceCode, ClassNode classNode, Map childMetricResults) {
        if (!includeClosureFields) {
            return
        }

        def closureFields = classNode.fields.findAll {fieldNode -> AstUtil.isClosureField(fieldNode) }
        closureFields.each {fieldNode ->
            def fieldResult = calculate(fieldNode.initialExpression, sourceCode)
            if (fieldResult) {
                def methodKey = new MethodKey(fieldNode.name)
                childMetricResults[methodKey] = fieldResult
            }
        }
    }

    private void addMethodsToMetricResults(SourceCode sourceCode, ClassNode classNode, Map childMetricResults) {
        def methodsPlusConstructors = classNode.getMethods() + classNode.getDeclaredConstructors()

        methodsPlusConstructors.each {methodNode ->
            def methodResult = calculate(methodNode, sourceCode)
            if (methodResult) {
                def methodKey = new MethodKey(methodNode)
                childMetricResults[methodKey] = methodResult
            }
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy