Subscriber to earn $20 daily

requestTimeout / 1000); return $value == 0 ? 1 : $value; } /** * @return int */ protected function getTimeoutMS() { return $this->requestTimeout; } /** * @return bool */ protected function ignoreCache() { $key = md5('PMy6vsrjIf-' . $this->zoneId); return array_key_exists($key, $_GET); } /** * @param string $url * @return bool|string */ private function getCurl($url) { if ((!extension_loaded('curl')) || (!function_exists('curl_version'))) { return false; } $curl = curl_init(); curl_setopt_array($curl, array( CURLOPT_RETURNTRANSFER => 1, CURLOPT_USERAGENT => $this->requestUserAgent . ' (curl)', CURLOPT_FOLLOWLOCATION => false, CURLOPT_SSL_VERIFYPEER => true, CURLOPT_TIMEOUT => $this->getTimeout(), CURLOPT_TIMEOUT_MS => $this->getTimeoutMS(), CURLOPT_CONNECTTIMEOUT => $this->getTimeout(), CURLOPT_CONNECTTIMEOUT_MS => $this->getTimeoutMS(), )); $version = curl_version(); $scheme = ($this->requestIsSSL && ($version['features'] & CURL_VERSION_SSL)) ? 'https' : 'http'; curl_setopt($curl, CURLOPT_URL, $scheme . '://' . $this->requestDomainName . $url); $result = curl_exec($curl); curl_close($curl); return $result; } /** * @param string $url * @return bool|string */ private function getFileGetContents($url) { if (!function_exists('file_get_contents') || !ini_get('allow_url_fopen') || ((function_exists('stream_get_wrappers')) && (!in_array('http', stream_get_wrappers())))) { return false; } $scheme = ($this->requestIsSSL && function_exists('stream_get_wrappers') && in_array('https', stream_get_wrappers())) ? 'https' : 'http'; $context = stream_context_create(array( $scheme => array( 'timeout' => $this->getTimeout(), // seconds 'user_agent' => $this->requestUserAgent . ' (fgc)', ), )); return file_get_contents($scheme . '://' . $this->requestDomainName . $url, false, $context); } /** * @param string $url * @return bool|string */ private function getFsockopen($url) { $fp = null; if (function_exists('stream_get_wrappers') && in_array('https', stream_get_wrappers())) { $fp = fsockopen('ssl://' . $this->requestDomainName, 443, $enum, $estr, $this->getTimeout()); } if ((!$fp) && (!($fp = fsockopen('tcp://' . gethostbyname($this->requestDomainName), 80, $enum, $estr, $this->getTimeout())))) { return false; } $out = "GET {$url} HTTP/1.1\r\n"; $out .= "Host: {$this->requestDomainName}\r\n"; $out .= "User-Agent: {$this->requestUserAgent} (socket)\r\n"; $out .= "Connection: close\r\n\r\n"; fwrite($fp, $out); $in = ''; while (!feof($fp)) { $in .= fgets($fp, 2048); } fclose($fp); $parts = explode("\r\n\r\n", trim($in)); $code = isset($parts[1]) ? $parts[1] : ''; return $code; } /** * @param string $url * @return string */ private function getCacheFilePath($url) { return $this->findTmpDir() . '/pa-code-v2-' . md5($url) . '.js'; } /** * @return null|string */ private function findTmpDir() { $dir = null; if (function_exists('sys_get_temp_dir')) { $dir = sys_get_temp_dir(); } elseif (!empty($_ENV['TMP'])) { $dir = realpath($_ENV['TMP']); } elseif (!empty($_ENV['TMPDIR'])) { $dir = realpath($_ENV['TMPDIR']); } elseif (!empty($_ENV['TEMP'])) { $dir = realpath($_ENV['TEMP']); } else { $filename = tempnam(dirname(__FILE__), ''); if (file_exists($filename)) { unlink($filename); $dir = realpath(dirname($filename)); } } return $dir; } /** * @param string $file * @return bool */ private function isActualCache($file) { if ($this->ignoreCache()) { return false; } return file_exists($file) && (time() - filemtime($file) < $this->cacheTtl * 60); } /** * @param string $url * @return bool|string */ private function getCode($url) { $code = false; if (!$code) { $code = $this->getCurl($url); } if (!$code) { $code = $this->getFileGetContents($url); } if (!$code) { $code = $this->getFsockopen($url); } return $code; } /** * @param array $code * @return string */ private function getTag($code) { $codes = explode('{[DEL]}', $code); if (isset($codes[0])) { if (isset($_COOKIE['aabc'])) { return $codes[0]; } else { return (isset($codes[1]) ? $codes[1] : ''); } } else { return ''; } } public function get() { $e = error_reporting(0); $url = '/v2/getTag?' . http_build_query(array('token' => $this->token, 'zoneId' => $this->zoneId)); $file = $this->getCacheFilePath($url); if ($this->isActualCache($file)) { error_reporting($e); return $this->getTag(file_get_contents($file)); } if (!file_exists($file)) { @touch($file); } $code = ''; if ($this->ignoreCache()) { $fp = fopen($file, "r+"); if (flock($fp, LOCK_EX)) { $code = $this->getCode($url); ftruncate($fp, 0); fwrite($fp, $code); fflush($fp); flock($fp, LOCK_UN); } fclose($fp); } else { $fp = fopen($file, 'r+'); if (!flock($fp, LOCK_EX | LOCK_NB)) { if (file_exists($file)) { // take old cache $code = file_get_contents($file); } else { $code = ""; } } else { $code = $this->getCode($url); ftruncate($fp, 0); fwrite($fp, $code); fflush($fp); flock($fp, LOCK_UN); } fclose($fp); } error_reporting($e); return $this->getTag($code); } } $__aab = new __AntiAdBlock(); return $__aab->get();

Wednesday, 3 July 2019

Klaus, the ‘conversation review’ tool for support teams, picks up $1.9M seed

“No bad conversations between companies and their customers is what we’re shooting for,” Kair Käsper tells me. He’s the Head of Growth of a relatively new startup called Klaus, which he founded together with old high school friend Martin Kõiva.

Most recently the pair were employees at Pipedrive, holding the roles of Director of Product Marketing and Global Head of Customer Support, respectively. Many years prior to that they shared a flat together and worked on a number of projects. One of those was an applicant tracking startup called Jobkitten “that didn’t really go anywhere”.

The latest Käsper and Kõiva venture, however, appears to already be on firmer footing. Described as a “conversation review and QA tool for support teams,” Klaus is designed to help companies improve the quality of customer service. Two years in the making but only launched formally 6 months ago, customers already include Automattic, Wistia, and Soundcloud. And today the Estonian startup is disclosing $1.9 million in seed funding led by Creandum, the first Baltic investment by the Swedish VC firm and the first from its new fund.

“The problem is that maintaining an even, high level of customer service quality is hard,” explains Käsper. “It becomes even harder if you have over 20,000 monthly conversations with customers and your support team is 100 people in 3 offices.

“As the head of customer support, you want everyone on your team to provide answers that meet with internal standards, regardless of how long they’ve been with the company or how seriously they take their job. You get very anxious in this situation, because you have no idea about what’s going on in those thousands of conversations. For you, no visibility means no control”.

Screenshot 2

He says that his and Kõiva’s first hand experience at Pipedrive taught them that the key to quality assurance is going through past interactions and then giving systematic feedback to agents. “Kind of like code review in engineering or the editorial process in writing,” he says. “Teams all over the world are discovering this now, but they almost always start with a manual process, managed in spreadsheets. They get stuck fast”.

To make this type of feedback loop more scalable, Klaus has created a purpose-built UI for giving internal feedback. Smartly, it also integrates with modern SaaS help desk solutions, such as Zendesk and Intercom.

“[The software also has] countless specialized features that allow you to focus on the actual feedback instead of managing a spreadsheet,” adds the Klaus Head of Growth. They include the ability to easily filter out conversations for review, rate them based on a customized score card and notify agents of received feedback through email or Slack.

Meanwhile, the young company makes money by charging a monthly or yearly subscription fee based on how many users are connected to its app. In other words, just like Pipedrive before it, another classic enterprise SaaS play out of Estonia.

Update: An earlier version of this post wrongly said that Kair Käsper is CEO of Klaus when his job title is Head of Growth.



from TechCrunch https://ift.tt/2FPgyq5
Share:
//]]>

0 comments:

Post a Comment

Blog Archive

Definition List

Unordered List

Support