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

org.apache.camel.parser.helper.CamelJavaRestDslParserHelper Maven / Gradle / Ivy

There is a newer version: 4.9.0
Show newest version
/**
 * Licensed to the Apache Software Foundation (ASF) under one or more
 * contributor license agreements.  See the NOTICE file distributed with
 * this work for additional information regarding copyright ownership.
 * The ASF licenses this file to You 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.apache.camel.parser.helper;

import java.io.BufferedReader;
import java.io.File;
import java.io.FileReader;
import java.util.ArrayList;
import java.util.Collections;
import java.util.List;

import org.apache.camel.parser.model.RestConfigurationDetails;
import org.apache.camel.parser.model.RestServiceDetails;
import org.apache.camel.parser.model.RestVerbDetails;
import org.apache.camel.parser.roaster.StatementFieldSource;
import org.jboss.forge.roaster._shade.org.eclipse.jdt.core.dom.ASTNode;
import org.jboss.forge.roaster._shade.org.eclipse.jdt.core.dom.AnonymousClassDeclaration;
import org.jboss.forge.roaster._shade.org.eclipse.jdt.core.dom.Block;
import org.jboss.forge.roaster._shade.org.eclipse.jdt.core.dom.BooleanLiteral;
import org.jboss.forge.roaster._shade.org.eclipse.jdt.core.dom.Expression;
import org.jboss.forge.roaster._shade.org.eclipse.jdt.core.dom.ExpressionStatement;
import org.jboss.forge.roaster._shade.org.eclipse.jdt.core.dom.FieldDeclaration;
import org.jboss.forge.roaster._shade.org.eclipse.jdt.core.dom.InfixExpression;
import org.jboss.forge.roaster._shade.org.eclipse.jdt.core.dom.MemberValuePair;
import org.jboss.forge.roaster._shade.org.eclipse.jdt.core.dom.MethodDeclaration;
import org.jboss.forge.roaster._shade.org.eclipse.jdt.core.dom.MethodInvocation;
import org.jboss.forge.roaster._shade.org.eclipse.jdt.core.dom.NormalAnnotation;
import org.jboss.forge.roaster._shade.org.eclipse.jdt.core.dom.NumberLiteral;
import org.jboss.forge.roaster._shade.org.eclipse.jdt.core.dom.ParenthesizedExpression;
import org.jboss.forge.roaster._shade.org.eclipse.jdt.core.dom.QualifiedName;
import org.jboss.forge.roaster._shade.org.eclipse.jdt.core.dom.SimpleName;
import org.jboss.forge.roaster._shade.org.eclipse.jdt.core.dom.SingleMemberAnnotation;
import org.jboss.forge.roaster._shade.org.eclipse.jdt.core.dom.StringLiteral;
import org.jboss.forge.roaster._shade.org.eclipse.jdt.core.dom.Type;
import org.jboss.forge.roaster._shade.org.eclipse.jdt.core.dom.VariableDeclarationFragment;
import org.jboss.forge.roaster._shade.org.eclipse.jdt.core.dom.VariableDeclarationStatement;
import org.jboss.forge.roaster.model.Annotation;
import org.jboss.forge.roaster.model.source.FieldSource;
import org.jboss.forge.roaster.model.source.JavaClassSource;
import org.jboss.forge.roaster.model.source.MethodSource;

/**
 * A Camel Java Rest DSL parser that only depends on the Roaster API.
 * 

* This implement is used for parsing the Camel routes and build a tree structure of the Rest DSL services. */ public final class CamelJavaRestDslParserHelper { public List parseRestConfiguration(JavaClassSource clazz, String baseDir, String fullyQualifiedFileName, MethodSource configureMethod) { List answer = new ArrayList<>(); if (configureMethod != null) { MethodDeclaration md = (MethodDeclaration) configureMethod.getInternal(); Block block = md.getBody(); if (block != null) { for (Object statement : md.getBody().statements()) { // must be a method call expression if (statement instanceof ExpressionStatement) { ExpressionStatement es = (ExpressionStatement) statement; Expression exp = es.getExpression(); boolean valid = isRestConfiguration(exp); if (valid) { RestConfigurationDetails node = new RestConfigurationDetails(); answer.add(node); // include source code details int pos = exp.getStartPosition(); int line = findLineNumber(fullyQualifiedFileName, pos); if (line > -1) { node.setLineNumber("" + line); } pos = exp.getStartPosition() + exp.getLength(); line = findLineNumber(fullyQualifiedFileName, pos); if (line > -1) { node.setLineNumberEnd("" + line); } node.setFileName(fullyQualifiedFileName); node.setClassName(clazz.getQualifiedName()); node.setMethodName(configureMethod.getName()); parseExpression(node, fullyQualifiedFileName, clazz, configureMethod, block, exp); } } } } } return answer; } public List parseRestService(JavaClassSource clazz, String baseDir, String fullyQualifiedFileName, MethodSource configureMethod) { List answer = new ArrayList<>(); if (configureMethod != null) { MethodDeclaration md = (MethodDeclaration) configureMethod.getInternal(); Block block = md.getBody(); if (block != null) { for (Object statement : md.getBody().statements()) { // must be a method call expression if (statement instanceof ExpressionStatement) { ExpressionStatement es = (ExpressionStatement) statement; Expression exp = es.getExpression(); boolean valid = isRest(exp); if (valid) { RestServiceDetails node = new RestServiceDetails(); answer.add(node); // include source code details int pos = exp.getStartPosition(); int line = findLineNumber(fullyQualifiedFileName, pos); if (line > -1) { node.setLineNumber("" + line); } pos = exp.getStartPosition() + exp.getLength(); line = findLineNumber(fullyQualifiedFileName, pos); if (line > -1) { node.setLineNumberEnd("" + line); } node.setFileName(fullyQualifiedFileName); node.setClassName(clazz.getQualifiedName()); node.setMethodName(configureMethod.getName()); parseExpression(node, null, fullyQualifiedFileName, clazz, configureMethod, block, exp); // flip order of verbs as we parse bottom-up if (node.getVerbs() != null) { Collections.reverse(node.getVerbs()); } } } } } } return answer; } private boolean isRestConfiguration(Expression exp) { String rootMethodName = null; // find out if this is from a Camel route (eg from, route etc.) Expression sub = exp; while (sub instanceof MethodInvocation) { sub = ((MethodInvocation) sub).getExpression(); if (sub instanceof MethodInvocation) { Expression parent = ((MethodInvocation) sub).getExpression(); if (parent == null) { break; } } } if (sub instanceof MethodInvocation) { rootMethodName = ((MethodInvocation) sub).getName().getIdentifier(); } else if (sub instanceof SimpleName) { rootMethodName = ((SimpleName) sub).getIdentifier(); } // must be from rest configuration return "restConfiguration".equals(rootMethodName); } private boolean isRest(Expression exp) { String rootMethodName = null; // find out if this is from a Camel route (eg from, route etc.) Expression sub = exp; while (sub instanceof MethodInvocation) { sub = ((MethodInvocation) sub).getExpression(); if (sub instanceof MethodInvocation) { Expression parent = ((MethodInvocation) sub).getExpression(); if (parent == null) { break; } } } if (sub instanceof MethodInvocation) { rootMethodName = ((MethodInvocation) sub).getName().getIdentifier(); } else if (sub instanceof SimpleName) { rootMethodName = ((SimpleName) sub).getIdentifier(); } // must be from rest return "rest".equals(rootMethodName); } private void parseExpression(RestConfigurationDetails node, String fullyQualifiedFileName, JavaClassSource clazz, MethodSource configureMethod, Block block, Expression exp) { if (exp == null) { return; } if (exp instanceof MethodInvocation) { MethodInvocation mi = (MethodInvocation) exp; doParseRestConfiguration(node, fullyQualifiedFileName, clazz, configureMethod, block, mi); // if the method was called on another method, then recursive exp = mi.getExpression(); parseExpression(node, fullyQualifiedFileName, clazz, configureMethod, block, exp); } } private void parseExpression(RestServiceDetails node, RestVerbDetails verb, String fullyQualifiedFileName, JavaClassSource clazz, MethodSource configureMethod, Block block, Expression exp) { if (exp == null) { // this rest service is not complete, if there is any details on verb then they are actually general // for this rest service and we should pass the details to it if (verb != null) { node.setConsumes(verb.getConsumes()); node.setProduces(verb.getProduces()); node.setSkipBindingOnErrorCode(verb.getSkipBindingOnErrorCode()); node.setClientRequestValidation(verb.getClientRequestValidation()); node.setApiDocs(verb.getApiDocs()); node.setDescription(verb.getDescription()); } return; } if (exp instanceof MethodInvocation) { MethodInvocation mi = (MethodInvocation) exp; verb = doParseRestService(node, verb, fullyQualifiedFileName, clazz, configureMethod, block, mi); // if the method was called on another method, then recursive exp = mi.getExpression(); parseExpression(node, verb, fullyQualifiedFileName, clazz, configureMethod, block, exp); } } private void doParseRestConfiguration(RestConfigurationDetails node, String fullyQualifiedFileName, JavaClassSource clazz, MethodSource configureMethod, Block block, MethodInvocation mi) { // end line number is the first node in the method chain we parse if (node.getLineNumberEnd() == null) { int pos = mi.getStartPosition() + mi.getLength(); int line = findLineNumber(fullyQualifiedFileName, pos); if (line > -1) { node.setLineNumberEnd("" + line); } } String name = mi.getName().getIdentifier(); if ("component".equals(name)) { node.setComponent(extractValueFromFirstArgument(clazz, block, mi)); } else if ("apiComponent".equals(name)) { node.setApiComponent(extractValueFromFirstArgument(clazz, block, mi)); } else if ("producerComponent".equals(name)) { node.setProducerComponent(extractValueFromFirstArgument(clazz, block, mi)); } else if ("scheme".equals(name)) { node.setScheme(extractValueFromFirstArgument(clazz, block, mi)); } else if ("host".equals(name)) { node.setHost(extractValueFromFirstArgument(clazz, block, mi)); } else if ("apiHost".equals(name)) { node.setApiHost(extractValueFromFirstArgument(clazz, block, mi)); } else if ("port".equals(name)) { node.setPort(extractValueFromFirstArgument(clazz, block, mi)); } else if ("producerApiDoc".equals(name)) { node.setProducerApiDoc(extractValueFromFirstArgument(clazz, block, mi)); } else if ("contextPath".equals(name)) { node.setContextPath(extractValueFromFirstArgument(clazz, block, mi)); } else if ("apiContextPath".equals(name)) { node.setApiContextPath(extractValueFromFirstArgument(clazz, block, mi)); } else if ("apiContextRouteId".equals(name)) { node.setApiContextRouteId(extractValueFromFirstArgument(clazz, block, mi)); } else if ("apiContextIdPattern".equals(name)) { node.setApiContextIdPattern(extractValueFromFirstArgument(clazz, block, mi)); } else if ("apiContextListening".equals(name)) { node.setApiContextListening(extractValueFromFirstArgument(clazz, block, mi)); } else if ("apiVendorExtension".equals(name)) { node.setApiVendorExtension(extractValueFromFirstArgument(clazz, block, mi)); } else if ("hostNameResolver".equals(name)) { node.setHostNameResolver(extractValueFromFirstArgument(clazz, block, mi)); } else if ("bindingMode".equals(name)) { node.setBindingMode(extractValueFromFirstArgument(clazz, block, mi)); } else if ("skipBindingOnErrorCode".equals(name)) { node.setSkipBindingOnErrorCode(extractValueFromFirstArgument(clazz, block, mi)); } else if ("clientRequestValidation".equals(name)) { node.setClientRequestValidation(extractValueFromFirstArgument(clazz, block, mi)); } else if ("enableCORS".equals(name)) { node.setEnableCORS(extractValueFromFirstArgument(clazz, block, mi)); } else if ("jsonDataFormat".equals(name)) { node.setJsonDataFormat(extractValueFromFirstArgument(clazz, block, mi)); } else if ("xmlDataFormat".equals(name)) { node.setXmlDataFormat(extractValueFromFirstArgument(clazz, block, mi)); } else if ("componentProperty".equals(name)) { String key = extractValueFromFirstArgument(clazz, block, mi); String value = extractValueFromSecondArgument(clazz, block, mi); node.addComponentProperty(key, value); } else if ("endpointProperty".equals(name)) { String key = extractValueFromFirstArgument(clazz, block, mi); String value = extractValueFromSecondArgument(clazz, block, mi); node.addEndpointProperty(key, value); } else if ("consumerProperty".equals(name)) { String key = extractValueFromFirstArgument(clazz, block, mi); String value = extractValueFromSecondArgument(clazz, block, mi); node.addConsumerProperty(key, value); } else if ("dataFormatProperty".equals(name)) { String key = extractValueFromFirstArgument(clazz, block, mi); String value = extractValueFromSecondArgument(clazz, block, mi); node.addDataFormatProperty(key, value); } else if ("apiProperty".equals(name)) { String key = extractValueFromFirstArgument(clazz, block, mi); String value = extractValueFromSecondArgument(clazz, block, mi); node.addApiProperty(key, value); } else if ("corsHeaderProperty".equals(name)) { String key = extractValueFromFirstArgument(clazz, block, mi); String value = extractValueFromSecondArgument(clazz, block, mi); node.addCorsHeader(key, value); } } private RestVerbDetails doParseRestService(RestServiceDetails node, RestVerbDetails verb, String fullyQualifiedFileName, JavaClassSource clazz, MethodSource configureMethod, Block block, MethodInvocation mi) { // end line number is the first node in the method chain we parse if (node.getLineNumberEnd() == null) { int pos = mi.getStartPosition() + mi.getLength(); int line = findLineNumber(fullyQualifiedFileName, pos); if (line > -1) { node.setLineNumberEnd("" + line); } } String name = mi.getName().getIdentifier(); if ("rest".equals(name)) { node.setPath(extractValueFromFirstArgument(clazz, block, mi)); } else if (isParentMethod(mi, "rest")) { verb = doParseRestVerb(node, verb, clazz, configureMethod, block, mi); } return verb; } private RestVerbDetails doParseRestVerb(RestServiceDetails node, RestVerbDetails verb, JavaClassSource clazz, MethodSource configureMethod, Block block, MethodInvocation mi) { if (verb == null) { verb = new RestVerbDetails(); } String name = mi.getName().getIdentifier(); if ("description".equals(name)) { verb.setDescription(extractValueFromFirstArgument(clazz, block, mi)); } else if ("bindingMode".equals(name)) { verb.setBindingMode(extractValueFromFirstArgument(clazz, block, mi)); } else if ("skipBindingOnErrorCode".equals(name)) { verb.setSkipBindingOnErrorCode(extractValueFromFirstArgument(clazz, block, mi)); } else if ("clientRequestValidation".equals(name)) { verb.setClientRequestValidation(extractValueFromFirstArgument(clazz, block, mi)); } else if ("consumes".equals(name)) { verb.setConsumes(extractValueFromFirstArgument(clazz, block, mi)); } else if ("produces".equals(name)) { verb.setProduces(extractValueFromFirstArgument(clazz, block, mi)); } else if ("type".equals(name)) { verb.setType(extractValueFromFirstArgument(clazz, block, mi)); } else if ("outType".equals(name)) { verb.setOutType(extractValueFromFirstArgument(clazz, block, mi)); } else if ("apiDocs".equals(name)) { verb.setApiDocs(extractValueFromFirstArgument(clazz, block, mi)); } else if ("to".equals(name)) { verb.setTo(extractValueFromFirstArgument(clazz, block, mi)); } else if ("toD".equals(name)) { verb.setToD(extractValueFromFirstArgument(clazz, block, mi)); } else if ("tag".equals(name)) { // tag is only available on the node node.setTag(extractValueFromFirstArgument(clazz, block, mi)); } if ("delete".equals(name)) { node.addVerb(verb); verb.setMethod("delete"); verb.setUri(extractValueFromFirstArgument(clazz, block, mi)); verb = null; // reset as this verb is not complete } else if ("get".equals(name)) { node.addVerb(verb); verb.setMethod("get"); verb.setUri(extractValueFromFirstArgument(clazz, block, mi)); verb = null; // reset as this verb is not complete } else if ("head".equals(name)) { node.addVerb(verb); verb.setMethod("head"); verb.setUri(extractValueFromFirstArgument(clazz, block, mi)); verb = null; // reset as this verb is not complete } else if ("patch".equals(name)) { node.addVerb(verb); verb.setMethod("patch"); verb.setUri(extractValueFromFirstArgument(clazz, block, mi)); verb = null; // reset as this verb is not complete } else if ("post".equals(name)) { node.addVerb(verb); verb.setMethod("post"); verb.setUri(extractValueFromFirstArgument(clazz, block, mi)); verb = null; // reset as this verb is not complete } else if ("put".equals(name)) { node.addVerb(verb); verb.setMethod("put"); verb.setUri(extractValueFromFirstArgument(clazz, block, mi)); verb = null; // reset as this verb is not complete } return verb; } private static boolean isParentMethod(MethodInvocation mi, String parentName) { String name = mi.getName().getIdentifier(); if (parentName.equals(name)) { return true; } // find out if this is from a Camel route (eg from, route etc.) Expression sub = mi; while (sub instanceof MethodInvocation) { sub = ((MethodInvocation) sub).getExpression(); if (sub instanceof MethodInvocation) { name = ((MethodInvocation) sub).getName().getIdentifier(); if (parentName.equals(name)) { return true; } } } return false; } private static String extractValueFromFirstArgument(JavaClassSource clazz, Block block, MethodInvocation mi) { List args = mi.arguments(); if (args != null && args.size() > 0) { Expression exp = (Expression) args.get(0); return getLiteralValue(clazz, block, exp); } return null; } private static String extractValueFromSecondArgument(JavaClassSource clazz, Block block, MethodInvocation mi) { List args = mi.arguments(); if (args != null && args.size() > 1) { Expression exp = (Expression) args.get(1); return getLiteralValue(clazz, block, exp); } return null; } @SuppressWarnings("unchecked") private static FieldSource getField(JavaClassSource clazz, Block block, SimpleName ref) { String fieldName = ref.getIdentifier(); if (fieldName != null) { // find field in class FieldSource field = clazz != null ? clazz.getField(fieldName) : null; if (field == null) { field = findFieldInBlock(clazz, block, fieldName); } return field; } return null; } @SuppressWarnings("unchecked") private static FieldSource findFieldInBlock(JavaClassSource clazz, Block block, String fieldName) { for (Object statement : block.statements()) { // try local statements first in the block if (statement instanceof VariableDeclarationStatement) { final Type type = ((VariableDeclarationStatement) statement).getType(); for (Object obj : ((VariableDeclarationStatement) statement).fragments()) { if (obj instanceof VariableDeclarationFragment) { VariableDeclarationFragment fragment = (VariableDeclarationFragment) obj; SimpleName name = fragment.getName(); if (name != null && fieldName.equals(name.getIdentifier())) { return new StatementFieldSource(clazz, fragment, type); } } } } // okay the field may be burried inside an anonymous inner class as a field declaration // outside the configure method, so lets go back to the parent and see what we can find ASTNode node = block.getParent(); if (node instanceof MethodDeclaration) { node = node.getParent(); } if (node instanceof AnonymousClassDeclaration) { List declarations = ((AnonymousClassDeclaration) node).bodyDeclarations(); for (Object dec : declarations) { if (dec instanceof FieldDeclaration) { FieldDeclaration fd = (FieldDeclaration) dec; final Type type = fd.getType(); for (Object obj : fd.fragments()) { if (obj instanceof VariableDeclarationFragment) { VariableDeclarationFragment fragment = (VariableDeclarationFragment) obj; SimpleName name = fragment.getName(); if (name != null && fieldName.equals(name.getIdentifier())) { return new StatementFieldSource(clazz, fragment, type); } } } } } } } return null; } public static String getLiteralValue(JavaClassSource clazz, Block block, Expression expression) { // unwrap parenthesis if (expression instanceof ParenthesizedExpression) { expression = ((ParenthesizedExpression) expression).getExpression(); } if (expression instanceof StringLiteral) { return ((StringLiteral) expression).getLiteralValue(); } else if (expression instanceof BooleanLiteral) { return "" + ((BooleanLiteral) expression).booleanValue(); } else if (expression instanceof NumberLiteral) { return ((NumberLiteral) expression).getToken(); } // if it a method invocation then add a dummy value assuming the method invocation will return a valid response if (expression instanceof MethodInvocation) { String name = ((MethodInvocation) expression).getName().getIdentifier(); return "{{" + name + "}}"; } // if its a qualified name, then its an enum where we should grab the simple name if (expression instanceof QualifiedName) { QualifiedName qn = (QualifiedName) expression; return qn.getName().getIdentifier(); } if (expression instanceof SimpleName) { FieldSource field = getField(clazz, block, (SimpleName) expression); if (field != null) { // is the field annotated with a Camel endpoint if (field.getAnnotations() != null) { for (Annotation ann : field.getAnnotations()) { boolean valid = "org.apache.camel.EndpointInject".equals(ann.getQualifiedName()) || "org.apache.camel.cdi.Uri".equals(ann.getQualifiedName()); if (valid) { Expression exp = (Expression) ann.getInternal(); if (exp instanceof SingleMemberAnnotation) { exp = ((SingleMemberAnnotation) exp).getValue(); } else if (exp instanceof NormalAnnotation) { List values = ((NormalAnnotation) exp).values(); for (Object value : values) { MemberValuePair pair = (MemberValuePair) value; if ("uri".equals(pair.getName().toString())) { exp = pair.getValue(); break; } } } if (exp != null) { return getLiteralValue(clazz, block, exp); } } } } // is the field an org.apache.camel.Endpoint type? if ("Endpoint".equals(field.getType().getSimpleName())) { // then grab the uri from the first argument VariableDeclarationFragment vdf = (VariableDeclarationFragment) field.getInternal(); expression = vdf.getInitializer(); if (expression instanceof MethodInvocation) { MethodInvocation mi = (MethodInvocation) expression; List args = mi.arguments(); if (args != null && args.size() > 0) { // the first argument has the endpoint uri expression = (Expression) args.get(0); return getLiteralValue(clazz, block, expression); } } } else { // no annotations so try its initializer VariableDeclarationFragment vdf = (VariableDeclarationFragment) field.getInternal(); expression = vdf.getInitializer(); if (expression == null) { // its a field which has no initializer, then add a dummy value assuming the field will be initialized at runtime return "{{" + field.getName() + "}}"; } else { return getLiteralValue(clazz, block, expression); } } } else { // we could not find the field in this class/method, so its maybe from some other super class, so insert a dummy value final String fieldName = ((SimpleName) expression).getIdentifier(); return "{{" + fieldName + "}}"; } } else if (expression instanceof InfixExpression) { String answer = null; // is it a string that is concat together? InfixExpression ie = (InfixExpression) expression; if (InfixExpression.Operator.PLUS.equals(ie.getOperator())) { String val1 = getLiteralValue(clazz, block, ie.getLeftOperand()); String val2 = getLiteralValue(clazz, block, ie.getRightOperand()); // if numeric then we plus the values, otherwise we string concat boolean numeric = isNumericOperator(clazz, block, ie.getLeftOperand()) && isNumericOperator(clazz, block, ie.getRightOperand()); if (numeric) { Long num1 = val1 != null ? Long.valueOf(val1) : 0; Long num2 = val2 != null ? Long.valueOf(val2) : 0; answer = "" + (num1 + num2); } else { answer = (val1 != null ? val1 : "") + (val2 != null ? val2 : ""); } if (!answer.isEmpty()) { // include extended when we concat on 2 or more lines List extended = ie.extendedOperands(); if (extended != null) { for (Object ext : extended) { String val3 = getLiteralValue(clazz, block, (Expression) ext); if (numeric) { Long num3 = val3 != null ? Long.valueOf(val3) : 0; Long num = Long.valueOf(answer); answer = "" + (num + num3); } else { answer += val3 != null ? val3 : ""; } } } } } return answer; } return null; } private static boolean isNumericOperator(JavaClassSource clazz, Block block, Expression expression) { if (expression instanceof NumberLiteral) { return true; } else if (expression instanceof SimpleName) { FieldSource field = getField(clazz, block, (SimpleName) expression); if (field != null) { return field.getType().isType("int") || field.getType().isType("long") || field.getType().isType("Integer") || field.getType().isType("Long"); } } return false; } private static int findLineNumber(String fullyQualifiedFileName, int position) { int lines = 0; try { int current = 0; try (BufferedReader br = new BufferedReader(new FileReader(new File(fullyQualifiedFileName)))) { String line; while ((line = br.readLine()) != null) { lines++; current += line.length() + 1; // add 1 for line feed if (current >= position) { return lines; } } } } catch (Exception e) { // ignore return -1; } return lines; } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy