How to use copy method of com.intuit.karate.core.Variable class

Best Karate code snippet using com.intuit.karate.core.Variable.copy

Source:OpenApiExamplesHook.java Github

copy

Full Screen

1package io.github.apimock;2import com.fasterxml.jackson.core.JsonProcessingException;3import com.fasterxml.jackson.databind.ObjectMapper;4import com.intuit.karate.Json;5import com.intuit.karate.JsonUtils;6import com.intuit.karate.core.Feature;7import com.intuit.karate.core.MockHandlerHook;8import com.intuit.karate.core.ScenarioEngine;9import com.intuit.karate.core.ScenarioRuntime;10import com.intuit.karate.core.Variable;11import com.intuit.karate.http.HttpUtils;12import com.intuit.karate.http.Request;13import com.intuit.karate.http.Response;14import org.openapi4j.parser.model.v3.Example;15import org.openapi4j.parser.model.v3.MediaType;16import org.openapi4j.parser.model.v3.OpenApi3;17import org.openapi4j.parser.model.v3.Operation;18import org.slf4j.Logger;19import org.slf4j.LoggerFactory;20import java.text.SimpleDateFormat;21import java.util.AbstractMap;22import java.util.ArrayList;23import java.util.Calendar;24import java.util.Collections;25import java.util.Date;26import java.util.GregorianCalendar;27import java.util.HashMap;28import java.util.List;29import java.util.Map;30import java.util.function.BiFunction;31import java.util.function.Function;32import java.util.function.Supplier;33import java.util.regex.Matcher;34import java.util.regex.Pattern;35/**36 *37 * @author ivangsa38 */39public class OpenApiExamplesHook implements MockHandlerHook {40 private Logger logger = LoggerFactory.getLogger(getClass());41 private final OpenApiValidator4Karate openApiValidator;42 private OpenApi3 api;43 private ObjectMapper jacksonMapper = new ObjectMapper();44 public OpenApiExamplesHook(OpenApiValidator4Karate openApiValidator) {45 super();46 this.openApiValidator = openApiValidator;47 this.api = openApiValidator.getApi();48 }49 @Override50 public void reload() {51 openApiValidator.reload();52 this.api = openApiValidator.getApi();53 sequenceNext = 0;54 }55 @Override56 public void onSetup(Map<Feature, ScenarioRuntime> features, Map<String, Variable> globals) {57 if(!globals.containsKey(UUID)) {58 globals.put(UUID, new Variable((Supplier<String>) this::uuid));59 }60 if(!globals.containsKey(SEQUENCE_NEXT)) {61 globals.put(SEQUENCE_NEXT, new Variable((Supplier<Integer>) this::sequenceNext));62 }63 if(!globals.containsKey(NOW)) {64 globals.put(NOW, new Variable((Function<String,String>) this::now));65 }66 if(!globals.containsKey(DATE)) {67 globals.put(DATE, new Variable((BiFunction<String, String, String>) this::date));68 }69 if(api.getComponents() != null && api.getComponents().getExamples() != null) {70 ScenarioEngine engine = new ScenarioEngine(features.values().stream().findFirst().get(), new HashMap<>(globals));71 engine.init();72 for (Example example : api.getComponents().getExamples().values()) {73 String karateVar = (String) firstNotNull(example.getExtensions(), Collections.emptyMap()).get("x-apimock-karate-var");74 if(isNotEmpty(karateVar)) {75 Object seeds = firstNotNull(firstNotNull(example.getExtensions(), Collections.emptyMap()).get("x-apimock-seed"), 1);76 Map<String, Object> seedsMap = seeds instanceof Integer? defaultRootSeed((Integer) seeds): (Map<String, Object>) seeds;77 Object seededExample = seed(example.getValue(), seedsMap);78 try {79 Map<String, String> transforms = (Map) firstNotNull(example.getExtensions(), Collections.emptyMap()).get("x-apimock-transform");80 String json = processObjectDynamicProperties(engine, transforms, seededExample);81 Variable exampleVariable = new Variable(Json.of(json).value());82 addExamplesVariableToKarateGlobals(globals, karateVar, exampleVariable);83 } catch (Exception e) {84 logger.error("Error setting openapi examples {} into karate globals ({})", karateVar, e.getMessage(), e);85 }86 }87 }88 }89 }90 private Map<String, Object> defaultRootSeed(Integer seed) {91 Map<String, Object> seedMap = new HashMap<>();92 seedMap.put("$", seed);93 return seedMap;94 }95 private Object seed(Object value, Map<String, Object> seedsMap) {96 Json json = Json.of(value);97 for (Map.Entry<String, Object> seedEntry : seedsMap.entrySet()) {98 int seed = (Integer) seedEntry.getValue();99 if(seed == 1) {100 continue;101 }102 String seedPath = String.valueOf(seedEntry.getKey());103 Object inner = json.get(seedPath);104 Object seeded = seedValue(inner, seed);105 json = replace(json, seedPath, seeded);106 }107 return json.get("$");108 }109 private List seedValue(Object value, int seed) {110 List seeded = new ArrayList();111 for (int i = 0; i < seed; i++) {112 if(value instanceof List) {113 seeded.addAll((List) JsonUtils.deepCopy(value));114 } else {115 seeded.add(JsonUtils.deepCopy(value));116 }117 }118 return seeded;119 }120 private Json replace(Json json, String path, Object replacement) {121 if("$".equals(path)) {122 return Json.of(replacement);123 }124 json.set(path, replacement);125 return json;126 }127 private void addExamplesVariableToKarateGlobals(Map<String, Variable> globals, String karateVar, Variable examplesVariable) {128 if(!globals.containsKey(karateVar)) {129 globals.put(karateVar, examplesVariable);130 } else {131 Variable karateVariable = globals.get(karateVar);132 if(karateVariable.isList()) {133 if(examplesVariable.isList()) {134 ((List)karateVariable.getValue()).addAll(examplesVariable.getValue());135 } else {136 ((List)karateVariable.getValue()).add(examplesVariable.getValue());137 }138 }139 if(karateVariable.isMap() && examplesVariable.isMap()) {140 ((Map)karateVariable.getValue()).putAll(examplesVariable.getValue());141 }142 }143 }144 @Override145 public Response noMatchingScenario(Request req, Response response, ScenarioEngine engine) {146 Operation operation = OpenApiValidator4Karate.findOperation(req.getMethod(), req.getPath(), api);147 if(operation == null) {148 logger.debug("Operation not found for {}", req.getPath());149 return response;150 }151 logger.debug("Searching examples in openapi definition for operationId {}", operation.getOperationId());152 Map<String, org.openapi4j.parser.model.v3.Response> responses = OpenApiValidator4Karate.find2xxResponses(operation);153 loadPathParams(req.getPath(), (String) operation.getExtensions().get("x-apimock-internal-path"), engine);154 if(!responses.isEmpty()) {155 String status = responses.keySet().stream().findFirst().get();156 org.openapi4j.parser.model.v3.Response oasRespose = responses.get(status);157 // match media type from request158 String contentType = getContentType(req);159 Map.Entry<String, MediaType> mediaTypeEntry = oasRespose.getContentMediaTypes().entrySet().stream()160 .filter(e -> e.getKey().startsWith(contentType))161 .findFirst().orElse(new AbstractMap.SimpleEntry("", new MediaType()));162 if(mediaTypeEntry.getValue().getExamples() == null && mediaTypeEntry.getValue().getExample() != null) {163 logger.debug("Returning default example in openapi for operationId {}", operation.getOperationId());164 response = new Response(Integer.valueOf(status.toLowerCase().replaceAll("x", "0")));165 response.setBody(processObjectDynamicProperties(engine, null, mediaTypeEntry.getValue().getExample()));166 response.setContentType(mediaTypeEntry.getKey());167 response.setHeader("access-control-allow-origin", "*");168 unloadPathParams(engine);169 return response;170 }171 for (Map.Entry<String, Example> exampleEntry: mediaTypeEntry.getValue().getExamples().entrySet()) {172 Map<String, Object> extensions = exampleEntry.getValue().getExtensions();173 if(extensions == null) {174 continue;175 }176 Object when = extensions.get("x-apimock-when");177 Map<String, String> generators = (Map<String, String>) extensions.get("x-apimock-transform");178 if(when != null) {179 if(evalBooleanJs(engine, when.toString())) {180 logger.debug("Found example[{}] for x-apimock-when {} in openapi for operationId {}", exampleEntry.getKey(), when, operation.getOperationId());181 Example example = exampleEntry.getValue();182 Object seeds = firstNotNull(firstNotNull(example.getExtensions(), Collections.emptyMap()).get("x-apimock-seed"), 1);183 Map<String, Object> seedsMap = seeds instanceof Integer? defaultRootSeed((Integer) seeds): (Map<String, Object>) seeds;184 Object seededExample = seed(example.getValue(), seedsMap);185 logger.debug("Returning example in openapi for operationId {}", operation.getOperationId());186 response = new Response(Integer.valueOf(status.toLowerCase().replaceAll("x", "0")));187 response.setBody(processObjectDynamicProperties(engine, generators, seededExample));188 response.setContentType(mediaTypeEntry.getKey());189 response.setHeader("access-control-allow-origin", "*");190 break;191 }192 }193 }194 }195 unloadPathParams(engine);196 return response;197 }198 private String getContentType(Request req) {199 String contentType = firstNotNull(req.getContentType(), "application/json");200 return contentType.contains(";")? contentType.substring(0, contentType.indexOf(";")) : contentType;201 }202 protected void evaluateJsAndReplacePath(ScenarioEngine engine, Json json, String path, String js) {203 Object replacement = evalJsAsObject(engine, js);204 try {205 if (replacement != null) {206 json.set(path, replacement);207 }208 } catch (Exception e) {209 logger.error("Error replacing jsonPath: {} ({})", path, e.getMessage());210 }211 }212 Pattern generatorsPattern = Pattern.compile("\\{\\{(.+)\\}\\}");213 protected String processObjectDynamicProperties(ScenarioEngine engine, Map<String, String> generators, Object value) {214 if(value == null) {215 return null;216 }217 Json json = Json.of(value);218 if(generators != null) {219 for (Map.Entry<String, String> entry: generators.entrySet()){220 if(entry.getKey().startsWith("$[*]") && json.isArray()) {221 List list = json.asList();222 for(int i = 0; i < list.size(); i++) {223 evaluateJsAndReplacePath(engine, json, entry.getKey().replace("$[*]", "$[" + i + "]"), entry.getValue());224 }225 } else {226 evaluateJsAndReplacePath(engine, json, entry.getKey(), entry.getValue());227 }228 }229 }230 String jsonString = toJsonPrettyString(json);231 final Matcher matcher = generatorsPattern.matcher(jsonString);232 while (matcher.find()) {233 String match = matcher.group(0);234 String script = matcher.group(1);235 logger.trace("Processing inline replacement for script: {}", script);236 String replacement = evalJsAsString(engine, script);237 if(replacement != null) {238 jsonString = jsonString.replace(match, replacement);239 }240 }241 return JsonUtils.toStrictJson(jsonString);242 }243 private String toJsonPrettyString(Json json) {244 try {245 return jacksonMapper.writerWithDefaultPrettyPrinter().writeValueAsString(json.value());246 } catch (JsonProcessingException e) {247 return json.toStringPretty();248 }249 }250 private void loadPathParams(String uri, String pattern, ScenarioEngine engine) {251 Map<String, String> pathParams = HttpUtils.parseUriPattern(pattern, uri);252 if (pathParams != null) {253 engine.setVariable("pathParams", pathParams);254 }255 }256 private void unloadPathParams(ScenarioEngine engine) {257 engine.setVariable("pathParams", null);258 }259 private boolean evalBooleanJs(ScenarioEngine engine, String js) {260 try {261 return engine.evalJs(js).isTrue();262 } catch (Exception e) {263 logger.error("Error evaluating boolean script: '{}' ({})", js, e.getMessage());264 return false;265 }266 }267 private String evalJsAsString(ScenarioEngine engine, String js) {268 try {269 return engine.evalJs(js).getAsString();270 } catch (Exception e) {271 logger.error("Error evaluating string script: '{}' ({})", js, e.getMessage());272 return null;273 }274 }275 private Object evalJsAsObject(ScenarioEngine engine, String js) {276 try {277 Object result = engine.evalJs(js).getValue();278 return result != null? result : "";279 } catch (Exception e) {280 logger.error("Error evaluating script: '{}' ({})", js, e.getMessage());281 return null;282 }283 }284 private String uuid() {285 return java.util.UUID.randomUUID().toString();286 }287 private int sequenceNext = 0;288 private int sequenceNext() {289 return sequenceNext++;290 }291 private String now(String format) {292 Date now = new Date();293 return new SimpleDateFormat(format).format(now);294 }295 private String date(String format, String intervalExpression) {296 int length = intervalExpression.length();297 String intervalString = intervalExpression.trim().substring(0, length - 1);298 String range = intervalExpression.trim().substring(length);299 int amount = Integer.parseInt(intervalString);300 int field = Calendar.DATE;301 if(range.equalsIgnoreCase("d")) {302 field = Calendar.DATE;303 }304 if(range.equalsIgnoreCase("h")) {305 field = Calendar.HOUR;306 }307 if(range.equalsIgnoreCase("s")) {308 field = Calendar.SECOND;309 }310 GregorianCalendar calendar = new GregorianCalendar();311 calendar.setTime(new Date());312 calendar.add(field, amount);313 return new SimpleDateFormat(format).format(calendar.getTime());314 }315 private <T> T firstNotNull(T one, T two) {316 return one != null? one : two;317 }318 private boolean isNotEmpty(String str) {319 return str != null && !str.trim().equals("");320 }321 private static final String UUID = "uuid";322 private static final String SEQUENCE_NEXT = "sequenceNext";323 private static final String NOW = "now";324 private static final String DATE = "date";325}...

Full Screen

Full Screen

Source:SuiteReports.java Github

copy

Full Screen

2 * The MIT License3 *4 * Copyright 2021 Intuit Inc.5 *6 * Permission is hereby granted, free of charge, to any person obtaining a copy7 * of this software and associated documentation files (the "Software"), to deal8 * in the Software without restriction, including without limitation the rights9 * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell10 * copies of the Software, and to permit persons to whom the Software is11 * furnished to do so, subject to the following conditions:12 *13 * The above copyright notice and this permission notice shall be included in14 * all copies or substantial portions of the Software.15 *16 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR17 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,18 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE19 * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER20 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,21 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN22 * THE SOFTWARE.23 */24package com.intuit.karate.report;25import com.intuit.karate.FileUtils;26import com.intuit.karate.Results;27import com.intuit.karate.Suite;...

Full Screen

Full Screen

copy

Using AI Code Generation

copy

Full Screen

1import com.intuit.karate.core.Variable;2import com.intuit.karate.core.Feature;3import com.intuit.karate.core.FeatureContext;4import com.intuit.karate.core.Scenario;5import com.intuit.karate.core.ScenarioContext;6import com.intuit.karate.core.FeatureRuntime;7import com.intuit.karate.core.ScenarioRuntime;8import com.intuit.karate.core.ScenarioResu

Full Screen

Full Screen

copy

Using AI Code Generation

copy

Full Screen

1import com.intuit.karate.core.Variable;2import java.util.HashMap;3import java.util.Map;4public class 4 {5 public static void main(String[] args) {6 Map<String, Object> map = new HashMap<>();7 map.put("a", "b");8 map.put("c", "d");9 map.put("e", "f");10 Variable var = new Variable(map);11 System.out.println(var);12 Variable var2 = var.copy();13 System.out.println(var2);14 }15}16import com.intuit.karate.core.Variable;17import java.util.HashMap;18import java.util.Map;19public class 4 {20 public static void main(String[] args) {21 Map<String, Object> map = new HashMap<>();22 map.put("a", "b");23 map.put("c", "d");24 map.put("e", "f");25 Variable var = new Variable(map);26 System.out.println(var);27 Variable var2 = var.copy();28 System.out.println(var2);29 }30}31import com.intuit.karate.core.Variable;32import java.util.HashMap;33import java.util.Map;34public class 4 {35 public static void main(String[] args) {36 Map<String, Object> map = new HashMap<>();37 map.put("a", "b");38 map.put("c", "d");39 map.put("e", "f");40 Variable var = new Variable(map);

Full Screen

Full Screen

copy

Using AI Code Generation

copy

Full Screen

1import com.intuit.karate.core.Variable;2import java.util.HashMap;3import java.util.Map;4public class 4 {5 public static void main(String[] args) {6 Map<String, Object> map = new HashMap<>();7 map.put("foo", "bar");8 map.put("key", "value");9 Variable var = new Variable(map);10 Variable copy = var.copy();11 copy.setValue("foo", "baz");12 System.out.println(var.getValue("foo"));13 }14}15Your name to display (optional):

Full Screen

Full Screen

copy

Using AI Code Generation

copy

Full Screen

1import com.intuit.karate.core.Variable;2import java.util.HashMap;3import java.util.Map;4import java.util.List;5import java.util.ArrayList;6import com.intuit.karate.core.Feature;7import com.intuit.karate.core.FeatureContext;8import com.intuit.karate.core.ScenarioContext;9import com.intuit.karate.core.Scenario;10import com.intuit.karate.core.ScenarioRuntime;11import com.intuit.karate.core.FeatureRuntime;12import com.intuit.karate.core.FeatureResult;13import com.intuit.karate.core.ScenarioResult;14import com.intuit.karate

Full Screen

Full Screen

copy

Using AI Code Generation

copy

Full Screen

1import com.intuit.karate.core.Variable;2import java.util.Map;3import java.util.HashMap;4public class 4{5 public static void main(String[] args) {6 Map<String, Object> map1 = new HashMap<String, Object>();7 map1.put("test", "test");8 Map<String, Object> map2 = new HashMap<String, Object>();9 map2.put("test", "test");10 Variable var1 = new Variable(map1);11 Variable var2 = new Variable(map2);12 var1.copy(var2);13 System.out.println(var1);14 }15}16import com.intuit.karate.core.Variable;17import java.util.Map;18import java.util.HashMap;19public class 5{20 public static void main(String[] args) {21 Map<String, Object> map1 = new HashMap<String, Object>();22 map1.put("test", "test");23 Map<String, Object> map2 = new HashMap<String, Object>();24 map2.put("test", "test");25 Variable var1 = new Variable(map1);26 Variable var2 = new Variable(map2);27 var1.copy(var2);28 System.out.println(var1);29 }30}31import com.intuit.karate.core.Variable;32import java.util.Map;33import java.util.HashMap;34public class 6{35 public static void main(String[] args) {36 Map<String, Object> map1 = new HashMap<String, Object>();37 map1.put("test", "test");38 Map<String, Object> map2 = new HashMap<String, Object>();39 map2.put("test", "test");40 Variable var1 = new Variable(map1);41 Variable var2 = new Variable(map2);42 var1.copy(var2);43 System.out.println(var1);44 }45}46import com.intuit.karate.core.Variable;47import java.util.Map;48import java.util.HashMap;49public class 7{50 public static void main(String[] args) {51 Map<String, Object> map1 = new HashMap<String, Object>();52 map1.put("test", "

Full Screen

Full Screen

copy

Using AI Code Generation

copy

Full Screen

1import com.intuit.karate.core.Variable;2import java.io.*;3import java.util.*;4import org.apache.commons.io.FileUtils;5public class 4 {6 public static void main(String[] args) throws IOException {7 Variable var = new Variable("test");8 File file = new File("C:\\Users\\sakshi\\Desktop\\test.txt");9 var.copy(file);10 }11}12import com.intuit.karate.core.Variable;13import java.io.*;14import java.util.*;15import org.apache.commons.io.FileUtils;16public class 5 {17 public static void main(String[] args) throws IOException {18 Variable var = new Variable("test");19 File file = new File("C:\\Users\\sakshi\\Desktop\\test.txt");20 var.copy(file);21 }22}23import com.intuit.karate.core.Variable;24import java.io.*;25import java.util.*;26import org.apache.commons.io.FileUtils;27public class 6 {28 public static void main(String[] args) throws IOException {29 Variable var = new Variable("test");30 File file = new File("C:\\Users\\sakshi\\Desktop\\test.txt");31 var.copy(file);32 }33}34import com.intuit.karate.core.Variable;35import java.io.*;36import java.util.*;37import org.apache.commons.io.FileUtils;38public class 7 {39 public static void main(String[] args) throws IOException {40 Variable var = new Variable("test");41 File file = new File("C:\\Users\\sakshi\\Desktop\\test.txt");42 var.copy(file);43 }44}45import com.intuit.karate.core.Variable;46import java.io.*;47import java.util.*;48import org.apache.commons.io.FileUtils;49public class 8 {50 public static void main(String[] args) throws IOException {51 Variable var = new Variable("test");52 File file = new File("C:\\Users\\sakshi\\Desktop\\test.txt");53 var.copy(file);54 }55}

Full Screen

Full Screen

copy

Using AI Code Generation

copy

Full Screen

1import com.intuit.karate.core.Variable;2import java.util.HashMap;3import java.util.Map;4import java.util.ArrayList;5public class 4{6 public static void main(String[] args){7 Map<String,Object> m = new HashMap();8 m.put("a", "a");9 m.put("b", "b");10 m.put("c", "c");11 m.put("d", "d");12 m.put("e", "e");13 m.put("f", "f");14 m.put("g", "g");15 m.put("h", "h");16 m.put("i", "i");17 m.put("j", "j");18 m.put("k", "k");19 m.put("l", "l");20 m.put("m", "m");21 m.put("n", "n");22 m.put("o", "o");23 m.put("p", "p");24 m.put("q", "q");25 m.put("r", "r");26 m.put("s", "s");27 m.put("t", "t");28 m.put("u", "u");29 m.put("v", "v");30 m.put("w", "w");31 m.put("x", "x");32 m.put("y", "y");33 m.put("z", "z");34 m.put("1", "1");35 m.put("2", "2");36 m.put("3", "3");37 m.put("4", "4");38 m.put("5", "5");39 m.put("6", "6");40 m.put("7", "7");41 m.put("8", "8");42 m.put("9", "9");43 m.put("0", "0");44 m.put("!", "!");45 m.put("@", "@");46 m.put("#", "#");47 m.put("$", "$");48 m.put("%", "%");49 m.put("^", "^");50 m.put("&", "&");51 m.put("*", "*");52 m.put("(", "(");53 m.put(")", ")");54 m.put("-", "-");55 m.put("_", "_");56 m.put("+", "+");57 m.put("=", "=");58 m.put("{", "{");59 m.put("}", "}");60 m.put("[", "[");61 m.put("]", "]");

Full Screen

Full Screen

copy

Using AI Code Generation

copy

Full Screen

1import com.intuit.karate.core.Variable;2import com.intuit.karate.core.Variable.Type;3public class 4 {4 public static void main(String[] args) {5 Variable var = new Variable(Type.STRING, "This is a string");6 Variable var2 = Variable.copy(var);7 System.out.println(var2.getValue());8 }9}10import com.intuit.karate.core.Variable;11import com.intuit.karate.core.Variable.Type;12public class 5 {13 public static void main(String[] args) {14 Variable var = new Variable(Type.STRING, "This is a string");15 Variable var2 = Variable.copy(var);16 System.out.println(var.equals(var2));17 }18}19import com.intuit.karate.core.Variable;20import com.intuit.karate.core.Variable.Type;21public class 6 {22 public static void main(String[] args) {23 Variable var = new Variable(Type.STRING, "This is a string");24 System.out.println(var.hashCode());25 }26}

Full Screen

Full Screen

Automation Testing Tutorials

Learn to execute automation testing from scratch with LambdaTest Learning Hub. Right from setting up the prerequisites to run your first automation test, to following best practices and diving deeper into advanced test scenarios. LambdaTest Learning Hubs compile a list of step-by-step guides to help you be proficient with different test automation frameworks i.e. Selenium, Cypress, TestNG etc.

LambdaTest Learning Hubs:

YouTube

You could also refer to video tutorials over LambdaTest YouTube channel to get step by step demonstration from industry experts.

Try LambdaTest Now !!

Get 100 minutes of automation test minutes FREE!!

Next-Gen App & Browser Testing Cloud

Was this article helpful?

Helpful

NotHelpful