How to use ScenarioOutline method of com.intuit.karate.core.ScenarioOutline class

Best Karate code snippet using com.intuit.karate.core.ScenarioOutline.ScenarioOutline

Source:VSCodeOutputRuntimeHook.java Github

copy

Full Screen

...5import com.intuit.karate.Suite;6import com.intuit.karate.core.Feature;7import com.intuit.karate.core.FeatureRuntime;8import com.intuit.karate.core.Scenario;9import com.intuit.karate.core.ScenarioOutline;10import com.intuit.karate.core.ScenarioRuntime;11import com.intuit.karate.core.Step;12import com.intuit.karate.core.StepResult;13import com.intuit.karate.http.HttpRequest;14import com.intuit.karate.http.Response;15import org.slf4j.LoggerFactory;16import java.io.IOException;17import java.net.InetSocketAddress;18import java.net.StandardSocketOptions;19import java.nio.ByteBuffer;20import java.nio.channels.SocketChannel;21import java.text.SimpleDateFormat;22import java.util.Collections;23import java.util.Date;24import java.util.HashMap;25import java.util.List;26import java.util.Map;27import java.util.concurrent.ExecutionException;28import java.util.concurrent.TimeoutException;29import java.util.stream.Collectors;30import static java.nio.charset.StandardCharsets.UTF_8;31/**32 * @author ivangsa33 */34public class VSCodeOutputRuntimeHook implements ExtendedRuntimeHook {35 private org.slf4j.Logger log = LoggerFactory.getLogger(this.getClass());36 @Override37 public void beforeSuite(Suite suite) {38 try {39 String features = suite.features.stream().map(f -> f.getResource().getRelativePath()).collect(Collectors.joining(";"));40 println(String.format(SUITE_STARTED, getCurrentTime(), features, suite.featuresFound));41 // log.trace(String.format(SUITE_STARTED, getCurrentTime(), features, suite.featuresFound));42 } catch (Exception e) {43 log.error("beforeSuite error: {}", e.getMessage());44 }45 }46 @Override47 public void afterSuite(Suite suite) {48 try {49 println(String.format(SUITE_FINISHED, getCurrentTime(), suite.buildResults().getEndTime() - suite.startTime));50 // log.trace(String.format(SUITE_FINISHED, getCurrentTime(), suite.buildResults().getEndTime() - suite.startTime));51 } catch (Exception e) {52 log.error("afterSuite error: {}", e.getMessage());53 }54 }55 @Override56 public boolean beforeFeature(FeatureRuntime fr) {57 try {58 if (fr.caller.depth == 0) {59 String path = fr.feature.getResource().getRelativePath();60 println(String.format(FEATURE_STARTED, getCurrentTime(), path + ":" + fr.feature.getLine(), escape(fr.feature.getNameForReport())));61 // log.trace(String.format(FEATURE_STARTED, getCurrentTime(), path + ":" + fr.feature.getLine(), escape(fr.feature.getNameForReport())));62 }63 } catch (Exception e) {64 log.error("beforeFeature error: {}", e.getMessage());65 }66 return true;67 }68 @Override69 public void afterFeature(FeatureRuntime fr) {70 try {71 if (fr.caller.depth == 0) {72 String path = fr.feature.getResource().getRelativePath();73 println(String.format(FEATURE_FINISHED, getCurrentTime(), path + ":" + fr.feature.getLine(), (int) fr.result.getDurationMillis(), escape(fr.feature.getNameForReport())));74 // log.trace(String.format(FEATURE_FINISHED, getCurrentTime(), path + ":" + fr.feature.getLine(), (int) fr.result.getDurationMillis(), escape(fr.feature.getNameForReport())));75 }76 } catch (Exception e) {77 log.error("afterFeature error: {}", e.getMessage());78 }79 }80 @Override81 public boolean beforeScenario(ScenarioRuntime sr) {82 try {83 if (sr.caller.depth == 0) {84 String path = sr.scenario.getFeature().getResource().getRelativePath();85 println(String.format(SCENARIO_STARTED, getCurrentTime(), path + ":" + sr.scenario.getLine(), escape(sr.scenario.getRefIdAndName()), sr.scenario.isOutlineExample(), sr.scenario.isDynamic()));86 // log.trace(String.format(SCENARIO_STARTED, getCurrentTime(), path + ":" + sr.scenario.getLine(), escape(sr.scenario.getRefIdAndName()), sr.scenario.isOutlineExample(), sr.scenario.isDynamic()));87 }88 } catch (Exception e) {89 log.error("beforeScenario error: {}", e.getMessage());90 }91 return true;92 }93 @Override94 public void afterScenario(ScenarioRuntime sr) {95 try {96 // System.out.println(String.format("#vscode afterScenario %s %s", sr.caller.depth, sr.scenario.getRefIdAndName()));97 if (sr.caller.depth == 0) {98 String path = sr.scenario.getFeature().getResource().getRelativePath();99 if (sr.result.isFailed()) {100 StringUtils.Pair error = details(sr.result.getErrorMessage());101 println(String.format(SCENARIO_FAILED, getCurrentTime(), path + ":" + sr.scenario.getLine(), (int) sr.result.getDurationMillis(), sr.scenario.isOutlineExample(), sr.scenario.isDynamic(), escape(error.right), escape(error.left), escape(sr.scenario.getRefIdAndName()), ""));102 // log.trace(String.format(SCENARIO_FAILED, getCurrentTime(), path + ":" + fr.feature.getLine(), (int) sr.result.getDurationMillis(), sr.scenario.isOutlineExample(), sr.scenario.isDynamic(), escape(error.right), escape(error.right), escape(error.left), escape(sr.scenario.getRefIdAndName()), ""));103 } else {104 println(String.format(SCENARIO_FINISHED, getCurrentTime(), path + ":" + sr.scenario.getLine(), (int) sr.result.getDurationMillis(), sr.scenario.isOutlineExample(), sr.scenario.isDynamic(), escape(sr.scenario.getRefIdAndName())));105 // log.trace(String.format(SCENARIO_FINISHED, getCurrentTime(), path + ":" + fr.feature.getLine(), (int) sr.result.getDurationMillis(), sr.scenario.isOutlineExample(), sr.scenario.isDynamic(), escape(sr.scenario.getRefIdAndName())));106 }107 }108 } catch (Exception e) {109 log.error("afterScenario error: {}", e.getMessage());110 }111 }112 @Override113 public boolean beforeScenarioOutline(ScenarioOutline scenarioOutline, ScenarioRuntime sr) {114 try {115 if (sr.caller.depth == 0) {116 String path = sr.scenario.getFeature().getResource().getRelativePath();117 String outlineName = getOutlineName(sr);118 println(String.format(SCENARIO_OUTLINE_STARTED, getCurrentTime(), path + ":" + sr.scenario.getSection().getScenarioOutline().getLine(), escape(outlineName), sr.scenario.isOutlineExample(), sr.scenario.isDynamic()));119 // log.trace(String.format(SCENARIO_OUTLINE_STARTED, getCurrentTime(), path + ":" + sr.scenario.getSection().getScenarioOutline().getLine(), escape(outlineName), sr.scenario.isOutlineExample(), sr.scenario.isDynamic()));120 }121 } catch (Exception e) {122 log.error("beforeScenarioOutline error: {}", e.getMessage());123 }124 return true;125 }126 @Override127 public void afterScenarioOutline(ScenarioOutline scenarioOutline, ScenarioRuntime sr) {128 try {129 if (sr.caller.depth == 0) {130 String path = sr.scenario.getFeature().getResource().getRelativePath();131 String outlineName = getOutlineName(sr);132 println(String.format(SCENARIO_OUTLINE_FINISHED, getCurrentTime(), path + ":" + scenarioOutline.getLine(), (int) sr.result.getDurationMillis(), escape(outlineName)));133 // log.trace(String.format(SCENARIO_OUTLINE_FINISHED, getCurrentTime(), path + ":" + scenarioOutline.getLine(), (int) sr.result.getDurationMillis(), escape(outlineName)));134 }135 } catch (Exception e) {136 log.error("afterScenarioOutline error: {}", e.getMessage());137 }138 }139 static void println(String s) {140 System.out.println(s);141 }142 private static String getCurrentTime() {143 return DATE_FORMAT.format(new Date());144 }145 private static String escape(String source) {146 if (source == null) {147 return "";148 }149 return source.replace("\n", "\\n").replace("\r", "\\r").replace("'", "\'").replace("\"", "\\\"");150 }...

Full Screen

Full Screen

Source:VSCodeHook.java Github

copy

Full Screen

1package vscode;2import com.intuit.karate.RuntimeHook;3import com.intuit.karate.Suite;4import com.intuit.karate.core.FeatureRuntime;5import com.intuit.karate.core.ScenarioOutline;6import com.intuit.karate.core.ScenarioRuntime;7import com.intuit.karate.core.Step;8import com.intuit.karate.core.StepResult;9import com.intuit.karate.http.HttpRequest;10import com.intuit.karate.http.Response;11import org.slf4j.LoggerFactory;12import java.util.Arrays;13import java.util.Collections;14import java.util.List;15/**16 * @author ivangsa17 */18public class VSCodeHook implements RuntimeHook {19 private org.slf4j.Logger log = LoggerFactory.getLogger(this.getClass());20 private List<ExtendedRuntimeHook> runtimeHooks;21 public VSCodeHook() {22 this(new VSCodeOutputRuntimeHook(), new VSCodeSocketRuntimeHook());23 }24 public VSCodeHook(ExtendedRuntimeHook... runtimeHooks) {25 this.runtimeHooks = runtimeHooks != null? Arrays.asList(runtimeHooks) : Collections.emptyList();26 }27 // used to track feature end when no scenarios where selected28 private ThreadLocal<String> currentFeature = new ThreadLocal<>();29 // used to track scenario outline start/stop30 private ThreadLocal<ScenarioOutline> currentOutline = new ThreadLocal<>();31 private ThreadLocal<ScenarioRuntime> currentScenarioRuntime = new ThreadLocal<>();32 @Override33 public void beforeSuite(Suite suite) {34 runtimeHooks.stream().forEach(h -> h.beforeSuite(suite));35 }36 @Override37 public void afterSuite(Suite suite) {38 runtimeHooks.stream().forEach(h -> h.afterSuite(suite));39 }40 @Override41 public boolean beforeFeature(FeatureRuntime fr) {42 if(currentFeature.get() != null) {43 // fixes afterFeature not being called for empty features (no scenarios selected by tags)44 afterFeature(fr);45 }46 this.currentFeature.set(fr.feature.getNameForReport());47 return runtimeHooks.stream().reduce(true, (accumulated, hook) -> hook.beforeFeature(fr), (accumulated, current) -> accumulated && current);48 }49 @Override50 public void afterFeature(FeatureRuntime fr) {51 if (fr.caller.depth == 0 && currentOutline.get() != null) {52 runtimeHooks.stream().forEach(h -> h.afterScenarioOutline(currentOutline.get(), currentScenarioRuntime.get()));53 currentOutline.set(null);54 currentScenarioRuntime.set(null);55 }56 runtimeHooks.stream().forEach(h -> h.afterFeature(fr));57 currentFeature.set(null);58 }59 @Override60 public boolean beforeScenario(ScenarioRuntime sr) {61 sr.evaluateScenarioName();62 if (sr.caller.depth == 0) {63 ScenarioOutline scenarioOutline = sr.scenario.getSection().isOutline() ? sr.scenario.getSection().getScenarioOutline() : null;64 if (currentOutline.get() != null && !currentOutline.get().equals(scenarioOutline)) { // changing from an outline65 runtimeHooks.stream().forEach(h -> h.afterScenarioOutline(currentOutline.get(), sr));66 currentOutline.set(null);67 }68 if (scenarioOutline != null && !scenarioOutline.equals(currentOutline.get())) { // entering an outline69 currentOutline.set(scenarioOutline);70 runtimeHooks.stream().forEach(h -> h.beforeScenarioOutline(sr.scenario.getSection().getScenarioOutline(), sr));71 }72 currentScenarioRuntime.set(sr);73 }74 return runtimeHooks.stream().reduce(true, (accumulated, hook) -> hook.beforeScenario(sr), (accumulated, current) -> accumulated && current);75 }76 @Override77 public void afterScenario(ScenarioRuntime sr) {78 runtimeHooks.stream().forEach(h -> h.afterScenario(sr));79 }80 @Override81 public boolean beforeStep(Step step, ScenarioRuntime sr) {82 return runtimeHooks.stream().reduce(true, (accumulated, hook) -> hook.beforeStep(step, sr), (accumulated, current) -> accumulated && current);83 }84 @Override...

Full Screen

Full Screen

Source:ExtendedRuntimeHook.java Github

copy

Full Screen

1package vscode;2import com.intuit.karate.RuntimeHook;3import com.intuit.karate.core.ScenarioOutline;4import com.intuit.karate.core.ScenarioRuntime;5public interface ExtendedRuntimeHook extends RuntimeHook {6 default String getOutlineName(ScenarioRuntime sr) {7 return sr.scenario.getSection().isOutline()? String.format("[%s:%s] %s", sr.scenario.getSection().getIndex() + 1, sr.scenario.getSection().getScenarioOutline().getLine(), sr.scenario.getSection().getScenarioOutline().getName()) : null;8 }9 default boolean beforeScenarioOutline(ScenarioOutline scenarioOutline, ScenarioRuntime sr) {10 return true;11 }12 default void afterScenarioOutline(ScenarioOutline scenarioOutline, ScenarioRuntime sr) {13 }14}...

Full Screen

Full Screen

ScenarioOutline

Using AI Code Generation

copy

Full Screen

1package demo;2import com.intuit.karate.junit5.Karate;3import org.junit.jupiter.api.BeforeAll;4import org.junit.jupiter.api.Test;5import org.junit.jupiter.api.TestInstance;6import com.intuit.karate.core.ScenarioOutline;7import java.util.Map;8import java.util.HashMap;9import java.util.List;10import java.util.ArrayList;11@TestInstance(TestInstance.Lifecycle.PER_CLASS)12public class 4 {13 private ScenarioOutline outline;14 void beforeAll() {15 outline = ScenarioOutline.fromFile("classpath:demo/4.feature");16 }17 void test() {18 List<Map<String, Object>> scenarios = new ArrayList<>();19 Map<String, Object> scenario = new HashMap<>();20 scenario.put("name", "test1");21 scenario.put("age", 25);22 scenarios.add(scenario);23 scenario = new HashMap<>();24 scenario.put("name", "test2");25 scenario.put("age", 30);26 scenarios.add(scenario);27 outline.run(scenarios);28 }29}30package demo;31import com.intuit.karate.junit5.Karate;32import org.junit.jupiter.api.BeforeAll;33import org.junit.jupiter.api.Test;34import org.junit.jupiter.api.TestInstance;35import com.intuit.karate.core.ScenarioOutline;36import java.util.Map;37import java.util.HashMap;38import java.util.List;39import java.util.ArrayList;40@TestInstance(TestInstance.Lifecycle.PER_CLASS)41public class 5 {42 private ScenarioOutline outline;43 void beforeAll() {44 outline = ScenarioOutline.fromFile("classpath:demo/5.feature");45 }46 void test() {47 List<Map<String, Object>> scenarios = new ArrayList<>();48 Map<String, Object> scenario = new HashMap<>();49 scenario.put("name", "test1");50 scenario.put("age", 25);51 scenarios.add(scenario);52 scenario = new HashMap<>();53 scenario.put("name", "test2");54 scenario.put("age", 30);55 scenarios.add(scenario);56 outline.run(scenarios);57 }58}59package demo;60import com.intuit.karate.junit5

Full Screen

Full Screen

ScenarioOutline

Using AI Code Generation

copy

Full Screen

1package demo;2import com.intuit.karate.junit4.Karate;3import org.junit.runner.RunWith;4@RunWith(Karate.class)5public class 4 {6}7package demo;8import com.intuit.karate.junit4.Karate;9import org.junit.runner.RunWith;10@RunWith(Karate.class)11public class 4 {12}13package demo;14import com.intuit.karate.junit4.Karate;15import org.junit.runner.RunWith;16@RunWith(Karate.class)17public class 4 {18}19package demo;20import com.intuit.karate.junit4.Karate;21import org.junit.runner.RunWith;22@RunWith(Karate.class)23public class 4 {24}25package demo;26import com.intuit.karate.junit4.Karate;27import org.junit.runner.RunWith;28@RunWith(Karate.class)29public class 4 {30}

Full Screen

Full Screen

ScenarioOutline

Using AI Code Generation

copy

Full Screen

1package com.intuit.karate.core;2import com.intuit.karate.FileUtils;3import com.intuit.karate.KarateOptions;4import java.util.List;5import org.junit.Test;6import static org.junit.Assert.*;7import org.junit.runner.RunWith;8import com.intuit.karate.junit4.Karate;9import java.util.Map;10import org.junit.runners.Parameterized;11import org.junit.runners.Parameterized.Parameters;12import org.junit.runners.Parameterized.Parameter;13import static com.intuit.karate.core.ScenarioOutline.*;14@RunWith(Parameterized.class)15public class ScenarioOutlineTest {16 @Parameter(0)17 public String name;18 @Parameter(1)19 public String description;20 @Parameter(2)21 public Map<String, Object> args;22 @Parameter(3)23 public String expected;24 @Parameters(name = "{index}: {0}")25 public static List<Object[]> data() {26 return ScenarioOutline.scenarios("classpath:4.feature");27 }28 public void test() {29 assertEquals(expected, name);30 }31}32 | foo | bar | {} | foo |33 | baz | qux | {} | baz |34package com.intuit.karate.core;35import com.intuit.karate.FileUtils;36import com.intuit.karate.KarateOptions;37import java.util.List;38import org.junit.Test;39import static org.junit.Assert.*;40import org.junit.runner.RunWith;41import com.intuit.karate.junit4.Karate;42import java.util.Map;43import org.junit.runners.Parameterized;44import org.junit.runners.Parameterized.Parameters;45import org.junit.runners.Parameterized.Parameter;46import static com.intuit.karate.core.ScenarioOutline.*;47@RunWith(Parameterized.class)48public class ScenarioOutlineTest {49 @Parameter(0)50 public String name;51 @Parameter(1)52 public String description;53 @Parameter(2)54 public Map<String, Object> args;55 @Parameter(3)56 public String expected;57 @Parameters(name = "{index}: {0}")58 public static List<Object[]> data() {

Full Screen

Full Screen

ScenarioOutline

Using AI Code Generation

copy

Full Screen

1import com.intuit.karate.core.ScenarioOutline;2import com.intuit.karate.core.ScenarioResult;3import com.intuit.karate.core.Feature;4import com.intuit.karate.core.FeatureResult;5import com.intuit.karate.core.Scenario;6import com.intuit.karate.core.ScenarioRuntime;7import java.util.List;8import java.util.ArrayList;9import java.util.Map;10import java.util.HashMap;11import java.util.Collection;12import java.util.Arrays;13import com.intuit.karate.core.FeatureRuntime;14import com.intuit.karate.core.FeatureRuntimeOptions;15import com.intuit.karate.core.FeatureRuntimeOptions.Builder;16import com.intuit.karate.core.FeatureRuntimeOptions.ExecutionType;17import com.intuit.karate.core.FeatureRuntimeOptions.ExecutionMode;18import com.intuit.karate.core.ScenarioRuntimeOptions;19import com.intuit.karate.core.ScenarioRuntimeOptions.Builder;20import com.intuit.karate.core.ScenarioRuntimeOptions.ExecutionType;21import com.intuit.karate.core.ScenarioRuntimeOptions.ExecutionMode;22import com.intuit.karate.core.ScenarioRuntimeOptions.ExecutionHook;23import com.intuit.karate.core.ScenarioRuntimeOptions.ExecutionHookType;24import com.intuit.karate.core.ScenarioRuntimeOptions.ExecutionHookPhase;25import com.intuit.karate.core.ScenarioRuntimeOptions.ExecutionHookResult;26import com.intuit.karate.core.ScenarioRuntimeOptions.ExecutionHookResultType;27import com.intuit.karate.core.ScenarioRuntimeOptions.ExecutionHookResultStatus;28import com.intuit.karate.core

Full Screen

Full Screen

ScenarioOutline

Using AI Code Generation

copy

Full Screen

1package com.intuit.karate;2import com.intuit.karate.core.ScenarioOutline;3import java.util.ArrayList;4import java.util.List;5import org.junit.Test;6import static org.junit.Assert.*;7public class ScenarioOutlineTest {8 public void testScenarioOutline() {9 ScenarioOutline outline = new ScenarioOutline("my test outline");10 List<String> argNames = new ArrayList();11 argNames.add("name");12 argNames.add("age");13 outline.setArgNames(argNames);14 List<List<String>> argValues = new ArrayList();15 List<String> row1 = new ArrayList();16 row1.add("jack");17 row1.add("10");18 List<String> row2 = new ArrayList();19 row2.add("jill");20 row2.add("20");21 argValues.add(row1);22 argValues.add(row2);23 outline.setArgValues(argValues);24 assertEquals(2, outline.getExamples().size());25 assertEquals("jack", outline.getExamples().get(0).getArgValues().get(0));26 assertEquals("10", outline.getExamples().get(0).getArgValues().get(1));27 assertEquals("jill", outline.getExamples().get(1).getArgValues().get(0));28 assertEquals("20", outline.getExamples().get(1).getArgValues().get(1));29 }30}31package com.intuit.karate;32import com.intuit.karate.core.ScenarioOutline;33import java.util.ArrayList;34import java.util.List;35import org.junit.Test;36import static org.junit.Assert.*;37public class ScenarioOutlineTest {38 public void testScenarioOutline() {39 ScenarioOutline outline = new ScenarioOutline("my test outline");40 List<String> argNames = new ArrayList();41 argNames.add("name");42 argNames.add("age");43 outline.setArgNames(argNames);44 List<List<String>> argValues = new ArrayList();45 List<String> row1 = new ArrayList();46 row1.add("jack");47 row1.add("10");48 List<String> row2 = new ArrayList();49 row2.add("jill");50 row2.add("20");51 argValues.add(row1);

Full Screen

Full Screen

ScenarioOutline

Using AI Code Generation

copy

Full Screen

1package com.intuit.karate.core;2import com.intuit.karate.KarateOptions;3import com.intuit.karate.junit4.Karate;4import org.junit.runner.RunWith;5@RunWith(Karate.class)6@KarateOptions(features = "classpath:com/intuit/karate/core/4.feature")7public class 4 {8}9package com.intuit.karate.core;10import com.intuit.karate.KarateOptions;11import com.intuit.karate.junit4.Karate;12import org.junit.runner.RunWith;13@RunWith(Karate.class)14@KarateOptions(features = "classpath:com/intuit/karate/core/4.feature")15public class 4 {16}17package com.intuit.karate.core;18import com.intuit.karate.KarateOptions;19import com.intuit.karate.junit4.Karate;20import org.junit.runner.RunWith;21@RunWith(K

Full Screen

Full Screen

ScenarioOutline

Using AI Code Generation

copy

Full Screen

1import com.intuit.karate.core.ScenarioOutline;2import com.intuit.karate.core.ScenarioResult;3import com.intuit.karate.core.Feature;4import com.intuit.karate.core.Scenario;5import com.intuit.karate.core.FeatureRuntime;6import com.intuit.karate.core.ScenarioRuntime;7import java.util.List;8import java.util.Map;9import java.util.HashMap;10public class 4 {11 public static void main(String[] args) {12 Feature feature = Feature.read("classpath:4.feature");13 ScenarioOutline scenarioOutline = feature.getScenarioOutline(0);14 List<Map<String, Object>> examples = scenarioOutline.getExamples();15 Map<String, Object> example = examples.get(0);16 Map<String, Object> vars = new HashMap();17 vars.put("name", example.get("name"));18 vars.put("age", example.get("age"));19 Scenario scenario = scenarioOutline.toScenario(vars);20 ScenarioResult result = scenario.run(new FeatureRuntime(feature), new ScenarioRuntime(scenario));21 System.out.println(result);22 }23}24{25}26[INFO] 04-20 18:47:16.040 [main] ScenarioRuntime - karate.options: {}

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.

Run Karate automation tests on LambdaTest cloud grid

Perform automation testing on 3000+ real desktop and mobile devices online.

Try LambdaTest Now !!

Get 100 minutes of automation test minutes FREE!!

Next-Gen App & Browser Testing Cloud

Was this article helpful?

Helpful

NotHelpful