mirror of
https://github.com/shlinkio/shlink-web-client.git
synced 2025-01-10 02:07:26 +03:00
Refactored visits parser to be functional
This commit is contained in:
parent
4924f24e1a
commit
6d0753481d
3 changed files with 74 additions and 60 deletions
|
@ -7,13 +7,21 @@ import Moment from 'react-moment'
|
||||||
import { connect } from 'react-redux'
|
import { connect } from 'react-redux'
|
||||||
import { Card, CardBody, CardHeader, UncontrolledTooltip } from 'reactstrap'
|
import { Card, CardBody, CardHeader, UncontrolledTooltip } from 'reactstrap'
|
||||||
import DateInput from '../common/DateInput'
|
import DateInput from '../common/DateInput'
|
||||||
import visitsParser from '../visits/services/VisitsParser'
|
import {
|
||||||
|
processOsStats,
|
||||||
|
processBrowserStats,
|
||||||
|
processCountriesStats,
|
||||||
|
processReferrersStats,
|
||||||
|
} from '../visits/services/VisitsParser'
|
||||||
import { getShortUrlVisits } from './reducers/shortUrlVisits'
|
import { getShortUrlVisits } from './reducers/shortUrlVisits'
|
||||||
import './ShortUrlVisits.scss'
|
import './ShortUrlVisits.scss'
|
||||||
import MutedMessage from '../utils/MuttedMessage';
|
import MutedMessage from '../utils/MuttedMessage';
|
||||||
|
|
||||||
const defaultProps = {
|
const defaultProps = {
|
||||||
visitsParser,
|
processOsStats,
|
||||||
|
processBrowserStats,
|
||||||
|
processCountriesStats,
|
||||||
|
processReferrersStats,
|
||||||
};
|
};
|
||||||
|
|
||||||
export class ShortUrlsVisits extends React.Component {
|
export class ShortUrlsVisits extends React.Component {
|
||||||
|
@ -34,7 +42,10 @@ export class ShortUrlsVisits extends React.Component {
|
||||||
const {
|
const {
|
||||||
match: { params },
|
match: { params },
|
||||||
selectedServer,
|
selectedServer,
|
||||||
visitsParser,
|
processOsStats,
|
||||||
|
processBrowserStats,
|
||||||
|
processCountriesStats,
|
||||||
|
processReferrersStats,
|
||||||
shortUrlVisits: { visits, loading, error, shortUrl }
|
shortUrlVisits: { visits, loading, error, shortUrl }
|
||||||
} = this.props;
|
} = this.props;
|
||||||
const serverUrl = selectedServer ? selectedServer.url : '';
|
const serverUrl = selectedServer ? selectedServer.url : '';
|
||||||
|
@ -96,10 +107,10 @@ export class ShortUrlsVisits extends React.Component {
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="row">
|
<div className="row">
|
||||||
{renderGraphCard('Operating systems', visitsParser.processOsStats(visits), false)}
|
{renderGraphCard('Operating systems', processOsStats(visits), false)}
|
||||||
{renderGraphCard('Browsers', visitsParser.processBrowserStats(visits), false)}
|
{renderGraphCard('Browsers', processBrowserStats(visits), false)}
|
||||||
{renderGraphCard('Countries', visitsParser.processCountriesStats(visits), true, 'Visits')}
|
{renderGraphCard('Countries', processCountriesStats(visits), true, 'Visits')}
|
||||||
{renderGraphCard('Referrers', visitsParser.processReferrersStats(visits), true, 'Visits')}
|
{renderGraphCard('Referrers', processReferrersStats(visits), true, 'Visits')}
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
@ -173,4 +184,7 @@ export class ShortUrlsVisits extends React.Component {
|
||||||
|
|
||||||
ShortUrlsVisits.defaultProps = defaultProps;
|
ShortUrlsVisits.defaultProps = defaultProps;
|
||||||
|
|
||||||
export default connect(pick(['selectedServer', 'shortUrlVisits']), { getShortUrlVisits })(ShortUrlsVisits);
|
export default connect(
|
||||||
|
pick(['selectedServer', 'shortUrlVisits']),
|
||||||
|
{ getShortUrlVisits }
|
||||||
|
)(ShortUrlsVisits);
|
||||||
|
|
|
@ -41,53 +41,48 @@ const extractDomain = url => {
|
||||||
return domain.split(':')[0];
|
return domain.split(':')[0];
|
||||||
};
|
};
|
||||||
|
|
||||||
export class VisitsParser {
|
export const processOsStats = visits =>
|
||||||
processOsStats = visits =>
|
reduce(
|
||||||
reduce(
|
(stats, visit) => {
|
||||||
(stats, visit) => {
|
const userAgent = visit.userAgent;
|
||||||
const userAgent = visit.userAgent;
|
const os = isNil(userAgent) ? 'Others' : osFromUserAgent(userAgent);
|
||||||
const os = isNil(userAgent) ? 'Others' : osFromUserAgent(userAgent);
|
return assoc(os, (stats[os] || 0) + 1, stats);
|
||||||
return assoc(os, (stats[os] || 0) + 1, stats);
|
},
|
||||||
},
|
{},
|
||||||
{},
|
visits,
|
||||||
visits,
|
);
|
||||||
);
|
|
||||||
|
|
||||||
processBrowserStats = visits =>
|
export const processBrowserStats = visits =>
|
||||||
reduce(
|
reduce(
|
||||||
(stats, visit) => {
|
(stats, visit) => {
|
||||||
const userAgent = visit.userAgent;
|
const userAgent = visit.userAgent;
|
||||||
const browser = isNil(userAgent) ? 'Others' : browserFromUserAgent(userAgent);
|
const browser = isNil(userAgent) ? 'Others' : browserFromUserAgent(userAgent);
|
||||||
return assoc(browser, (stats[browser] || 0) + 1, stats);
|
return assoc(browser, (stats[browser] || 0) + 1, stats);
|
||||||
},
|
},
|
||||||
{},
|
{},
|
||||||
visits,
|
visits,
|
||||||
);
|
);
|
||||||
|
|
||||||
processReferrersStats = visits =>
|
export const processReferrersStats = visits =>
|
||||||
reduce(
|
reduce(
|
||||||
(stats, visit) => {
|
(stats, visit) => {
|
||||||
const notHasDomain = isNil(visit.referer) || isEmpty(visit.referer);
|
const notHasDomain = isNil(visit.referer) || isEmpty(visit.referer);
|
||||||
const domain = notHasDomain ? 'Unknown' : extractDomain(visit.referer);
|
const domain = notHasDomain ? 'Unknown' : extractDomain(visit.referer);
|
||||||
return assoc(domain, (stats[domain]|| 0) + 1, stats);
|
return assoc(domain, (stats[domain]|| 0) + 1, stats);
|
||||||
},
|
},
|
||||||
{},
|
{},
|
||||||
visits,
|
visits,
|
||||||
);
|
);
|
||||||
|
|
||||||
processCountriesStats = visits =>
|
export const processCountriesStats = visits =>
|
||||||
reduce(
|
reduce(
|
||||||
(stats, { visitLocation }) => {
|
(stats, { visitLocation }) => {
|
||||||
const notHasCountry = isNil(visitLocation)
|
const notHasCountry = isNil(visitLocation)
|
||||||
|| isNil(visitLocation.countryName)
|
|| isNil(visitLocation.countryName)
|
||||||
|| isEmpty(visitLocation.countryName);
|
|| isEmpty(visitLocation.countryName);
|
||||||
const country = notHasCountry ? 'Unknown' : visitLocation.countryName;
|
const country = notHasCountry ? 'Unknown' : visitLocation.countryName;
|
||||||
return assoc(country, (stats[country]|| 0) + 1, stats);
|
return assoc(country, (stats[country]|| 0) + 1, stats);
|
||||||
},
|
},
|
||||||
{},
|
{},
|
||||||
visits,
|
visits,
|
||||||
);
|
);
|
||||||
}
|
|
||||||
|
|
||||||
const visitsParser = new VisitsParser();
|
|
||||||
export default visitsParser;
|
|
||||||
|
|
|
@ -1,4 +1,9 @@
|
||||||
import visitsParser from '../../../src/visits/services/VisitsParser';
|
import {
|
||||||
|
processOsStats,
|
||||||
|
processBrowserStats,
|
||||||
|
processReferrersStats,
|
||||||
|
processCountriesStats,
|
||||||
|
} from '../../../src/visits/services/VisitsParser';
|
||||||
|
|
||||||
describe('VisitsParser', () => {
|
describe('VisitsParser', () => {
|
||||||
const visits = [
|
const visits = [
|
||||||
|
@ -36,7 +41,7 @@ describe('VisitsParser', () => {
|
||||||
|
|
||||||
describe('processOsStats', () => {
|
describe('processOsStats', () => {
|
||||||
it('properly parses OS stats', () => {
|
it('properly parses OS stats', () => {
|
||||||
expect(visitsParser.processOsStats(visits)).toEqual({
|
expect(processOsStats(visits)).toEqual({
|
||||||
'Linux': 3,
|
'Linux': 3,
|
||||||
'Windows': 1,
|
'Windows': 1,
|
||||||
'MacOS': 1,
|
'MacOS': 1,
|
||||||
|
@ -46,7 +51,7 @@ describe('VisitsParser', () => {
|
||||||
|
|
||||||
describe('processBrowserStats', () => {
|
describe('processBrowserStats', () => {
|
||||||
it('properly parses browser stats', () => {
|
it('properly parses browser stats', () => {
|
||||||
expect(visitsParser.processBrowserStats(visits)).toEqual({
|
expect(processBrowserStats(visits)).toEqual({
|
||||||
'Firefox': 2,
|
'Firefox': 2,
|
||||||
'Chrome': 2,
|
'Chrome': 2,
|
||||||
'Opera': 1,
|
'Opera': 1,
|
||||||
|
@ -56,7 +61,7 @@ describe('VisitsParser', () => {
|
||||||
|
|
||||||
describe('processReferrersStats', () => {
|
describe('processReferrersStats', () => {
|
||||||
it('properly parses referrer stats', () => {
|
it('properly parses referrer stats', () => {
|
||||||
expect(visitsParser.processReferrersStats(visits)).toEqual({
|
expect(processReferrersStats(visits)).toEqual({
|
||||||
'Unknown': 2,
|
'Unknown': 2,
|
||||||
'google.com': 2,
|
'google.com': 2,
|
||||||
'm.facebook.com': 1,
|
'm.facebook.com': 1,
|
||||||
|
@ -66,7 +71,7 @@ describe('VisitsParser', () => {
|
||||||
|
|
||||||
describe('processCountriesStats', () => {
|
describe('processCountriesStats', () => {
|
||||||
it('properly parses countries stats', () => {
|
it('properly parses countries stats', () => {
|
||||||
expect(visitsParser.processCountriesStats(visits)).toEqual({
|
expect(processCountriesStats(visits)).toEqual({
|
||||||
'Spain': 3,
|
'Spain': 3,
|
||||||
'United States': 1,
|
'United States': 1,
|
||||||
'Unknown': 1,
|
'Unknown': 1,
|
||||||
|
|
Loading…
Reference in a new issue